如何使用 curl 测试 Java RESTful 网络服务 - PUT 动词的问题
How to test a Java RESTful web service with curl - problems with PUT verb
我正在尝试我的第一个 Java RESTful 网络服务,可能我没有明确的一些机制。
这是我的代码示例:
@Path(Paths.USERS)
public class UserService {
private static final String OK_MESSAGE_USERSERVICE_PUT = Messages.OK_MESSAGE_USERSERVICE_PUT;
private Client esClient = ElasticSearch.getClient();
@GET
@Produces(MediaType.APPLICATION_JSON)
public String get(@QueryParam(QParams.ID) String id) {
// TODO Authentication
try {
GetResponse response = esClient
.prepareGet(PRIMARY_INDEX_NAME, USERS_TYPE_NAME, id)
.execute().actionGet();
if (response != null) {
return response.getSourceAsString();
}
}catch (ElasticsearchException e) {
e.printStackTrace();
return e.getMessage();
}
return Messages.RESOURCE_NOT_FOUND;
}
@PUT
@Consumes(MediaType.APPLICATION_JSON)
public String update(@QueryParam(QParams.ID) String id,
@PathParam("metadata") String metadata) {
// TODO Authentication
boolean isMyself = true;
// precondition, the user exsists. If the check fails, you
// should put the isMyself flag at false.
if (isMyself){
esClient
.prepareIndex(PRIMARY_INDEX_NAME, USERS_TYPE_NAME, id)
.setSource(metadata).execute().actionGet();
}
return OK_MESSAGE_USERSERVICE_PUT;
}
我的问题是:
我应该如何将元数据传递给网络服务?
我试过
curl -g -X PUT 'http://localhost:8080/geocon/users?id=007&metadata={"name":{"first":"james","last":"bond"}}'
但我遇到了
这样的错误
Root Cause: java.net.URISyntaxException: Illegal character in query at index 33: /geocon/users?id=007&metadata=%7B"name":%7B"first":"james","last":"bond"%7D%7D
java.net.URI$Parser.fail(URI.java:2848)
谷歌搜索,我尝试了这个不同的解决方案:
curl -X PUT -H "application/json" -d '{"name":{"first":"james","last":"bond"}}' http://localhost:8080/geocon/users/
但是使用这种方法,我不知道如何将我更新 ID 为 007 的用户的意愿传递给网络服务(因为,据我所知,我只是在交流 {"name":{"first":"james","last":"bond"}}).
你会怎么做?谢谢!
看起来您将参数传递给 GET
请求 (?foo=bar
) 的方式与 PUT
方法混合在一起,这更像是 POST
。
我希望您将 JSON 数据发送到 URL,例如 /gecon/user/13
,当然是在正确编辑 URL 映射之后。
尝试阅读 the wikipedia and this answer。
我会完全重构这个解决方案。
首先:更改url方案,使id成为URI路径的一部分。这是一种更 RESTful 的方法
@PUT
@Path("{id}")
@Consumes(MediaType.APPLICATION_JSON)
public String update(@PathParam("id") String id) {
其二:请求实体主体应该是JSON。我不认为这是 "metadata" 的任何理由。你可以完全摆脱它。你想用你发送的数据更新资源表示,所以这应该是正文的一部分。
第三: 如果您正在使用 JSON,您应该利用像 Jackson 这样的提供商提供的 Pojo 映射。这会自动将 JSON 解析为 Pojo。你可以做类似
的事情
public class Agent {
private Name name;
// getters and setters
public static class Name {
private String first;
private String last;
// getters and setters
}
}
然后在方法签名中只包含 Agent
参数,表示它是请求的主体。
@PUT
@Path("{id}")
@Consumes(MediaType.APPLICATION_JSON)
public String update(@PathParam("id") String id, Agent agent) {
您需要将 Jackson 提供程序添加到项目中。希望您正在使用 Maven。您可以添加
<dependency>
<groupId>com.fasterxml.jackson.jaxrs</groupId>
<artifactId>jackson-jaxrs-json-provider</artifactId>
<version>2.4.0</version>
</dependency>
然后在您的应用程序配置中,注册 JacksonJsonProvider
。如果您需要这方面的帮助,请展示您是如何配置您的应用程序的(无论是 web.xml 还是 Application
子类)。如果您想跳过 Pojo 映射并只获取原始 JSON,只需将方法参数类型设置为 String
,就像您已经在做的那样(只是没有注释)
第四:那么你的要求是(我在Windows,这里我们需要使用双引号并转义内部引号)
curl -v -X PUT
-H "Content-Type:application/json"
-d "{\"name\":{\"first\":\"james\", \"last\":\"bond\"}}"
http://localhost:8080/geocon/users/007
第五: 我也会更改 @GET
方法,使 id 成为 URL 路径的一部分,而不是查询参数。
我正在尝试我的第一个 Java RESTful 网络服务,可能我没有明确的一些机制。
这是我的代码示例:
@Path(Paths.USERS)
public class UserService {
private static final String OK_MESSAGE_USERSERVICE_PUT = Messages.OK_MESSAGE_USERSERVICE_PUT;
private Client esClient = ElasticSearch.getClient();
@GET
@Produces(MediaType.APPLICATION_JSON)
public String get(@QueryParam(QParams.ID) String id) {
// TODO Authentication
try {
GetResponse response = esClient
.prepareGet(PRIMARY_INDEX_NAME, USERS_TYPE_NAME, id)
.execute().actionGet();
if (response != null) {
return response.getSourceAsString();
}
}catch (ElasticsearchException e) {
e.printStackTrace();
return e.getMessage();
}
return Messages.RESOURCE_NOT_FOUND;
}
@PUT
@Consumes(MediaType.APPLICATION_JSON)
public String update(@QueryParam(QParams.ID) String id,
@PathParam("metadata") String metadata) {
// TODO Authentication
boolean isMyself = true;
// precondition, the user exsists. If the check fails, you
// should put the isMyself flag at false.
if (isMyself){
esClient
.prepareIndex(PRIMARY_INDEX_NAME, USERS_TYPE_NAME, id)
.setSource(metadata).execute().actionGet();
}
return OK_MESSAGE_USERSERVICE_PUT;
}
我的问题是:
我应该如何将元数据传递给网络服务? 我试过
curl -g -X PUT 'http://localhost:8080/geocon/users?id=007&metadata={"name":{"first":"james","last":"bond"}}'
但我遇到了
这样的错误Root Cause: java.net.URISyntaxException: Illegal character in query at index 33: /geocon/users?id=007&metadata=%7B"name":%7B"first":"james","last":"bond"%7D%7D
java.net.URI$Parser.fail(URI.java:2848)
谷歌搜索,我尝试了这个不同的解决方案:
curl -X PUT -H "application/json" -d '{"name":{"first":"james","last":"bond"}}' http://localhost:8080/geocon/users/
但是使用这种方法,我不知道如何将我更新 ID 为 007 的用户的意愿传递给网络服务(因为,据我所知,我只是在交流 {"name":{"first":"james","last":"bond"}}).
你会怎么做?谢谢!
看起来您将参数传递给 GET
请求 (?foo=bar
) 的方式与 PUT
方法混合在一起,这更像是 POST
。
我希望您将 JSON 数据发送到 URL,例如 /gecon/user/13
,当然是在正确编辑 URL 映射之后。
尝试阅读 the wikipedia and this answer。
我会完全重构这个解决方案。
首先:更改url方案,使id成为URI路径的一部分。这是一种更 RESTful 的方法
@PUT
@Path("{id}")
@Consumes(MediaType.APPLICATION_JSON)
public String update(@PathParam("id") String id) {
其二:请求实体主体应该是JSON。我不认为这是 "metadata" 的任何理由。你可以完全摆脱它。你想用你发送的数据更新资源表示,所以这应该是正文的一部分。
第三: 如果您正在使用 JSON,您应该利用像 Jackson 这样的提供商提供的 Pojo 映射。这会自动将 JSON 解析为 Pojo。你可以做类似
的事情public class Agent {
private Name name;
// getters and setters
public static class Name {
private String first;
private String last;
// getters and setters
}
}
然后在方法签名中只包含 Agent
参数,表示它是请求的主体。
@PUT
@Path("{id}")
@Consumes(MediaType.APPLICATION_JSON)
public String update(@PathParam("id") String id, Agent agent) {
您需要将 Jackson 提供程序添加到项目中。希望您正在使用 Maven。您可以添加
<dependency>
<groupId>com.fasterxml.jackson.jaxrs</groupId>
<artifactId>jackson-jaxrs-json-provider</artifactId>
<version>2.4.0</version>
</dependency>
然后在您的应用程序配置中,注册 JacksonJsonProvider
。如果您需要这方面的帮助,请展示您是如何配置您的应用程序的(无论是 web.xml 还是 Application
子类)。如果您想跳过 Pojo 映射并只获取原始 JSON,只需将方法参数类型设置为 String
,就像您已经在做的那样(只是没有注释)
第四:那么你的要求是(我在Windows,这里我们需要使用双引号并转义内部引号)
curl -v -X PUT
-H "Content-Type:application/json"
-d "{\"name\":{\"first\":\"james\", \"last\":\"bond\"}}"
http://localhost:8080/geocon/users/007
第五: 我也会更改 @GET
方法,使 id 成为 URL 路径的一部分,而不是查询参数。