com.stripe.net.ApiRequestParamsConverter Maven / Gradle / Ivy
package com.stripe.net;
import com.google.gson.FieldNamingPolicy;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import com.google.gson.TypeAdapter;
import com.google.gson.TypeAdapterFactory;
import com.google.gson.reflect.TypeToken;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.JsonWriter;
import com.stripe.Stripe;
import com.stripe.param.common.EmptyParam;
import java.io.IOException;
import java.util.Map;
/**
* Converter to map an api request object to an untyped map. It is not called a *Serializer because
* the outcome is not a JSON data. It is not called *UntypedMapDeserializer because it is not
* converting from JSON.
*/
class ApiRequestParamsConverter {
/** Strategy to flatten extra params in the API request parameters. */
private static class ExtraParamsFlatteningStrategy implements UntypedMapDeserializer.Strategy {
@Override
public void deserializeAndTransform(
Map outerMap,
Map.Entry jsonEntry,
UntypedMapDeserializer untypedMapDeserializer) {
String key = jsonEntry.getKey();
JsonElement jsonValue = jsonEntry.getValue();
if (ApiRequestParams.EXTRA_PARAMS_KEY.equals(key)) {
if (!jsonValue.isJsonObject()) {
throw new IllegalStateException(
String.format(
"Unexpected schema for extra params. JSON object is expected at key `%s`, but found"
+ " `%s`. This is likely a problem with this current library version `%s`. "
+ "Please contact [email protected] for assistance.",
ApiRequestParams.EXTRA_PARAMS_KEY, jsonValue, Stripe.VERSION));
}
// JSON value now corresponds to the extra params map, and is also deserialized as a map.
// Instead of putting this result map under the original key, flatten the map
// by adding all its key/value pairs to the outer map instead.
Map extraParamsMap =
untypedMapDeserializer.deserialize(jsonValue.getAsJsonObject());
for (Map.Entry entry : extraParamsMap.entrySet()) {
validateDuplicateKey(outerMap, entry.getKey(), entry.getValue());
outerMap.put(entry.getKey(), entry.getValue());
}
} else {
Object value = untypedMapDeserializer.deserializeJsonElement(jsonValue);
validateDuplicateKey(outerMap, key, value);
// Normal deserialization where output map has the same structure as the given JSON content.
// The deserialized content is an untyped `Object` and added to the outer map at the
// original key.
outerMap.put(key, value);
}
}
}
private static void validateDuplicateKey(
Map outerMap, String paramKey, Object paramValue) {
if (outerMap.containsKey(paramKey)) {
throw new IllegalArgumentException(
String.format(
"Found multiple param values for the same param key. This can happen because you passed "
+ "additional parameters via `putExtraParam` that conflict with the existing params. "
+ "Found param key `%s` with values `%s` and `%s`. "
+ "If you wish to pass additional params for nested parameters, you "
+ "should add extra params at the nested params themselves, not from the "
+ "top-level param.",
paramKey, outerMap.get(paramKey), paramValue));
}
}
/**
* Type adapter to convert an empty enum to null value to comply with the lower-lever encoding
* logic for the API request parameters.
*/
private static class HasEmptyEnumTypeAdapterFactory implements TypeAdapterFactory {
@SuppressWarnings("unchecked")
@Override
public TypeAdapter create(Gson gson, TypeToken type) {
if (!ApiRequestParams.EnumParam.class.isAssignableFrom(type.getRawType())) {
return null;
}
TypeAdapter paramEnum =
new TypeAdapter() {
@Override
public void write(JsonWriter out, ApiRequestParams.EnumParam value) throws IOException {
if (value.getValue().equals("")) {
// need to restore serialize null setting
// not to affect other fields
boolean previousSetting = out.getSerializeNulls();
out.setSerializeNulls(true);
out.nullValue();
out.setSerializeNulls(previousSetting);
} else {
out.value(value.getValue());
}
}
@Override
public ApiRequestParams.EnumParam read(JsonReader in) {
throw new UnsupportedOperationException(
"No deserialization is expected from this private type adapter for enum param.");
}
};
return (TypeAdapter) paramEnum.nullSafe();
}
}
private static final Gson GSON =
new GsonBuilder()
.setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES)
.registerTypeAdapterFactory(
new ApiRequestParamsConverter.HasEmptyEnumTypeAdapterFactory())
.create();
private static final UntypedMapDeserializer FLATTENING_EXTRA_PARAMS_DESERIALIZER =
new UntypedMapDeserializer(new ExtraParamsFlatteningStrategy());
/**
* Convert the given request params into an untyped map. This map is composed of {@code
* Map}, {@code List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy