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

com.stripe.model.EventDataObjectDeserializer Maven / Gradle / Ivy

There is a newer version: 28.2.0
Show newest version
package com.stripe.model;

import com.google.gson.JsonObject;
import com.google.gson.JsonParseException;
import com.stripe.Stripe;
import com.stripe.exception.EventDataObjectDeserializationException;
import com.stripe.net.ApiMode;
import com.stripe.net.StripeResponseGetter;
import java.util.Map;
import java.util.Optional;
import lombok.EqualsAndHashCode;

/**
 * Deserialization helper to get {@code StripeObject} and handle failure due to schema
 * incompatibility between the data object and the model classes. Event data object by default
 * corresponds to the schema at API version tied to your Stripe account at the event creation time.
 * This event version is in {@link Event#getApiVersion()}. The model classes for deserialization,
 * however, corresponds to a specific version pinned to this library {@link Stripe#API_VERSION}.
 * Thus, only data object with same API versions is guaranteed to deserialize safely.
 *
 * 

To avoid this problem of API version mismatch, create a new webhook endpoint `api_versions` * corresponding to {@link Stripe#API_VERSION}. For more information, see API reference * *

In practice, each API version update only affects specific set of classes, so event data * object for the unaffected classes can still be serialized successfully -- even when the API * versions do not match. (Although it is considered unsafe by the API version comparison.) In that * case, you can use {@link EventDataObjectDeserializer#deserializeUnsafe()} * *

Old events from {@link Event#retrieve(String)} or {@link Event#list(Map)} will have immutable * API versions on them, and there is currently no support for rendering it at different API * versions. If you find failure from reading these events, consider defining your own custom {@link * CompatibilityTransformer} to transform the raw JSON to one with schema compatible with this * current model classes. * *

En event integration from webhook may look like the example below. Assuming that you have the * event api version matching this library, you should safely find deserialized object from the * deserializer. * *

 *   Event event = Webhook.constructEvent(payload, sigHeader, secret);
 *   EventDataObjectDeserializer dataObjectDeserializer = event.getDataObjectDeserializer();
 *   if (dataObjectDeserializer.getObject().isPresent()) {
 *     StripeObject stripeObject = dataObjectDeserializer.getObject().get();
 *     doSomething(stripeObject);
 *   } else {
 *     throw new IllegalStateException(
 *       String.format("Unable to deserialize event data object for %s", event));
 *   }
 * 
*/ @EqualsAndHashCode(callSuper = false) public class EventDataObjectDeserializer { /** API version of the event data object. */ String apiVersion; /** Event type to which this event data object belongs to. */ String eventType; /** Raw JSON response to be deserialized into {@code StripeObject}. */ JsonObject rawJsonObject; /** Deserialized {@code StripeObject} set during successful/safe deserialization. */ private StripeObject object; private final StripeResponseGetter responseGetter; EventDataObjectDeserializer( String apiVersion, String eventType, JsonObject rawJsonObject, StripeResponseGetter responseGetter) { this.apiVersion = apiVersion; this.rawJsonObject = rawJsonObject; this.eventType = eventType; this.responseGetter = responseGetter; } /** * Gets an {@code Optional} of data event object. When the optional is present, the deserialized * {@code StripeObject} preserves high data integrity because of correspondence between schema of * the PI response and the model class (the underlying concrete class for abstract {@code * StripeObject}) schema. This is when {@link Event#getApiVersion()} matches {@link * Stripe#API_VERSION}. Otherwise, the optional is empty. * * @return {@code Optional} of stripe object when deserialization is safe. */ public Optional getObject() { if (object != null) { return Optional.of(object); } if (deserialize()) { return Optional.of(object); } else { return Optional.empty(); } } /** * Get raw JSON string for the data object. This is the same data available in {@link * EventDataObjectDeserializationException#getRawJson()} upon deserialization failure. * * @return JSON string the event data object. */ public String getRawJson() { return rawJsonObject.toString(); } /** * Safe deserialize raw JSON into {@code StripeObject}. This operation mutates the state, and the * successful result can be accessed via {@link EventDataObjectDeserializer#getObject()}. Matching * {@link Event#getApiVersion()} and {@link Stripe#API_VERSION} is necessary condition to * guarantee safe deserialization. * * @return whether deserialization has been successful. */ private boolean deserialize() { if (!apiVersionMatch()) { // when version mismatch, even when deserialization is successful, // we cannot guarantee data correctness. Old events containing fields that should be // translated/mapped to the new schema will simply not be captured by the new schema. return false; } else if (object != null) { // already successfully deserialized return true; } else { try { object = StripeObject.deserializeStripeObject(rawJsonObject, this.responseGetter, ApiMode.V1); return true; } catch (JsonParseException e) { // intentionally ignore exception to fulfill simply whether deserialization succeeds return false; } } } /** * Force deserialize raw JSON to {@code StripeObject}. The deserialized data is not guaranteed to * fully represent the JSON. For example, events of new API version having fields that are not * captured by current model class will be lost. Similarly, events of old API version having * fields that should be translated into the new fields, like field rename, will be lost. * *

Upon deserialization failure, consider making the JSON compatible to the current model * classes and recover from failure with {@link * EventDataObjectDeserializer#deserializeUnsafeWith(CompatibilityTransformer)}. * * @return Object with no guarantee on full representation of its original raw JSON response. * @throws EventDataObjectDeserializationException exception that contains the message error and * the raw JSON response of the {@code StripeObject} to be deserialized. */ public StripeObject deserializeUnsafe() throws EventDataObjectDeserializationException { try { return StripeObject.deserializeStripeObject(rawJsonObject, this.responseGetter, ApiMode.V1); } catch (JsonParseException e) { String errorMessage; if (!apiVersionMatch()) { errorMessage = String.format( "Current `stripe-java` integration has Stripe API version %s, but the event data " + "object has %s. The JSON data might have schema not compatible with the " + "current model classes; such incompatibility can be the cause of " + "deserialization failure. " + "If you are deserializing webhoook events, consider creating a different webhook " + "endpoint with `api_version` at %s. See Stripe API reference for more details. " + "If you are deserializing old events from `Event#retrieve`, " + "consider transforming the raw JSON data object to be compatible with this " + "current model class schemas using `deserializeUnsafeWith`. " + "Original error message: %s", getIntegrationApiVersion(), this.apiVersion, getIntegrationApiVersion(), e.getMessage()); } else { errorMessage = String.format( "Unable to deserialize event data object to respective Stripe " + "object. Please see the raw JSON, and contact [email protected] for " + "assistance. Original error message: %s", e.getMessage()); } throw new EventDataObjectDeserializationException(errorMessage, rawJsonObject.toString()); } } /** * Deserialize JSON that has been processed by {@link * CompatibilityTransformer#transform(JsonObject, String, String)} into {@code StripeObject}. This * deserialization method should only be used to handle events with schema incompatible to model * class schema of this library. Throws {@link JsonParseException} when the transformed JSON * remains incompatible with the model classes. * * @return deserialized {@code StripeObject} from user-supplied compatible JSON. */ public StripeObject deserializeUnsafeWith(CompatibilityTransformer transformer) { return StripeObject.deserializeStripeObject( transformer.transform(rawJsonObject.deepCopy(), apiVersion, eventType), this.responseGetter, ApiMode.V1); } private boolean apiVersionMatch() { // Preserved for testing; we have tests that hook getIntegrationApiVersion // to test with other api versions. String currentApiVersion = getIntegrationApiVersion(); if (!currentApiVersion.contains(".")) { return this.apiVersion.equals(currentApiVersion); } // If the event api version is from before we started adding // a major release identifier, there's no way its compatible with this // version if (!this.apiVersion.contains(".")) { return false; } // versions are yyyy-MM-dd.releaseIdentifier String eventReleaseTrain = this.apiVersion.split("\\.", 2)[1]; String currentReleaseTrain = getIntegrationApiVersion().split("\\.", 2)[1]; return eventReleaseTrain.equals(currentReleaseTrain); } /** Internal method to allow for testing with different Stripe version. */ String getIntegrationApiVersion() { return Stripe.API_VERSION; } /** * Definition of event data object JSON transformation to be compatible to API version of the * library. */ public interface CompatibilityTransformer { /** * Transform event data object JSON into a schema compatible with model classes of the library. * When used in {@link * EventDataObjectDeserializer#deserializeUnsafeWith(CompatibilityTransformer)}. the resulting * JSON will be deserialized to {@code StripeObject}. * * @param rawJsonObject event data object JSON to be transformed. Direct mutation is allowed. * @param apiVersion API version of the event data object * @param eventType event type to which this event data object belongs to. * @return transformed JSON with schema compatible to the model class in this library. */ JsonObject transform(JsonObject rawJsonObject, String apiVersion, String eventType); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy