All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.fasterxml.jackson.jakarta.rs.yaml.JacksonYAMLProvider Maven / Gradle / Ivy

Go to download

Functionality to handle YAML input/output for Jakarta-RS implementations (like Jersey and RESTeasy) using standard Jackson data binding.

The newest version!
package com.fasterxml.jackson.jakarta.rs.yaml;

import java.io.IOException;
import java.io.InputStream;
import java.io.PushbackInputStream;
import java.lang.annotation.Annotation;

import jakarta.ws.rs.Consumes;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.Context;
import jakarta.ws.rs.core.MediaType;
import jakarta.ws.rs.ext.*;

import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.Version;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ObjectReader;
import com.fasterxml.jackson.databind.ObjectWriter;
import com.fasterxml.jackson.dataformat.yaml.YAMLMapper;
import com.fasterxml.jackson.jakarta.rs.base.ProviderBase;
import com.fasterxml.jackson.jakarta.rs.cfg.Annotations;

/**
 * Basic implementation of Jakarta-RS abstractions ({@link MessageBodyReader},
 * {@link MessageBodyWriter}) needed for binding
 * YAML content to and from Java Objects ("POJO"s).
 * 

* Actual data binding functionality is implemented by {@link ObjectMapper}: * mapper to use can be configured in multiple ways: *

    *
  • By explicitly passing mapper to use in constructor *
  • By explcitly setting mapper to use by {@link #setMapper} *
  • By defining Jakarta-RS Provider that returns {@link YAMLMapper}s. *
  • By doing none of above, in which case a default mapper instance is * constructed (and configured if configuration methods are called) *
* The last method ("do nothing specific") is often good enough; explicit passing * of Mapper is simple and explicit; and Provider-based method may make sense * with Dependency Injection frameworks, or if Mapper has to be configured differently * for different media types. *

* Note that the default mapper instance will be automatically created if * one of explicit configuration methods (like {@link #configure}) * is called: if so, Provider-based introspection is NOT used, but the * resulting Mapper is used as configured. *

* Note that there is also a sub-class -- ({@link JacksonXmlBindYAMLProvider}) -- that * is configured by default to use both Jackson and XmlBind annotations for configuration * (base class when used as-is defaults to using just Jackson annotations) */ @Provider @Consumes(MediaType.WILDCARD) // As per https://www.rfc-editor.org/rfc/rfc9512.html @Produces({ "application/yaml", MediaType.WILDCARD }) public class JacksonYAMLProvider extends ProviderBase { /** * Default annotation sets to use, if not explicitly defined during * construction: only Jackson annotations are used for the base * class. Sub-classes can use other settings. */ public final static Annotations[] BASIC_ANNOTATIONS = { Annotations.JACKSON }; /* /********************************************************** /* Context configuration /********************************************************** */ /** * Injectable context object used to locate configured * instance of {@link YAMLMapper} to use for actual * serialization. */ @Context protected Providers _providers; /* /********************************************************** /* Construction /********************************************************** */ /** * Default constructor, usually used when provider is automatically * configured to be used with Jakarta-RS implementation. */ public JacksonYAMLProvider() { this(null, BASIC_ANNOTATIONS); } /** * @param annotationsToUse Annotation set(s) to use for configuring * data binding */ public JacksonYAMLProvider(Annotations... annotationsToUse) { this(null, annotationsToUse); } public JacksonYAMLProvider(YAMLMapper mapper) { this(mapper, BASIC_ANNOTATIONS); } /** * Constructor to use when a custom mapper (usually components * like serializer/deserializer factories that have been configured) * is to be used. * * @param annotationsToUse Sets of annotations (Jackson, XmlBind) that provider should * support */ public JacksonYAMLProvider(YAMLMapper mapper, Annotations[] annotationsToUse) { super(new YAMLMapperConfigurator(mapper, annotationsToUse)); } /** * Method that will return version information stored in and read from jar * that contains this class. */ @Override public Version version() { return PackageVersion.VERSION; } /* /********************************************************** /* Abstract method impls /********************************************************** */ @Override protected YAMLEndpointConfig _configForReading(ObjectReader reader, Annotation[] annotations) { return YAMLEndpointConfig.forReading(reader, annotations); } @Override protected YAMLEndpointConfig _configForWriting(ObjectWriter writer, Annotation[] annotations) { return YAMLEndpointConfig.forWriting(writer, annotations); } /** * Helper method used to check whether given media type * is YAML type or sub type. * Current implementation essentially checks to see whether * {@link MediaType#getSubtype} returns "xml" or something * ending with "+yaml". */ @Override protected boolean hasMatchingMediaType(MediaType mediaType) { /* As suggested by Stephen D, there are 2 ways to check: either * being as inclusive as possible (if subtype is "yaml"), or * exclusive (major type "application", minor type "yaml"). * Let's start with inclusive one, hard to know which major * types we should cover aside from "application". */ if (mediaType != null) { String subtype = mediaType.getSubtype(); return "yaml".equalsIgnoreCase(subtype) || subtype.endsWith("+yaml"); //tarik: apparently there is not a standard for yaml types, should be discussed } /* Not sure if this can happen; but it seems reasonable * that we can at least produce yaml without media type? */ return true; } /** * Method called to locate {@link YAMLMapper} to use for serialization * and deserialization. If an instance has been explicitly defined by * {@link #setMapper} (or non-null instance passed in constructor), that * will be used. * If not, will try to locate it using standard Jakarta-RS * {@link ContextResolver} mechanism, if it has been properly configured * to access it (by Jakarta-RS runtime). * Finally, if no mapper is found, will return a default unconfigured * {@link ObjectMapper} instance (one constructed with default constructor * and not modified in any way) * * @param type Class of object being serialized or deserialized; * not checked at this point, since it is assumed that unprocessable * classes have been already weeded out, * but will be passed to {@link ContextResolver} as is. * @param mediaType Declared media type for the instance to process: * not used by this method, * but will be passed to {@link ContextResolver} as is. */ @Override public YAMLMapper _locateMapperViaProvider(Class type, MediaType mediaType) { // First: were we configured with a specific instance? YAMLMapper m = _mapperConfig.getConfiguredMapper(); if (m == null) { // If not, maybe we can get one configured via context? if (_providers != null) { ContextResolver resolver = _providers.getContextResolver(YAMLMapper.class, mediaType); /* Above should work as is, but due to this bug * [https://jersey.dev.java.net/issues/show_bug.cgi?id=288] * in Jersey, it doesn't. But this works until resolution of * the issue: */ if (resolver == null) { resolver = _providers.getContextResolver(YAMLMapper.class, null); } if (resolver != null) { m = resolver.getContext(type); } } if (m == null) { // If not, let's get the fallback default instance m = _mapperConfig.getDefaultMapper(); } } return m; } /* /********************************************************** /* Overrides /********************************************************** */ @Override protected JsonParser _createParser(ObjectReader reader, InputStream rawStream) throws IOException { // Fix for [Issue#4]: note, can not try to advance parser, XML parser complains PushbackInputStream wrappedStream = new PushbackInputStream(rawStream); int firstByte = wrappedStream.read(); if (firstByte == -1) { return null; } wrappedStream.unread(firstByte); return reader.getFactory().createParser(wrappedStream); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy