使用 Spring 数据休息和 MongoDB 更新特定字段

Update Specific Fields with Spring Data Rest and MongoDB

我正在使用 Spring Data MongoDB 和 Spring Data Rest 创建一个允许 GET、POST、PUT 和 DELETE 的 REST API我的 MongoDB 数据库上的操作,除了更新操作 (PUT) 外,一切正常。仅当我在请求正文中发送完整对象时它才有效。

例如我有以下实体:

@Document
public class User {
    @Id
    private String id;
    private String email;
    private String lastName;
    private String firstName;
    private String password;

    ...
}

要更新姓氏字段,我必须发送所有用户对象,包括密码!这显然是 非常错误的 。 如果我只发送要更新的字段,则所有其他字段在我的数据库中都设置为空。我什至尝试在这些字段上添加 @NotNull 约束,现在更新甚至不会发生,除非我发送所有用户对象的字段。

我尝试在这里搜索解决方案,但只找到以下 post 但没有解决方案:How to update particular field in mongo db by using MongoRepository Interface?

有没有办法实现这个?

Spring Data Rest 使用 Spring 数据存储库通过 Rest 调用自动检索和操作持久数据(查看 https://docs.spring.io/spring-data/rest/docs/current/reference/html/#reference)。

当使用 Spring 数据 MongoDB 时,您拥有 MongoOperations 接口,该接口用作 Rest 端点的存储库。 但是 MongoOperations 目前不支持特定字段更新!

PS:如果他们在 Spring Data JPA

中添加像 @DynamicUpdate 这样的功能,那就太棒了

但这并不意味着它可以完成,这是我遇到这个问题时所做的解决方法。

首先让我解释一下我们要做什么:

  1. 我们将创建一个控制器来覆盖所有 PUT 操作,以便我们可以实现自己的更新方法。
  2. 在该更新方法中,我们将使用具有更新特定字段能力的 MongoTemplate。

N.B。我们不想为应用程序中的每个模型重新执行这些步骤,因此我们将检索要动态更新的模型。为此,我们将创建一个实用程序 class。 [这是可选的]

让我们首先将 org.reflections api 添加到我们的项目依赖项中,这允许我们获得所有具有特定注释的 classes(我们的 @Document案例):

<dependency>
    <groupId>org.reflections</groupId>
    <artifactId>reflections</artifactId>
    <version>0.9.12</version>
</dependency>

然后创建一个名为 UpdateUtility 的新 class 并添加以下方法,并将 MODEL_PACKAGE 属性替换为您自己的包含实体的包:

public class UpdateUtility {

    private static final String MODEL_PACKAGE = "com.mycompany.myproject.models";
    private static boolean initialized =  false;
    private static HashMap<String, Class> classContext = new HashMap<>();

    private static void init() {
        if(!initialized) {
            Reflections reflections = new Reflections(MODEL_PACKAGE);
            Set<Class<?>> classes = reflections.getTypesAnnotatedWith(Document.class); // Get all the classes annotated with @Document in the specified package

            for(Class<?> model : classes) {
                classContext.put(model.getSimpleName().toLowerCase(), model);
            }

            initialized = true;
        }
    }

    public static Class getClassFromType(String type) throws Exception{
        init();
        if(classContext.containsKey(type)) {
            return classContext.get(type);
        }
        else {
            throw new Exception("Type " + type + " does not exists !");
        }
    }
}

使用此实用程序 class 我们可以检索模型 class 以从其类型进行更新。 例如:UpdateUtility.getClassFromType() 将 returns User.class

现在让我们创建我们的控制器:

public class UpdateController {

    @Autowired
    private MongoTemplate mongoTemplate;

    @PutMapping("/{type}/{id}")
    public Object update(@RequestBody HashMap<String, Object> fields,
                                  @PathVariable(name = "type") String type,
                                  @PathVariable(name = "id") String id) {
        try {
            Class classType = UpdatorUtility.getClassFromType(type); // Get the domain class from the type in the request
            Query query = new Query(Criteria.where("id").is(id)); // Update the document with the given ID
            Update update = new Update();

            // Iterate over the send fields and add them to the update object
            Iterator iterator = fields.entrySet().iterator();
            while(iterator.hasNext()) {
                HashMap.Entry entry = (HashMap.Entry) iterator.next();
                String key = (String) entry.getKey();
                Object value = entry.getValue();
                update.set(key, value);
            }

            mongoTemplate.updateFirst(query, update, classType); // Do the update
            return mongoTemplate.findById(id, classType); // Return the updated document
        } catch (Exception e) {
            // Handle your exception
        }
    }
}

现在我们可以在不更改调用的情况下更新指定的字段。 所以在你的情况下,调用将是:

PUT http://MY-DOMAIN/user/MY-USER-ID { lastName: "My new last name" }

PS:您可以通过添加更新嵌套对象中特定字段的可能性来改进它...