Acceptヘッダーに従ってJSONまたはXMLを返す単純なRestful Webサービスを設定しようとしています。 Spring、Maven、およびWebLogic Serverを使用しています。私はこの投稿から例を取り上げました http://software.sawano.se/2012/03/combining-json-and-xml-in-restful-web.html を改善しようとしました。 GETとDELETEはJSONとXMLの両方で機能しますが、PUTとPOSTは「405 Method Not Allowed」エラーを返します。 Chrome Extension Advanced Rest Clientでこれをテストしようとしています。以下は、応答ヘッダーです。
Status
405 Method Not Allowed Show explanation Loading time: 327
Request headers
Accept: Application/json
Origin: chrome-extension://hgmloofddffdnphfgcellkdfbfbjeloo
User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/29.0.1547.76 Safari/537.36
Content-Type: application/x-www-form-urlencoded
Accept-Encoding: gzip,deflate,sdch
Accept-Language: en-US,en;q=0.8
Response headers
Connection: close
Date: Tue, 11 Feb 2014 15:17:24 GMT
Content-Length: 34
Content-Type: text/html
Allow: GET, DELETE
X-Powered-By: Servlet/2.5 JSP/2.1
Raw
Parsed
私が与えるリクエストの本文は以下です:
{
id: 1
name: "manga"
}
私のコントローラークラスは次のとおりです。
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.*;
import Java.util.HashSet;
import Java.util.Set;
@Controller
@RequestMapping("/users")
public class RESTController {
Userlist obj2;
boolean flag=false;
private Logger logger = LoggerFactory.getLogger(getClass());
@RequestMapping(value = "/{id}",method = RequestMethod.GET)
@ResponseBody
public User getUser(@PathVariable int id, @RequestHeader("Accept") String acceptHeader) {
User temp = new User();
if(obj2==null)
{
temp= new User(0, "Null");
}
else {
Set<User> set1= obj2.getUsers();
for(User a:set1)
{
if(id==a.getId()) temp=a;
}
}
logger.trace("Serving resource for Accept header: {}", acceptHeader);
return temp;
}
@RequestMapping(value="",method = RequestMethod.GET)
@ResponseBody
public Userlist getUsers(){
if(flag==false){
User new1=new User(1,"Rob");
User new2=new User(2,"VAN");
User new3=new User(3,"DAM");
User new4=new User(4,"Helio");
Set<User> obj1 =new HashSet<User>();
obj1.add(new1);
obj1.add(new2);
obj1.add(new3);
obj1.add(new4);
obj2=new Userlist(obj1);
flag=true;
}
return obj2;
}
@RequestMapping(value="/{id}",method = RequestMethod.DELETE)
@ResponseStatus(HttpStatus.OK)
public void deleteUser(@PathVariable int id){
Set<User> set1= obj2.getUsers();
for(User a:set1)
{
if(id==a.getId()) set1.remove(a);
}
Userlist obj3=new Userlist(set1);
obj2=obj3;
//return obj3;
}
@RequestMapping(value="/{id}",method = RequestMethod.PUT, consumes = "Application/json")
@ResponseStatus(HttpStatus.OK)
public void updateUser(@PathVariable int id, @RequestBody User temp){
System.out.println("Inside the put function");
if(temp==null){System.out.println("This is a Null for PUT");}
}
}
現在、PUTには何もありません。
どうやら、PUT呼び出し関数updateUser
を変更する必要があったようです。 @Consumes
、@RequestMapping
を削除し、関数に@ResponseBody
も追加しました。したがって、私のメソッドは次のようになりました。
@RequestMapping(value="/{id}",method = RequestMethod.PUT)
@ResponseStatus(HttpStatus.OK)
@ResponseBody
public void updateUser(@PathVariable int id, @RequestBody User temp){
Set<User> set1= obj2.getUsers();
for(User a:set1)
{
if(id==a.getId())
{
set1.remove(a);
a.setId(temp.getId());
a.setName(temp.getName());
set1.add(a);
}
}
Userlist obj3=new Userlist(set1);
obj2=obj3;
}
そしてそれは働いた!!!回答ありがとうございます。
応答で許可されたメソッドに注意してください
Connection: close
Date: Tue, 11 Feb 2014 15:17:24 GMT
Content-Length: 34
Content-Type: text/html
Allow: GET, DELETE
X-Powered-By: Servlet/2.5 JSP/2.1
GETとDELETEのみを受け入れます。したがって、サーバーを調整してPUTを有効にする必要があり、POSTも同様です。
Allow: GET, DELETE
私が正しいかどうかはわかりませんが、あなたが投稿したリクエストヘッダーから:
リクエストヘッダー
受け入れる:Application/json
起源:chrome-extension:// hgmloofddffdnphfgcellkdfbfbjeloo
ユーザーエージェント:Mozilla/5.0(Windows NT 6.1; WOW64)AppleWebKit/537.36(KHTML、Geckoなど)Chrome/29.0.1547.76 Safari/537.36
コンテンツタイプ:application/x-www-form-urlencoded
Accept-Encoding:gzip、deflate、sdch Accept-Language:en-US、en; q = 0.8
リクエスト本文をJSONタイプに設定していないようです。
問題は、POSTメソッドがNginxサーバーの静的ファイルリクエストに対して禁止されていることです。回避策は次のとおりです。
# Pass 405 as 200 for requested address:
server {
listen 80;
server_name localhost;
location / {
root html;
index index.html index.htm;
}
error_page 404 /404.html;
error_page 403 /403.html;
error_page 405 =200 $uri;
}
プロキシを使用する場合:
# If Nginx is like proxy for Apache:
error_page 405 =200 @405;
location @405 {
root /htdocs;
proxy_pass http://localhost:8080;
}
FastCGIを使用している場合:
location ~\.php(.*) {
fastcgi_pass 127.0.0.1:9000;
fastcgi_split_path_info ^(.+\.php)(.*)$;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
fastcgi_param PATH_INFO $fastcgi_path_info;
fastcgi_param PATH_TRANSLATED $document_root$fastcgi_path_info;
include /etc/nginx/fastcgi_params;
}
ブラウザは通常GETを使用するため、 ApiTester などのオンラインツールを使用してリクエストをテストできます。
あなたのコードが正しい場合、私も同じことが起こり、405エラーも発生します。何らかの認証の問題によるこのエラー。認証メニューに移動し、「親から認証を継承する」に変更します。