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

com.couchbase.client.java.codec.JacksonJsonSerializer Maven / Gradle / Ivy

There is a newer version: 3.7.5
Show newest version
/*
 * Copyright (c) 2019 Couchbase, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.couchbase.client.java.codec;

// CHECKSTYLE:OFF IllegalImport - Allow unbundled Jackson

import com.couchbase.client.core.annotation.Stability;
import com.couchbase.client.core.encryption.CryptoManager;
import com.couchbase.client.core.error.DecodingFailureException;
import com.couchbase.client.core.error.EncodingFailureException;
import com.couchbase.client.java.encryption.annotation.Encrypted;
import com.couchbase.client.java.encryption.databind.jackson.EncryptionModule;
import com.couchbase.client.java.json.JsonObject;
import com.couchbase.client.java.json.JsonValueModule;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.ObjectMapper;

import static com.couchbase.client.core.logging.RedactableArgument.redactUser;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.Objects.requireNonNull;

/**
 * A serializer backed by a user-provided Jackson {@code ObjectMapper}.
 * 

* In order to use this class you must add Jackson to your class path. *

* Make sure to register {@link JsonValueModule} with your {@code ObjectMapper} * so it can handle Couchbase {@link JsonObject} instances. *

* Likewise, if you're using the {@link Encrypted} annotation for * Couchbase Field-Level Encryption, make sure to register * {@link EncryptionModule}. *

* Example usage without Couchbase Field-Level Encryption: *

 * ObjectMapper mapper = new ObjectMapper();
 * mapper.registerModule(new JsonValueModule());
 *
 * ClusterEnvironment env = ClusterEnvironment.builder()
 *     .jsonSerializer(new JacksonJsonSerializer(mapper))
 *     .build();
 * 
*

* Example usage with Couchbase Field-Level Encryption: *

 * CryptoManager cryptoManager = ...
 *
 * ObjectMapper mapper = new ObjectMapper();
 * mapper.registerModule(new JsonValueModule());
 * mapper.registerModule(new EncryptionModule(cryptoManager));
 *
 * ClusterEnvironment env = ClusterEnvironment.builder()
 *     .cryptoManager(cryptoManager)
 *     .jsonSerializer(new JacksonJsonSerializer(mapper))
 *     .build();
 * 
* * @see JsonValueModule * @see EncryptionModule */ public class JacksonJsonSerializer implements JsonSerializer { private final ObjectMapper mapper; /** * Returns a new instance backed by the given ObjectMapper. * * @param mapper the custom ObjectMapper to use. * @return the Jackson JSON serializer with a custom object mapper. */ public static JacksonJsonSerializer create(ObjectMapper mapper) { return new JacksonJsonSerializer(mapper); } /** * Returns a new instance backed by a default ObjectMapper without encryption support. * * @return the Jackson JSON serializer with the default object mapper. */ public static JacksonJsonSerializer create() { return create((CryptoManager) null); } /** * Returns a new instance backed by a default ObjectMapper * with optional encryption support. * * @param cryptoManager (nullable) The manager to use for activating the * {@link Encrypted} annotation, or null to disable encryption support. @return the Jackson JSON serializer with a provided crypto manager. */ public static JacksonJsonSerializer create(CryptoManager cryptoManager) { ObjectMapper mapper = new ObjectMapper(); mapper.registerModule(new JsonValueModule()); if (cryptoManager != null) { mapper.registerModule(new EncryptionModule(cryptoManager)); } return new JacksonJsonSerializer(mapper); } private JacksonJsonSerializer(ObjectMapper mapper) { this.mapper = requireNonNull(mapper); } @Override public byte[] serialize(final Object input) { if (input instanceof byte[]) { return (byte[]) input; } try { return mapper.writeValueAsBytes(input); } catch (Throwable t) { throw new EncodingFailureException("Serializing of content + " + redactUser(input) + " to JSON failed.", t); } } @Override public T deserialize(final Class target, final byte[] input) { if (target.equals(byte[].class)) { return (T) input; } try { return mapper.readValue(input, target); } catch (Throwable e) { throw new DecodingFailureException("Deserialization of content into target " + target + " failed; encoded = " + redactUser(new String(input, UTF_8)), e); } } @Override public T deserialize(final TypeRef target, final byte[] input) { try { JavaType type = mapper.getTypeFactory().constructType(target.type()); return mapper.readValue(input, type); } catch (Throwable e) { throw new DecodingFailureException("Deserialization of content into target " + target + " failed; encoded = " + redactUser(new String(input, UTF_8)), e); } } /** * Throws something if the user-provided Jackson library is absent or broken. * * @throws Throwable if JacksonJsonSerializer should not be used. * @implNote Looking for ObjectMapper on the class path is not sufficient; * we've seen at least one case where ObjectMapper was present but its * dependencies were missing. */ @SuppressWarnings("RedundantThrows") @Stability.Internal public static void preflightCheck() throws Throwable { JsonSerializer serializer = JacksonJsonSerializer.create(); PreflightCheckSubject serializeMe = new PreflightCheckSubject(); serializeMe.name = "x"; byte[] json = serializer.serialize(serializeMe); String expected = "{\"n\":\"x\"}"; String actual = new String(json, UTF_8); if (!expected.equals(actual)) { throw new RuntimeException("Serialization failed; expected " + expected + " but got " + actual); } // could use PreflightCheckSubject.class instead of TypeRef, but let's test the exotic stuff PreflightCheckSubject deserialized = serializer.deserialize(new TypeRef() { }, json); if (!"x".equals(deserialized.name)) { throw new RuntimeException("Deserialization failed; expected 'x' but got '" + deserialized.name + "'"); } } // public in case there's something about the runtime environment // that prevents Jackson from accessing private classes. @Stability.Internal public static class PreflightCheckSubject { @JsonProperty("n") public String name; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy