com.stripe.model.EventDataObjectDeserializer Maven / Gradle / Ivy
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);
}
}