简短版本是:如何使用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 . 我们还试图给该字段提供自己的类型,如_1556140中所述 .
我们的目标是使用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嵌套属性而不是只查找实体属性?
Nb: @Embeddable/@Embedded
注释最有可能被避免,因为它们意味着知道嵌套的属性名称,这会降低对 jsonb
字段的兴趣 .
谢谢你的阅读 .
2 回答
好吧,你的EntityManager不知道你的jsonObject字段里面有一些结构,它是纯字符串 . 您应该实现自己的解决方法 . 如何开始工作的一个例子就是https://github.com/bazar-nazar/pgjson但是这样的方法每次都需要你从数据库中读取对象,并进行另一次序列化/反序列化往返 .
但如果您使用的是postgresql,则可以使用它的所有功能(注意:这会使您的应用程序与postgresql紧密结合,因此数据库将变得更难更换)
我建议实现自定义jdbc查询,就像简单的例子:
还要注意:第一种方法会强制你制作预定义类型的jsonobjet字段,因此它可以用纯规范化实体替换,因此与它没什么关系 . 第二种方法不会强迫你在你的json中有任何类型的结构 .
希望这会帮助你 .
Assumes Hibernate 5 is used as JPA implementation
使
jsonobject
字段成为特定的类类型(包含所需的字段)而不是String
.然后,您可以为
jsonb
类型添加自定义Hibernate用户类型 .自定义类型实现非常冗长,但实际上它使用Jackson的
ObjectMapper
将对象作为String
传递给JDBC语句(反之亦然,从ResultSet中检索) .最后,您需要告诉hibernate将Java对象存储为
jsonb
Postgre类型 . 这意味着创建自定义方言类(并在其中配置) .有了这些,你应该没问题,Spring Data Rest补丁机制应该可行 .
PS
答案很大程度上受到了this github repo的启发,它基本上是相同的,但是使用了Hibernate 4.看一下 .