Jax-rs json pretty output

前端 未结 6 1613
执笔经年
执笔经年 2020-12-08 22:48

in Java when i use the

@Produces(\"application/json\")

annotation the output is not formated into human readable form. How do i achive that

相关标签:
6条回答
  • 2020-12-08 23:15

    If you are using Spring, then you can globally set the property

    spring.jackson.serialization.INDENT_OUTPUT=true
    

    More info at https://docs.spring.io/spring-boot/docs/current/reference/html/howto-properties-and-configuration.html

    0 讨论(0)
  • 2020-12-08 23:21

    Just for the record, if you want to enable the pretty output only for some resources you can use the @JacksonFeatures annotation on a resource method.

    Here is example:

    @Produces(MediaType.APPLICATION_JSON)
    @JacksonFeatures(serializationEnable =  { SerializationFeature.INDENT_OUTPUT })
    public Bean resource() {
        return new Bean();
    }
    
    0 讨论(0)
  • 2020-12-08 23:24

    Building on helpful DaTroop's answer, here is another version which allows choosing between optimized json and formatted json based on the absence or presence of a "pretty" parameter :

    package test;
    
    
    import javax.ws.rs.Produces;
    import javax.ws.rs.core.Context;
    import javax.ws.rs.core.MediaType;
    import javax.ws.rs.core.MultivaluedMap;
    import javax.ws.rs.core.UriInfo;
    import javax.ws.rs.ext.ContextResolver;
    import javax.ws.rs.ext.Provider;
    
    import org.codehaus.jackson.map.ObjectMapper;
    import org.codehaus.jackson.map.SerializationConfig;
    
    @Provider
    @Produces(MediaType.APPLICATION_JSON)
    public class JacksonContextResolver implements ContextResolver<ObjectMapper> {
    
        private ObjectMapper prettyPrintObjectMapper;
        private UriInfo uriInfoContext;
    
        public JacksonContextResolver(@Context UriInfo uriInfoContext) throws Exception {
            this.uriInfoContext = uriInfoContext;
    
            this.prettyPrintObjectMapper = new ObjectMapper();
            this.prettyPrintObjectMapper.configure(SerializationConfig.Feature.INDENT_OUTPUT, true);
        }
    
        @Override
        public ObjectMapper getContext(Class<?> objectType) {
    
            try {
                MultivaluedMap<String, String> queryParameters = uriInfoContext.getQueryParameters();
                if(queryParameters.containsKey("pretty")) {
                    return prettyPrintObjectMapper;
                }
    
            } catch(Exception e) {
                // protect from invalid access to uriInfoContext.getQueryParameters()
            }
    
            return null; // use default mapper
        }
    }
    
    0 讨论(0)
  • 2020-12-08 23:26

    This is how you can properly do conditional pretty/non-pretty json output based on presence of "pretty" in query string.

    Create a PrettyFilter that implements ContainerResponseFilter, that will be executed on every request:

    @Provider
    public class PrettyFilter implements ContainerResponseFilter {
    
        @Override
        public void filter(ContainerRequestContext reqCtx, ContainerResponseContext respCtx) throws IOException {
    
            UriInfo uriInfo = reqCtx.getUriInfo();
            //log.info("prettyFilter: "+uriInfo.getPath());
    
            MultivaluedMap<String, String> queryParameters = uriInfo.getQueryParameters();
            if(queryParameters.containsKey("pretty")) {
                ObjectWriterInjector.set(new IndentingModifier(true));
            }
    
        }
    
        public static class IndentingModifier extends ObjectWriterModifier {
    
            private final boolean indent;
    
            public IndentingModifier(boolean indent) {
                this.indent = indent;
            }
    
    
            @Override
            public ObjectWriter modify(EndpointConfigBase<?> endpointConfigBase, MultivaluedMap<String, Object> multivaluedMap, Object o, ObjectWriter objectWriter, JsonGenerator jsonGenerator) throws IOException {
                if(indent) jsonGenerator.useDefaultPrettyPrinter();
                return objectWriter;
            }
        }
    }
    

    And pretty much that's it!

    You will need to ensure that this class gets used by Jersey by either automated package scanning or registered manually.

    Spent few hours trying to achieve that and found that no-one has published a ready-to-use solution before.

    0 讨论(0)
  • 2020-12-08 23:34

    Create this class anywhere in your project. It will be loaded on deployment. Notice the .configure(SerializationConfig.Feature.INDENT_OUTPUT, true); which configures the mapper to format the output.

    For Jackson 2.0 and later, replace the two .configure() lines with these: .configure(DeserializationFeature.FAIL_ON_IGNORED_PROPERTIES, false) .configure(SerializationFeature.INDENT_OUTPUT, true);

    And change your imports accordingly.

    package com.secret;
    
    import javax.ws.rs.Produces;
    import javax.ws.rs.core.MediaType;
    import javax.ws.rs.ext.ContextResolver;
    import javax.ws.rs.ext.Provider;
    import org.codehaus.jackson.map.DeserializationConfig;
    import org.codehaus.jackson.map.ObjectMapper;
    import org.codehaus.jackson.map.SerializationConfig;
    
    /**
     *
     * @author secret
     */
    @Provider
    @Produces(MediaType.APPLICATION_JSON)
    public class JacksonContextResolver implements ContextResolver<ObjectMapper> {
        private ObjectMapper objectMapper;
    
        public JacksonContextResolver() throws Exception {
            this.objectMapper = new ObjectMapper();
        this.objectMapper
            .configure(DeserializationConfig.Feature.FAIL_ON_UNKNOWN_PROPERTIES, false)
            .configure(SerializationConfig.Feature.INDENT_OUTPUT, true);
        }
    
        @Override
        public ObjectMapper getContext(Class<?> objectType) {
            return objectMapper;
        }
    }
    

    Bear in mind that formatting has a negative effect on performance.

    0 讨论(0)
  • 2020-12-08 23:35

    If you are using the jersey-media-json-binding dependency, which uses Yasson (the official RI of JSR-367) and JAVAX-JSON, you can introduce pretty printing as follows:

    import javax.json.bind.Jsonb;
    import javax.json.bind.JsonbBuilder;
    import javax.json.bind.JsonbConfig;
    import javax.ws.rs.ext.ContextResolver;
    import javax.ws.rs.ext.Provider;
    
    @Provider
    public class RandomConfig implements ContextResolver<Jsonb> {
        private final Jsonb jsonb = JsonbBuilder.create(new JsonbConfig().withFormatting(true));
    
        public RandomConfig() { }
    
        @Override
        public Jsonb getContext(Class<?> objectType) {
            return jsonb;
        }
    }
    
    0 讨论(0)
提交回复
热议问题