我是mongoose / express的新手。我正在努力的是尝试从html表单更新新数据并将其保存到具有引用的数据库。我有一个商业模式,其位置参考位置模型。这是代码。
edit.ejs
<div class="container">
<div class="form-container">
<form action="/<%= bus._id %>?_method=PUT" method="POST">
<!-- business info -->
<h3>Business Information</h3>
<input class="form-input" type="input" name="bus[logo]" value="<%= bus.logo %>">
<input class="form-input" type="input" name="bus[name]" value="<%= bus.name %>">
<% bus.location.forEach(function(location) { %>
<input class="form-input" type="input" name="bus.location[street]" value="<%= location.street %>">
<input class="form-input" type="input" name="bus.location[city]" value="<%= location.city %>">
<div class="state-input">
<select class="form-inline" name="bus.location[state]">
<option value="" disabled selected><%= location.state %></option>
<option value="AL">Alabama</option>
...
<option value="WY">Wyoming</option>
</select>
</div><!--State-input -->
<input class="form-inline" type="input" name="bus.location[zipcode]" value="<%= location.zipcode %>">
<% }) %>
<!--Contact info-->
<h4>Contact Information</h4>
<% bus.contact.forEach(function(contact) { %>
<input class="form-input" type="url" name="bus[url]" value="<%= bus.url %>">
<input class="form-input" type="email" name="bus.contact[email]" value="<%= contact.email %>">
<input class="form-input" type="tel" name="bus.contact[phone]" value="<%= contact.phone %>">
<input class="form-input" type="input" name="bus.contact[twitter]" value= "<%= contact.twitter %>">
<input class="form-input" type="input" name="bus.contact[facebook]" value="<%= contact.facebook %>">
<input class="form-input" type="input" name="bus.contact[instagram]" value="<%= contact.instagram %>">
<% }) %>
index.js - 编辑路线
//(edit.ejs) Edit Route
app.get('/:id/edit', function(req, res) {
Business.findById(req.params.id)
.populate('location')
.populate('contact')
.populate('images')
.exec(function(err, bus) {
if(err) {
console.log(err);
} else {
res.render('edit', {bus:bus});
}
});
});
app.put('/:id', function(req, res) {
Business.findByIdAndUpdate(req.params.id, req.body.bus, function(err, bus) {
if(err) {
console.log(err);
res.redirect('/' + req.params.id + '/edit');
} else {
res.redirect('/' + req.params.id);
}
});
});
业务(总线)更新很好但bus.location不更新。 商业模式
//----------------------------------------------------------------------------\\
var mongoose = require('mongoose');
//----------------------------------------------------------------------------\\
var busSchema = new mongoose.Schema({
name: String,
type: String,
logo: String,
desc: String,
subs: Number,
video: String,
url: String,
firstRun: Boolean,
location:[
{
type: mongoose.Schema.Types.ObjectId,
ref: 'Location'
}
],
contact:[
{
type: mongoose.Schema.Types.ObjectId,
ref: 'Contact'
}
],
images:[
{
type: mongoose.Schema.Types.ObjectId,
ref: 'Image'
}
],
comments:[
{
type: mongoose.Schema.Types.ObjectId,
ref: 'Comment'
}
],
created: {
type: Date, default: Date.now
}
});
//----------------------------------------------------------------------------\\
module.exports = mongoose.model('Business', busSchema);
位置模型
//----------------------------------------------------------------------------\\
var mongoose = require('mongoose');
//----------------------------------------------------------------------------\\
var locSchema = new mongoose.Schema(
{
street: String,
city: String,
state: String,
zipcode: Number
}
);
//----------------------------------------------------------------------------\\
module.exports = mongoose.model('Location', locSchema);
答案 0 :(得分:0)
Business
,Location
和Contact
是不同的集合。
findByIdAndUpdate
仅更新一个集合,尤其是Business
。为了更新其他集合,您需要对这些集合执行操作。
如果您尝试更新“现有”位置和联系人,则还需要提供其ID。此外,您还有一个位置和联系人“阵列”,因此您需要额外[]
个名称。
<% bus.location.forEach(function(location, i) { %>
<input type="hidden" name="location[<?= i ?>][id]" value="<%= location.id %>">
<input class="form-input" type="input" name="location[<?= i ?>][street]" value="<%= location.street %>">
<!-- etc -->
<% }) %>
<% bus.contact.forEach(function(contact, i) { %>
<input type="hidden" name="contact[<?= i ?>][id]" value="<%= contact.id %>">
<input class="form-input" type="email" name="contact[<?= i ?>][email]" value="<%= contact.email %>">
<!-- etc -->
<% }) %>
从我的(和others)经验来看,我认为不可能一次更新多个文档。这意味着,在您的路由处理程序中,您将需要迭代给定的数组中的每个项目并逐个更新它们。另一篇文章中的答案虽然不太合适IMO,因为你不应该使用同步forEach
来执行异步猫鼬操作,因为它会导致意外行为。
您有三项主要任务要做:更新业务,更新现有位置以及更新现有联系人。
我喜欢使用async.js来执行多个异步操作。在您的特定情况下,我将使用async.series执行每项任务,并使用async.eachSeries对数组的每个元素执行操作。
警告这是未经测试但看起来像这样:
app.put('/:id', function(req, res) {
console.log(req.body.bus);
console.log(req.body.location); // should be an array of objects
console.log(req.body.contact); // should be an array of objects
// perform each task one by one
async.series([
function updateBusiness (done) {
// you need to always call the callback i.e. done to indicate the task is "done"
// - if you pass an error as an argument, it means the task failed and stop everything
// - otherwise, move onto the next task
/*Business.findByIdAndUpdate(req.params.id, req.body.bus, function (err) {
if (err) {
return done(err); // task failed and stop everything
}
done(); // task went well and proceed to the next task
});*/
// simplified
Business.findByIdAndUpdate(req.params.id, req.body.bus, done);
},
function updateLocations (done) {
// find and update each location
async.eachSeries(req.body.location || [], function (location, done2) {
Location.findByIdAndUpdate(location.id, location, done2);
}, done);
},
function updateContacts (done) {
// find and update each contact
async.eachSeries(req.body.contact || [], function (contact, done2) {
Contact.findByIdAndUpdate(contact.id, contact, done2);
}, done);
}
], function allDone (err) {
// a task failed somewhere
if (err) {
console.log(err);
res.redirect('/' + req.params.id + '/edit');
// all tasks were completed
} else {
res.redirect('/' + req.params.id);
}
});
});
如果出现错误,请查看控制台日志。