- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
简短版本是:如何使用 Spring Data Rest PATCH 方法修补包含在 Postgres jsonb
字段中的 JSON 对象?
这里是长版本,请考虑以下实体:
@Entity
@Table(name = "examples")
public class Example {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private long id;
private String jsonobject;
@JsonRawValue
public String getJsonobject() {
return jsonobject == null ? null : jsonobject;
}
public void setJsonobject(JsonNode jsonobject) {
this.jsonobject = jsonobject == null ? null : jsonobject.toString();
}
}
jsonobject
是 Postgres 类型 jsonb
。这些 getter/setter 是为提到的 Spring Data Rest 序列化/反序列化它的方式 here .我们还尝试为该字段提供自己的类型,如 these answers 中所述.
我们的目标是使用 Spring Data Rest 修补该字段包含的 JSON 对象。
例如:
GET /examples/1
{
"id": 1,
"jsonobject": {
"foo": {"bar": "Hello"},
"baz": 2
}
}
PATCH /examples/1
{
"jsonobject": {
"foo": {"bar": "Welcome"}
}
}
预期输出:
GET /examples/1
{
"id": 1,
"jsonobject": {
"foo": {"bar": "Welcome"},
"baz": 2
}
}
当前输出:
GET /examples/1
{
"id": 1,
"jsonobject": {
"foo": {"bar": "Welcome"}
}
}
Spring Data Rest 修补 Example 资源并覆盖每个请求的属性的值,而不是试图深入 JSON 对象的属性以仅修补请求的嵌套属性。
此时我们认为 Spring Data Rest 对 application/merge-patch+json
和 application/json-patch+json
媒体类型的支持会派上用场。以下是每种媒体类型的输出:
application/merge-patch+json
:
PATCH /examples/1
{
"jsonobject": {
"foo": {"bar": "Welcome"}
}
}
输出:
GET /examples/1
{
"id": 1,
"jsonobject": {
"foo": {"bar": "Welcome"}
}
}
application/json-patch+json
:
PATCH /examples/1
[
{ "op": "replace", "path": "/jsonobject/foo/bar", "value": "Welcome" }
]
输出:
{
"cause": {
"cause": null,
"message": "EL1008E:(pos 8): Property or field 'foo' cannot be found on object of type 'java.lang.String' - maybe not public?"
},
"message": "Could not read an object of type class com.example.Example from the request!; nested exception is org.springframework.expression.spel.SpelEvaluationException: EL1008E:(pos 8): Property or field 'foo' cannot be found on object of type 'java.lang.String' - maybe not public?"
}
归结为相同的想法:仅查找实体属性,然后完全覆盖或找不到。
问题如下:有没有办法让 Spring Data Rest 理解它正在处理 jsonb
字段,从而查找 JSON 嵌套属性而不是只查找实体属性?
注意:@Embeddable/@Embedded
注释最有可能被避免,因为它们暗示知道嵌套的属性名称,这会降低对 jsonb
字段的兴趣.
感谢阅读。
最佳答案
好吧,您的 EntityManager 不知道您的 jsonObject 字段中有一些结构,它是纯字符串。您应该实现自己的解决方法。您可以如何开始工作的一个例子在这里 https://github.com/bazar-nazar/pgjson但是这种方法将要求您每次都从数据库中读取对象,并进行另一次序列化/反序列化往返。
但是如果你在 postgresql 上,你可以使用它的所有功能(注意:这会使你的应用程序与 postgresql 紧密耦合,因此数据库将变得更难被替换)
我建议实现自定义 jdbc 查询,比如简单的例子:
public static class JsonPatchRequest {
String path;
String operation;
String value;
}
@Inject
private JdbcTemplate jdbcTemplate;
@PatchMapping(value = "/example/{id}")
public void doPatch(@PathVariable("id") Long id, @RequestBody JsonPatchRequest patchRequest) {
// this line should transform your request path from "/jsonobject/foo/bar" to "{foo,bar}" string
String postgresqlpath = "{" + patchRequest.path.replaceFirst("/jsonobject/", "").replaceAll("/", ",") + "}";
switch(patchRequest.operation) {
case "replace" :
jdbcTemplate.execute("UPDATE example SET jsonobject = jsonb_set(jsonobject, ?, jsonb ?) WHERE id = ?", new PreparedStatementCallback<Void>() {
@Override
public Void doInPreparedStatement(PreparedStatement ps) throws SQLException, DataAccessException {
ps.setString(1, postgresqlpath);
// this one transforms pure value, to string-escaped value (manual workaround) so 'value' should become '"value"'
ps.setString(2, "\"".concat(patchRequest.value).concat("\""));
ps.setLong(3, id);
ps.execute();
return null;
}
});
break;
case "delete" :
jdbcTemplate.execute("UPDATE example SET jsonobject = jsonobject #- ? WHERE id = ? ", new PreparedStatementCallback<Void>() {
@Override
public Void doInPreparedStatement(PreparedStatement ps) throws SQLException, DataAccessException {
ps.setString(1, postgresqlpath);
ps.setLong(2, id);
ps.execute();
return null;
}
});
break;
}
}
另请注意:第一种方法将强制您将 jsonobjet 字段设为预定义类型,因此可以将其替换为纯规范化实体,因此与它没有太大关系。第二种方法不会强制您在 json 中包含任何类型的结构。
希望对您有所帮助。
关于Spring Data Rest - PATCH Postgres jsonb 字段,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40366781/
我正在尝试使用 Android SDK 中的 Draw9Patch 工具,但在使用 Draw9Patch 打开我的图像后,它立即将我的所有图像区域显示为“坏补丁”。我手动绘制了定义可拉伸(stretc
我有一个运行补丁命令的构建工具,如果补丁命令返回非零值,它将导致构建失败。我正在应用一个可能已经或可能尚未应用的补丁,所以我使用 -N选项 patch ,它应该跳过。但是,当它确实跳过时,patch返
patching file chrome/browser/gpu_process_host_ui_shim.cc Unreversed patch detected! Skipping patch.
尝试查看哪些模型最适合api(更新少,但对象结构可能经常更改,高读取应用程序) 我有这样的资源 Epic(ID、名称、描述、开始日期、结束日期、状态、故事) 故事(ID、名称、说明、开始日期、结束日期
我创建了新问题,因为我觉得上一个问题已经得到解答,这属于一个新的地方。 我跑 bitbake core-image-minimal我收到以下错误: ERROR: systemd-1_232-r0 do
我正在使用 ColdFusion 10 与 PayPal 的服务器和 for some requests 进行通信我需要执行 HTTP PATCH 请求 are not supported by CF
我试图了解这两种模拟方法之间的区别。有人可以帮助区分它们吗?对于这个例子,我使用 passlib 库。 from passlib.context import CryptContext from un
WebTarget webTarget = httpClient.target(url); Invocation.Builder invocationBuilder = webTarget.reque
当提到“提交补丁”时,补丁这个词究竟是什么意思? 我已经看到它被大量使用,尤其是在开源世界中。它是什么意思,提交补丁到底涉及什么? 最佳答案 这是一个文件,其中列出了已更改的代码文件之间的差异。它通常
对于 matplotlib.patches,patch.contains_point(xy) 方法似乎与 patch.get_path().contains_point(xy) 不同,至少在拥有之后将
这是什么RFC 5789说: The PATCH method requests that a set of changes described in the request entity be ap
在 Draw 9-patch 中,一切看起来都很好。 但是,我的 SDK 说 9-patch png 格式不正确。因为我有类似 11-patch png 的东西。因为我不希望小抓取区域被缩放。如何让它
我创建了一个使用 javax.xml.ws.Endpoint 来创建 REST 端点的类: @WebServiceProvider @ServiceMode(value = javax.xml.ws.
语境 我有一个 spring boot (version 2.2.6.RELEASE) web 项目。 从这个 Web 应用程序(我称之为“APP1”)我想使用来自另一个 Web 应用程序的 PATC
在为我的应用程序编写单元测试时,我一直使用 @mock.patch 和 @patch.object 装饰器。但是现在,当我使用装饰器进行一些单元测试时,我收到错误消息“TypeError: stati
我在使用@mock.patch.object 函数时观察到 nosetests 的一个非常奇怪的行为: 当我同时运行多个测试时,我得到的结果与单独运行它们时不同。具体来说,在某些情况下,当我一起运行多
我正在使用 RestSharp v107。 我想更新测试用例的迭代路径。我可以使用 Postman 更新它,但使用 RestSharp 我收到“BAD Request”-“您必须在请求正文中传递有效的
我已经阅读了 GNU 项目中关于开源和其他许可证的文章。某些许可证允许您将更改发布为补丁,而不是完整的源代码(例如 Q 公共(public)许可证或 gnuplot 许可证)。这是什么意思?这样的补丁
有谁知道免费的优质补丁程序?您知道,可以将其中包含旧程序的目录放入其中,然后将其与具有新版本的目录进行比较,然后吐出一个补丁,这仅仅是两者之间的区别? 另外,我正在寻找可以修补整个目录的东西,而不仅仅
由于我当时一直在使用 Subversion 和 shell 工具,git-gui这些都是不可能的。是否有任何 shell 工具可以交互式地逐行应用补丁? 最佳答案 尝试通过 --dry-run选项 p
我是一名优秀的程序员,十分优秀!