如何使用泽西拦截器获取请求正文

2022-09-02 19:35:58

我正在我的项目中使用。所有 POST 数据都以格式发送,并在服务器端取消编组到各自的 Bean 中。像这样:REST-JerseyJSON

向服务器发送请求:

    $('a#sayHelloPost').click(function(event){
        event.preventDefault();
        var mangaData = {
            title:'Bleach',
            author:'Kubo Tite'
        }
        var formData=JSON.stringify(mangaData);
        console.log(formData);
        $.ajax({
                url:'rest/cred/sayposthello',
                type: 'POST',
                data: formData,
                dataType: 'json',
                contentType:'application/json'
        })
});

有效载荷:

{"title":"Bleach","author":"Kubo Tite"}

服务器端:

@POST
@Path("/sayposthello")
@Produces(MediaType.APPLICATION_JSON)
public Response sayPostHello(MangaBean mb){
    System.out.println(mb);
    return Response.status(200).build();
}

漫画豆:

public class MangaBean {
    private String title;
    private String author;
    @Override
    public String toString() {
        return "MangaBean [title=" + title + ", author=" + author + "]";
    }
    public String getTitle() {
        return title;
    }
    public void setTitle(String title) {
        this.title = title;
    }
    public String getAuthor() {
        return author;
    }
    public void setAuthor(String author) {
        this.author = author;
    }
}

控制台上的输出:

MangaBean [title=Bleach, author=Kubo Tite]

从这里得到了REST拦截器实现。

public class JerseyFilter implements ContainerRequestFilter{

    @Override
    public ContainerRequest filter(ContainerRequest req) {
        return req;
    }

} 

我想访问拦截器中的有效负载(请求正文)。由于数据采用 JSON 格式,因此无法作为请求参数进行访问。有没有办法在拦截器方法中获取请求正文?请指教。


答案 1

下面是您可以实现的一种方法,与 Jersey 实现其日志记录筛选器的方式非常相似。您可以读取实体并将其粘贴回请求,这样您就不会意外地在筛选器中使用它。

import com.sun.jersey.api.container.ContainerException;
import com.sun.jersey.core.util.ReaderWriter;
import com.sun.jersey.spi.container.ContainerRequest;
import com.sun.jersey.spi.container.ContainerRequestFilter;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;

public class JerseyFilter implements ContainerRequestFilter {

    @Override
    public ContainerRequest filter(ContainerRequest request) {
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        InputStream in = request.getEntityInputStream();
        final StringBuilder b = new StringBuilder();
        try {
            if (in.available() > 0) {
                ReaderWriter.writeTo(in, out);

                byte[] requestEntity = out.toByteArray();
                printEntity(b, requestEntity);

                request.setEntityInputStream(new ByteArrayInputStream(requestEntity));
            }
            return request;
        } catch (IOException ex) {
            throw new ContainerException(ex);
        }

    }

    private void printEntity(StringBuilder b, byte[] entity) throws IOException {
        if (entity.length == 0)
            return;
        b.append(new String(entity)).append("\n");
        System.out.println("#### Intercepted Entity ####");
        System.out.println(b.toString());
    }
}

答案 2

您只能读取内容一次,因为它是输入流。如果您在拦截器中拾取它,则将无法在主解析中提供它。

您需要做的是创建一个过滤器,该过滤器读取数据并将其提供给需要它的任何其他人。基本步骤如下:

  • 在过滤器中创建一个新的 HttpServletRequestWrapper 子类,该子类将输入读入缓冲区并覆盖 getInputStream() 以向该缓冲区提供新的输入流
  • 将新子类向下传递到链中

推荐