c# - RavenDB:PatchRequest 更新/合并文档中可能不存在的属性
问题描述
我有一个包含以下域模型的文档:
class Entity
{
...
Dictionary<string, string> Settings { get; set; }
...
}
并且需要更新指定的集合。但不能覆盖-与传入的更新合并。由于我需要以这种方式处理数千个文档,因此我选择了PatchCommand以获得更好的性能。得到以下信息:
new PatchCommandData
{
Key = upd.EntityId,
Patches = new[]
{
// Foreach incoming Setting remove existing value (if any) and add the new one
new PatchRequest
{
Type = PatchCommandType.Modify,
Name = nameof(Entity.Settings),
Nested = upd.UpdatedSettings.Keys
.Select(x => new PatchRequest
{
Type = PatchCommandType.Unset,
Name = x
})
.ToArray()
.Union(upd.UpdatedSettings.Keys
.Select(x => new PatchRequest
{
Type = PatchCommandType.Set,
Name = x,
Value = upd.UpdatedSettings[x]
})
.ToList())
.ToArray(),
Value = RavenJToken.FromObject(upd.UpdatedSettings)
}
}
}
这样执行下一次更新:
Before: { Setting1 = "Value1", Setting2 = "Value2" }
Update request: { Setting2 = "NewValue2", Setting3 = "Value3" }
After: { Setting1 = "Value1", Setting2 = "NewValue2", Setting3 = "Value3" }
但是..总有一个“但是”。如果db 中有一个没有 Settings 属性的文档,则提供的补丁会引发一个错误,提示“无法从 Settings 修改值,因为它没有找到”。
我找不到任何选项可以将补丁模式切换为 Set vs. Modify on the fly。并且没有加载所有文档的选项,在应用程序端应用更新并更新数千个文档。我能看到的唯一合理的选择是在类构造函数中为 Settings 属性创建 Dictionary 实例。
各位,能不能给点其他的建议?
PS RavenDB 版本限制为 3.5
解决方案
完毕。您可以在下面找到最终版本。实际上,使用了 ScriptedPatchCommandData。它包含 JavaScript 主体,Raven 将对文档进行评估和执行。从性能的角度来看,它可能不是最佳选择。但事实证明,没有什么重要的。希望它可能对某人有所帮助。
const string fieldParameterName = "fieldName";
const string valueParameterName = "value";
var patch = updates
.Select(update => new ScriptedPatchCommandData
{
Key = update.EntityId,
Patch = new ScriptedPatchRequest
{
Script = $"this[{fieldParameterName}] = _.assign({{}}, this[{fieldParameterName}], JSON.parse({valueParameterName}));",
Values = new Dictionary<string, object>
{
{
fieldParameterName,
nameof(Entity.Settings)
},
{
valueParameterName,
JsonConvert.SerializeObject(update.UpdatedSettings)
}
}
}
})
.ToList();
return await DocumentStore.AsyncDatabaseCommands.BatchAsync(patch);
推荐阅读
- javascript - 导出动态导入
- php - 如何在准备好的语句中强制将 int 作为字符串
- powershell - 尝试使用 PowerShell 发送电子邮件 (gmail)
- python - 无法找出打印语句的问题
- html - 为什么我需要为我创建的每个组件使用高度继承
- windows - 无法在 Powershell 5 中安装 CredentialManager 模块
- excel - 使用月末日期和函数从 Access 下载数据
- javascript - CSS 变换:旋转对象并缩放以适应容器内部
- sql-server - VS 2019下如何从SDK风格的C#测试项目调试T-SQL?
- angular - 如何设置 formControlName(在 HTML 中)以控制嵌套 3 次的 formArray 内?