1. ホーム
  2. java

[解決済み] POST リクエスト送信時に org.springframework.http.converter.HttpMessageNotReadableException が発生する。

2022-02-03 06:12:56

質問

以下のコントローラを持つSpringアプリケーションを持っています。

   @RestController
   @RequestMapping("/app") 
   public class RegisterRestController {
   @Autowired
    UserRepository userRepository;

   @Autowired
   PasswordEncoder passwordEncoder;

   @Autowired
   UserService userService;


   @RequestMapping( value="/loginuser", method =RequestMethod.POST,produces="application/json")
    public String loginUser(@RequestBody String requestBody) {
    System.out.println("inside");
    JSONObject responseJsonObject = new JSONObject();
    String phonenumber;
    String password;
    try{
        JSONObject object = new JSONObject(requestBody);
        phonenumber = object.getString("phonenumber");
        password = object.getString("password");
        User user = userService.findByNumber(phonenumber);
        String sha256Password = passwordEncoder.encode(password);
        if(sha256Password.equals(user.getPassword())){
            responseJsonObject.put("response", "Login Successful");
        }
        else {
            responseJsonObject.put("repsonse", "Login failed");
        }
    }
    catch (Exception e){
        e.printStackTrace();
        try {
            responseJsonObject.put("response", "Invalid Credentials");
        } catch (JSONException e1) {
            e1.printStackTrace();
        }

    }
    return responseJsonObject.toString();
}

しかし、Postmanから.NET Frameworkを含むPOSTリクエストを送ると、:

    {
      "phonenumber":"9123456789",
      "password":"password"
  }

次のような応答が返ってきます。

    {
   "timestamp": 1456043810789,
   "status": 400,
    "error": "Bad Request",
    "exception":      "org.springframework.http.converter.HttpMessageNotReadableException",
    "message": "Could not read JSON: Can not deserialize instance of   java.lang.String out of START_OBJECT token\n at [Source: java.io.PushbackInputStream@eaa3acb; line: 1, column: 1]; nested exception is com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of java.lang.String out of START_OBJECT token\n at [Source: java.io.PushbackInputStream@eaa3acb; line: 1, column: 1]",
    "path": "/app/loginuser"
}

あと、Spring Securityも同様に実験していました。サーバーにはエラーは出ず、コントローラには"inside"が出力されないので、リクエストを受信していないようです。私はSpringを知ろうとしているのですが、このようなエラーの原因を見つけることができませんでした。ご教示いただければ幸いです。よろしくお願いします。

解決方法は?

あなたのコードには2つの問題があります。

  1. コントローラ内でJSONをオブジェクトに変換しようとしている。 これはすでにSpringが行っています。リクエストのボディを受け取り、コントローラメソッド内の対応するパラメーターのJavaクラスに変換しようとします。
  2. コントローラのメソッドでは、1つの文字列を想定しています。 @RequestBody String requestBody

そして、2つのプロパティを持つオブジェクトを送信していますね。

{
    "phonenumber": "9123456789",
    "password": "password"
}

解決策

ログインに必要な値のクラスを作成します。

public class Login {
    public String phonenumber;
    public String password;
    // you need a zero argument constructor
    // maybe you have to add getter and setters
}

コントローラのメソッドを変更し、このタイプのオブジェクトを期待するようにします。

@RequestBody Login requestBody