使用 mongoose hook 重试保存重复键错误
Use mongoose hook to retry saving on duplicate key error
我想使用 mongoose 中间件挂钩重新尝试保存文档,以防初始保存因重复键错误而失败。使用案例如下:
我的模型使用自动生成的 slug 进行识别。例如。 item
、item-2
、item-3
、…、item-n
。如果 item
已经存在,则应向 slug 添加一个计数器。我无法提前检查 "next" slug,因为我需要在任何情况下都避免冲突。
由于这个逻辑会涉及到几个不同的模型,所以我想把它隔离成一个mongoose插件。
这个想法是否可行?例如。 schema.post('save', handler)
仅在成功保存时执行。还有其他我可以利用的钩子吗?
我正在使用预保存挂钩来检查 slug 是否已经存在并使用以下代码,到目前为止它似乎工作正常。
pageSchema.pre('save', function(next){
var page = this;
page.createdDT = new Date();
page.updatedDT = new Date();
page.slug = page.title.slug(); // create slug from title
var re = new RegExp(page.slug, 'i');
mongoose.models["page"].find({slug: { $regex: re}}, function(err, pages){
// slug doesn't exist, good to go
if(!err && !pages) return next();
var slugs = [];
// let's get all slugs
pages.forEach(function(page){
slugs.push(page.slug);
});
// keep increasing `i` until slug is unique
// set i to 1 to avoid hello-0
var i = 1;
var tempSlug = page.slug;
while(slugs.indexOf(tempSlug) >= 0){
tempSlug = page.slug + '-' + i;
i++;
}
//unique slug for example "hello-2"
page.slug = tempSlug;
next();
});
});
我最终采用了 mongoose-uniqueslugs
使用的解决方案,我根据我们的需要对其进行了调整。虽然此变体不仅仅适用于 pre/post 钩子,但它确保了原子性(即不提前检查可用的 slug 然后保存,而只是重新尝试)。
中心思想是覆盖模型的 save
函数(参见 enhanceModel
函数)并提供一个包装器,它通过 slug 碰撞捕获独特的错误,然后重新尝试保存(而不是附加的随机字符串,我们想要一个序列号方法)。
我想使用 mongoose 中间件挂钩重新尝试保存文档,以防初始保存因重复键错误而失败。使用案例如下:
我的模型使用自动生成的 slug 进行识别。例如。 item
、item-2
、item-3
、…、item-n
。如果 item
已经存在,则应向 slug 添加一个计数器。我无法提前检查 "next" slug,因为我需要在任何情况下都避免冲突。
由于这个逻辑会涉及到几个不同的模型,所以我想把它隔离成一个mongoose插件。
这个想法是否可行?例如。 schema.post('save', handler)
仅在成功保存时执行。还有其他我可以利用的钩子吗?
我正在使用预保存挂钩来检查 slug 是否已经存在并使用以下代码,到目前为止它似乎工作正常。
pageSchema.pre('save', function(next){
var page = this;
page.createdDT = new Date();
page.updatedDT = new Date();
page.slug = page.title.slug(); // create slug from title
var re = new RegExp(page.slug, 'i');
mongoose.models["page"].find({slug: { $regex: re}}, function(err, pages){
// slug doesn't exist, good to go
if(!err && !pages) return next();
var slugs = [];
// let's get all slugs
pages.forEach(function(page){
slugs.push(page.slug);
});
// keep increasing `i` until slug is unique
// set i to 1 to avoid hello-0
var i = 1;
var tempSlug = page.slug;
while(slugs.indexOf(tempSlug) >= 0){
tempSlug = page.slug + '-' + i;
i++;
}
//unique slug for example "hello-2"
page.slug = tempSlug;
next();
});
});
我最终采用了 mongoose-uniqueslugs
使用的解决方案,我根据我们的需要对其进行了调整。虽然此变体不仅仅适用于 pre/post 钩子,但它确保了原子性(即不提前检查可用的 slug 然后保存,而只是重新尝试)。
中心思想是覆盖模型的 save
函数(参见 enhanceModel
函数)并提供一个包装器,它通过 slug 碰撞捕获独特的错误,然后重新尝试保存(而不是附加的随机字符串,我们想要一个序列号方法)。