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

com.mercadopago.core.MPBase Maven / Gradle / Ivy

package com.mercadopago.core;

import com.google.common.collect.MapDifference;
import com.google.common.collect.Maps;
import com.google.gson.Gson;
import com.google.gson.JsonArray;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import com.google.gson.internal.LinkedTreeMap;
import com.google.gson.reflect.TypeToken;
import com.mercadopago.MercadoPago;
import com.mercadopago.core.annotations.idempotent.Idempotent;
import com.mercadopago.core.annotations.rest.DELETE;
import com.mercadopago.core.annotations.rest.GET;
import com.mercadopago.core.annotations.rest.POST;
import com.mercadopago.core.annotations.rest.PUT;
import com.mercadopago.core.annotations.rest.PayloadType;
import com.mercadopago.exceptions.MPException;
import com.mercadopago.insight.InsightDataManager;
import com.mercadopago.net.HttpMethod;
import com.mercadopago.net.MPRestClient;
import org.apache.commons.lang.StringUtils;

import java.lang.annotation.Annotation;
import java.lang.reflect.AnnotatedElement;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;

/**
 * Mercado Pago MercadoPago
 * Core MPBase class
 *
 * Created by Eduardo Paoletta on 11/4/16.
 */
public abstract class MPBase {

    private transient static final List ALLOWED_METHODS = Arrays.asList("findById", "find", "save", "update", "delete", "partialRefund", "updateReleaseDate", "updateDisbursementReleaseDate", "create");
    private transient static final List ALLOWED_BULK_METHODS = Arrays.asList("all", "search", "createAll");
    private transient static final List METHODS_TO_VALIDATE = Arrays.asList("save", "update");

    public transient static final Boolean WITHOUT_CACHE = Boolean.FALSE;
    public transient static final Boolean WITH_CACHE = Boolean.TRUE;

    protected transient JsonObject _lastKnownJson = null;

    private transient String idempotenceKey = null;
    protected transient MPApiResponse lastApiResponse;

    private transient String marketplaceAccessToken = null;

    private static MPRestClient restClient = new MPRestClient();

    public MPBase() {
        if (admitIdempotenceKey()) {
            this.idempotenceKey = UUID.randomUUID().toString();
        }
    }

    public String getIdempotenceKey() {
        return this.idempotenceKey;
    }

    public  T setIdempotenceKey(String idempotenceKey) throws MPException {
        if (!admitIdempotenceKey()) {
            throw new MPException(this.getClass().getSimpleName() + " does not admit an idempotence key");
        }
        this.idempotenceKey = idempotenceKey;
        return (T) this;
    }

    public MPApiResponse getLastApiResponse() {
        return this.lastApiResponse;
    }

    /**
     * Checks if the class is marked as an idempotent resource
     * @return a boolean
     */
    private boolean admitIdempotenceKey() {
        return (this.getClass().getAnnotation(Idempotent.class) != null);
    }

    /**
     * Process the method to call the api, usually used for create, update and delete methods
     *
     * @param methodName        a String with the decorated method to be processed
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @return                  a resource obj fill with the api response
     * @throws MPException
     */
    protected  T processMethod(String methodName, Boolean useCache) throws MPException {
        return processMethod(methodName, useCache, MPRequestOptions.createDefault());
    }

    /**
     * Process the method to call the api, usually used for create, update and delete methods
     *
     * @param methodName        a String with the decorated method to be processed
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @param requestOptions    a Object with request options that overrides default options
     * @return                  a resourse obj fill with the api response
     * @throws MPException
     */
    protected  T processMethod(String methodName, Boolean useCache, MPRequestOptions requestOptions) throws MPException {
        T resource = processMethod(this.getClass(), (T)this, methodName, null, useCache, requestOptions);
        fillResource(resource, this);
        return (T)this;
    }

    /**
     * Process the method to call the api, usually used for load methods
     *
     * @param clazz             a MPBase extended class
     * @param methodName        a String with the decorated method to be processed
     * @param param1            a String with the arg passed in the call of the method
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @return                  a resourse obj fill with the api response
     * @throws MPException
     */
    protected static  T processMethod(Class clazz, String methodName, String param1, Boolean useCache) throws MPException {
        return processMethod(clazz, methodName, useCache, MPRequestOptions.createDefault(), param1);
    }

    /**
     * Process the method to call the api, usually used for load methods
     *
     * @param clazz             a MPBase extended class
     * @param methodName        a String with the decorated method to be processed
     * @param param1            a String with the arg passed in the call of the method
     * @param param2            a String with the arg passed in the call of the method
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @return                  a resourse obj fill with the api response
     * @throws MPException
     */
    protected static  T processMethod(Class clazz, String methodName, String param1, String param2, Boolean useCache) throws MPException {
        return processMethod(clazz, methodName, useCache, MPRequestOptions.createDefault(), param1, param2);
    }

    /**
     * Process method to call the api, usually used for loadAll and search methods
     *
     * @param clazz                 a MPBase extended class
     * @param methodName            a String with the decorated method to be processed
     * @param useCache              a Boolean flag that indicates if the cache must be used
     * @param requestOptions        a Object with request options that overrides default options
     * @param params                a Array of String with the args passed in the call of the method
     * @return
     * @throws MPException
     */
    protected static  T processMethod(Class clazz, String methodName, Boolean useCache, MPRequestOptions requestOptions, String... params) throws MPException {
        Map mapParams = new HashMap();
        if (params != null) {
            for (int i = 1; i <= params.length; i++) {
                mapParams.put("param" + i, params[i - 1]);
            }
        }

        return processMethod(clazz, null, methodName, mapParams, useCache, requestOptions);
    }

    /**
     * Process the method to call the api
     *
     * @param clazz             a MPBase extended class
     * @param resource          an instance of the MPBase extended class, if its called from a non static method
     * @param methodName        a String with the decorated method to be processed
     * @param mapParams         a hashmap with the args passed in the call of the method
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @return                  a resourse obj fill with the api response
     * @throws MPException
     */
    protected static  T processMethod(Class clazz, T resource, String methodName, Map mapParams, Boolean useCache) throws MPException {
        return processMethod(clazz, resource, methodName, mapParams, useCache, MPRequestOptions.createDefault());
    }

    /**
     * Process the method to call the api
     *
     * @param clazz             a MPBase extended class
     * @param resource          an instance of the MPBase extended class, if its called from a non static method
     * @param methodName        a String with the decorated method to be processed
     * @param mapParams         a hashmap with the args passed in the call of the method
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @param requestOptions    a Object with request options that overrides default options
     * @return                  a resourse obj fill with the api response
     * @throws MPException
     */
    protected static  T processMethod(Class clazz, T resource, String methodName, Map mapParams, Boolean useCache, MPRequestOptions requestOptions) throws MPException {
        if (resource == null) {
            try {
                resource = (T) clazz.newInstance();
            } catch (Exception ex) {
                throw new MPException(ex);
            }
        }

        //Validates the method executed
        if (!ALLOWED_METHODS.contains(methodName)) {
            throw new MPException("Method \"" + methodName + "\" not allowed");
        }

        AnnotatedElement annotatedMethod = getAnnotatedMethod(clazz, methodName);
        Map hashAnnotation = getRestInformation(annotatedMethod);
        HttpMethod httpMethod = (HttpMethod)hashAnnotation.get("method");
        String path = parsePath(hashAnnotation.get("path").toString(), mapParams, resource, requestOptions);

        if (METHODS_TO_VALIDATE.contains(methodName)) {
            // Validator will throw an MPValidatorException, there is no need to do a conditional
            MPValidator.validate(resource);
        }

        PayloadType payloadType = (PayloadType) hashAnnotation.get("payloadType");
        JsonObject payload = generatePayload(httpMethod, resource);

        if (StringUtils.isNotEmpty(resource.getIdempotenceKey()) && !requestOptions.getCustomHeaders().containsKey("x-idempotency-key")) {
            requestOptions.getCustomHeaders().put("x-idempotency-key", resource.getIdempotenceKey());
        }
        //Insight custom header
        requestOptions.getCustomHeaders().put(InsightDataManager.HEADER_X_INSIGHTS_EVENT_NAME, methodName); 

        MPApiResponse response = callApi(httpMethod, path, payloadType, payload, useCache, requestOptions);

        if (response.getStatusCode() >= 200 && response.getStatusCode() < 300) {
            if (httpMethod != HttpMethod.DELETE) {
                resource = fillResourceWithResponseData(resource, response);
            } else {
                resource = cleanResource(resource);
            }
        }

        resource.lastApiResponse = response;

        return resource;
    }

    /**
     * Process method to call the api, usually used for loadAll and search methods
     *
     * @param clazz                 a MPBase extended class
     * @param methodName            a String with the decorated method to be processed
     * @param useCache              a Boolean flag that indicates if the cache must be used
     * @return
     * @throws MPException
     */
    protected static MPResourceArray processMethodBulk(Class clazz, String methodName, Boolean useCache) throws MPException {
        return processMethodBulk(clazz, methodName, null, useCache, MPRequestOptions.createDefault());
    }

    /**
     * Process method to call the api, usually used for loadAll and search methods
     *
     * @param clazz                 a MPBase extended class
     * @param methodName            a String with the decorated method to be processed
     * @param param1                a String with the arg passed in the call of the method
     * @param useCache              a Boolean flag that indicates if the cache must be used
     * @return
     * @throws MPException
     */
    protected static MPResourceArray processMethodBulk(Class clazz, String methodName, String param1, Boolean useCache) throws MPException {
        return processMethodBulk(clazz, methodName, useCache, MPRequestOptions.createDefault(), param1);
    }

    /**
     * Process method to call the api, usually used for loadAll and search methods
     *
     * @param clazz                 a MPBase extended class
     * @param methodName            a String with the decorated method to be processed
     * @param param1                a String with the arg passed in the call of the method
     * @param param2                a String with the arg passed in the call of the method
     * @param useCache              a Boolean flag that indicates if the cache must be used
     * @return
     * @throws MPException
     */
    protected static MPResourceArray processMethodBulk(Class clazz, String methodName, String param1, String param2, Boolean useCache) throws MPException {
        return processMethodBulk(clazz, methodName, useCache, MPRequestOptions.createDefault(), param1, param2);
    }

    /**
     * Process method to call the api, usually used for loadAll and search methods
     *
     * @param clazz                 a MPBase extended class
     * @param methodName            a String with the decorated method to be processed
     * @param useCache              a Boolean flag that indicates if the cache must be used
     * @param requestOptions        a Object with request options that overrides default options
     * @param params                a Array of String with the args passed in the call of the method
     * @return
     * @throws MPException
     */
    protected static MPResourceArray processMethodBulk(Class clazz, String methodName, Boolean useCache, MPRequestOptions requestOptions, String... params) throws MPException {
        Map mapParams = new HashMap();
        if (params != null) {
            for (int i = 1; i <= params.length; i++) {
                mapParams.put("param" + i, params[i - 1]);
            }
        }

        return processMethodBulk(clazz, methodName, mapParams, useCache, requestOptions);
    }

    /**
     * Process the method to call the api
     *
     * @param clazz             a MPBase extended class
     * @param methodName        a String with the decorated method to be processed
     * @param mapParams         a Map with the args passed in the call of the method
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @return                  a resourse obj fill with the api response
     * @throws MPException
     */
    protected static MPResourceArray processMethodBulk(Class clazz, String methodName, Map mapParams, Boolean useCache) throws MPException {
        return processMethodBulk(clazz, methodName, mapParams, useCache, MPRequestOptions.createDefault());
    }

    /**
     * Process the method to call the api
     *
     * @param clazz             a MPBase extended class
     * @param methodName        a String with the decorated method to be processed
     * @param mapParams         a Map with the args passed in the call of the method
     * @param useCache          a Boolean flag that indicates if the cache must be used
     * @param requestOptions    a Object with request options that overrides default options
     * @return                  a resourse obj fill with the api response
     * @throws MPException
     */
    protected static MPResourceArray processMethodBulk(Class clazz, String methodName, Map mapParams, Boolean useCache, MPRequestOptions requestOptions) throws MPException {
        //Validates the method executed
        if (!ALLOWED_BULK_METHODS.contains(methodName)) {
            throw new MPException("Method \"" + methodName + "\" not allowed");
        }

        AnnotatedElement annotatedMethod = getAnnotatedMethod(clazz, methodName);
        Map hashAnnotation = getRestInformation(annotatedMethod);
        HttpMethod httpMethod = (HttpMethod)hashAnnotation.get("method");
        String path = parsePath(hashAnnotation.get("path").toString(), mapParams, null, requestOptions);
        PayloadType payloadType = (PayloadType) hashAnnotation.get("payloadType");

        MPApiResponse response = callApi(httpMethod, path, payloadType, new JsonObject(), useCache, requestOptions);

        MPResourceArray resourceArray = new MPResourceArray();

        if (response.getStatusCode() >= 200 && response.getStatusCode() < 300) {
            resourceArray._resourceArray = fillArrayWithResponseData(clazz, response);
        }

        resourceArray.lastApiResponse = response;

        return resourceArray;

    }

    /**
     * Calls the api and returns an MPApiResponse.
     *
     * @param httpMethod                    the http method to be processed
     * @param path                          a String with the full url of the endpoint
     * @param payloadType                   a PayloadType obj
     * @param payload                       a JsonObject with the content of the payload
     * @param requestOptions                a Object with request options that overrides default options
     * @return
     * @throws MPException
     */
    private static MPApiResponse callApi(
            HttpMethod httpMethod,
            String path,
            PayloadType payloadType,
            JsonObject payload,
            Boolean useCache,
            MPRequestOptions requestOptions) throws MPException {

        String cacheKey = httpMethod.toString() + "_" + path;

        MPApiResponse response = null;
        if (useCache) {
            response = MPCache.getFromCache(cacheKey);
        }
        if (response == null) {
            response = restClient.executeRequest(
                    httpMethod,
                    path,
                    payloadType,
                    payload,
                    requestOptions);
           

            if (useCache) {
                MPCache.addToCache(cacheKey, response);
            } else {
                MPCache.removeFromCache(cacheKey);
            }
        }

        return response;
    }

    /**
     * It fills all the attributes members of the Resource obj.
     * Used when a Get or a Put request is called
     *
     * @param response              Response of the request
     * @throws MPException
     */
    protected static  T fillResourceWithResponseData(T resource, MPApiResponse response) throws MPException {
        if (response.getJsonElementResponse() != null &&
                response.getJsonElementResponse().isJsonObject()) {
            JsonObject jsonObject = (JsonObject) response.getJsonElementResponse();
            T resourceObject = MPCoreUtils.getResourceFromJson(resource.getClass(), jsonObject);
            resource = fillResource(resourceObject, resource);
            resource._lastKnownJson = MPCoreUtils.getJsonFromResource(resource);
        }
        return resource;
    }

    /**
     * It fills an array with the resource objects from the api response
     *
     * @param clazz                 a MPBase extended class
     * @param response              MPApiResponse obj.
     * @param 
     * @return
     * @throws MPException
     */
    protected static  ArrayList fillArrayWithResponseData(Class clazz, MPApiResponse response) throws MPException {
        ArrayList resourceArray = new ArrayList();
        if (response.getJsonElementResponse() != null) {
            JsonArray jsonArray = MPCoreUtils.getArrayFromJsonElement(response.getJsonElementResponse());
            if (jsonArray != null) {
                for (int i = 0; i < jsonArray.size(); i++) {
                    T resource = MPCoreUtils.getResourceFromJson(clazz, (JsonObject) jsonArray.get(i));
                    resource._lastKnownJson = MPCoreUtils.getJsonFromResource(resource);
                    resourceArray.add(resource);
                }
            }
        }
        return resourceArray;
    }

    /**
     * Copies the atributes from an obj to a destination obj
     *
     * @param sourceResource                source resource obj
     * @param destinationResource           destination resource obj
     * @param 
     * @return
     * @throws MPException
     */
    private static  T fillResource(T sourceResource, T destinationResource) throws MPException {
        Field[] declaredFields = destinationResource.getClass().getDeclaredFields();
        for (Field field : declaredFields) {
            try {
                Field originField = sourceResource.getClass().getDeclaredField(field.getName());
                field.setAccessible(true);
                originField.setAccessible(true);
                field.set(destinationResource, originField.get(sourceResource));

            } catch (Exception ex) {
                throw new MPException(ex);
            }
        }
        return destinationResource;
    }

    /**
     * Removes all data from the attributes members of the Resource obj.
     * Used when a delete request is called
     *
     * @throws MPException
     */
    private static  T cleanResource(T resource) throws MPException {
        Field[] declaredFields = resource.getClass().getDeclaredFields();

        for (Field field : declaredFields) {
            try {
                field.setAccessible(true);
                field.set(resource, null);

            } catch (Exception ex) {
                throw new MPException(ex);
            }
        }
        return resource;
    }

    /**
     * Evaluates the path of the resourse and use the args or the attributes members of the instance to complete it.
     * @param path              a String with the path as stated in the declaration of the method caller
     * @param mapParams         a HashMap with the args passed in the call of the method
     * @param resource          a resource object
     * @param requestOptions    a Object with request options that overrides default options
     * @return                  a String with the final path to call the API
     * @throws MPException
     */
    private static  String parsePath(String path, Map mapParams, T resource, MPRequestOptions requestOptions) throws MPException {
        StringBuilder processedPath = new StringBuilder();
        if (path.contains(":")) {
            int paramIterator = 0;
            while (path.contains(":")) {
                paramIterator++;

                processedPath.append(path.substring(0, path.indexOf(":")));
                path = path.substring(path.indexOf(":") + 1);
                String param = path;
                if (path.contains("/")) {
                    param = path.substring(0, path.indexOf("/"));
                }

                String value = null;
                if (paramIterator <= 2 && mapParams != null &&
                        StringUtils.isNotEmpty(mapParams.get("param" + String.valueOf(paramIterator)))) {
                    value = mapParams.get("param" + String.valueOf(paramIterator));
                    mapParams.remove(param);
                } else if (mapParams != null && StringUtils.isNotEmpty(mapParams.get(param))) {
                    value = mapParams.get(param);
                    mapParams.remove(param);
                } else {
                    if (resource != null) {
                        JsonObject json = MPCoreUtils.getJsonFromResource(resource);
                        if (json.get(param) != null) {
                            value = json.get(param).getAsString();
                        } else { // Search in no serialized properties
                            Class aClass = resource.getClass();
                            String paramAccessorName = null;
                            paramAccessorName = MPCoreUtils.toCamelCase("get_" + param);

                            try {
                                Method method = aClass.getMethod(paramAccessorName);
                                value = method.invoke(resource).toString();
                            } catch (NoSuchMethodException e) {
                                // Accessor method not found
                            } catch (InvocationTargetException e) {
                                // Do nothing
                            } catch (IllegalAccessException e) {
                                // Accessor is private
                            }

                        }
                    }
                }

                if (StringUtils.isEmpty(value)) {
                    throw new MPException("No argument supplied/found for method path");
                }

                processedPath.append(value);
                if (path.contains("/")) {
                    path = path.substring(path.indexOf("/"));
                } else {
                    path = "";
                }
            }
            if (StringUtils.isNotEmpty(path)) {
                processedPath.append(path);
            }
        } else {
            processedPath.append(path);
        }

        // URL
        processedPath.insert(0, MercadoPago.SDK.getBaseUrl());

        // Token
        String accessToken;
        if (StringUtils.isNotEmpty(requestOptions.getAccessToken())) {
            accessToken = requestOptions.getAccessToken();
        }else if (resource != null) {
            accessToken = resource.getMarketplaceAccessToken();
            if (StringUtils.isEmpty(accessToken)) {
                accessToken = MercadoPago.SDK.getAccessToken();
            }
        } else {
            accessToken = MercadoPago.SDK.getAccessToken();
        }

        if(!path.equals("/oauth/token")) {
            processedPath
                    .append("?access_token=")
                    .append(accessToken);
        }
        
        if (mapParams != null && !mapParams.isEmpty()) {
	        for (Map.Entry entry : mapParams.entrySet()) {
	            String key = entry.getKey();
	            String value = entry.getValue();
	            if (StringUtils.isNotEmpty(value)) {
                    processedPath.append("&").append(key).append("=").append(value);
                }
	        }
        }

        if (!MPCoreUtils.validateUrl(processedPath.toString())) {
            throw new MPException("Processed URL not valid: " + processedPath.toString());
        }
        return processedPath.toString();
    }

    /**
     * Transforms all attributes members of the instance in a JSON String. Only for POST and PUT methods.
     * POST gets the full object in a JSON object.
     * PUT gets only the differences with the last known state of the object.
     *
     * @return                  a JSON Object with the attributes members of the instance. Null for GET and DELETE methods
     */
    private static  JsonObject generatePayload(HttpMethod httpMethod, T resource) {
        JsonObject payload = null;
        if (httpMethod.equals(HttpMethod.POST) ||
                (httpMethod.equals(HttpMethod.PUT) && resource._lastKnownJson == null)) {
            payload = MPCoreUtils.getJsonFromResource(resource);
        } else if (httpMethod.equals(HttpMethod.PUT)) {

            JsonObject actualJson = MPCoreUtils.getJsonFromResource(resource);

            Type mapType = new TypeToken>(){}.getType();
            Gson gson = new Gson();
            Map oldMap = gson.fromJson(resource._lastKnownJson, mapType);
            Map newMap = gson.fromJson(actualJson, mapType);
            MapDifference mapDifferences = Maps.difference(oldMap, newMap);

            payload = new JsonObject();



            for (Map.Entry> entry : mapDifferences.entriesDiffering().entrySet()) {

                if (entry.getValue().rightValue() instanceof LinkedTreeMap) {
                    JsonElement jsonObject = gson.toJsonTree(entry.getValue().rightValue()).getAsJsonObject();
                    payload.add(entry.getKey(), jsonObject);
                } else {
                    if (entry.getValue().rightValue() instanceof Boolean) {
                        payload.addProperty(entry.getKey(), (Boolean)entry.getValue().rightValue());
                    } else if (entry.getValue().rightValue() instanceof Number) {
                        payload.addProperty(entry.getKey(), (Number)entry.getValue().rightValue());
                    } else {
                        payload.addProperty(entry.getKey(), entry.getValue().rightValue().toString());
                    }
                }

            }
            for (Map.Entry entry : mapDifferences.entriesOnlyOnRight().entrySet()) {

                if (entry.getValue() instanceof Boolean) {
                    payload.addProperty(entry.getKey(), (Boolean)entry.getValue());
                } else if (entry.getValue() instanceof Number) {
                    payload.addProperty(entry.getKey(), (Number)entry.getValue());
                } else {
                    payload.addProperty(entry.getKey(), entry.getValue().toString());
                }
            }
        }
        return payload;
    }

    /**
     * Iterates the annotations of the entity method implementation, it validates that only one method annotation
     * is used in the entity implementation method.
     *
     * @param element           The annotated method of the entity
     * @return                  a hashmap with keys 'method' and 'path'
     * @throws MPException
     */
    private static Map getRestInformation(AnnotatedElement element) throws MPException{
        if (element.getAnnotations().length == 0) {
            throw new MPException("No rest method found");
        }

        Map hashAnnotation = new HashMap();
        for (Annotation annotation : element.getAnnotations()) {
            if (annotation instanceof DELETE) {
                DELETE delete = (DELETE) annotation;
                if (StringUtils.isEmpty(delete.path())) {
                    throw new MPException("Path not found for DELETE method");
                }
                hashAnnotation = fillHashAnnotations(
                        hashAnnotation,
                        HttpMethod.DELETE,
                        delete.path(),
                        null,
                        delete.retries(),
                        delete.connectionTimeout(),
                        delete.connectionRequestTimeout(),
                        delete.socketTimeout());

            } else if (annotation instanceof GET) {
                GET get = (GET) annotation;
                if (StringUtils.isEmpty(get.path())) {
                    throw new MPException("Path not found for GET method");
                }
                hashAnnotation = fillHashAnnotations(
                        hashAnnotation,
                        HttpMethod.GET,
                        get.path(),
                        null,
                        get.retries(),
                        get.connectionTimeout(),
                        get.connectionRequestTimeout(),
                        get.socketTimeout());

            } else if (annotation instanceof POST) {
                POST post = (POST) annotation;
                if (StringUtils.isEmpty(post.path())) {
                    throw new MPException("Path not found for POST method");
                }
                hashAnnotation = fillHashAnnotations(
                        hashAnnotation,
                        HttpMethod.POST,
                        post.path(),
                        post.payloadType(),
                        post.retries(),
                        post.connectionTimeout(),
                        post.connectionRequestTimeout(),
                        post.socketTimeout());

            } else if (annotation instanceof PUT) {
                PUT put = (PUT) annotation;
                if (StringUtils.isEmpty(put.path())) {
                    throw new MPException("Path not found for PUT method");
                }
                hashAnnotation = fillHashAnnotations(
                        hashAnnotation,
                        HttpMethod.PUT,
                        put.path(),
                        put.payloadType(),
                        put.retries(),
                        put.connectionTimeout(),
                        put.connectionRequestTimeout(),
                        put.socketTimeout());
            }
        }
        return hashAnnotation;
    }

    /**
     * Fills a hashmap with the rest method and the path to call the api. It validates that no other method/path is filled in the hash
     *
     * @param hashAnnotation        a HashMap object that will contain the method and path
     * @param method                a String with the method
     * @param path                  a String with the path
     * @param payloadType           a PayloadType enum
     * @param retries               int with the retries for the api request
     * @param connectionTimeout     int with the connection timeout for the api request expressed in milliseconds
     * @param socketTimeout         int with the socket timeout for the api request expressed in milliseconds
     * @return                      the HashMap object that is received by param
     * @throws MPException
     */
    private static Map fillHashAnnotations(
            Map hashAnnotation,
            HttpMethod method,
            String path,
            PayloadType payloadType,
            int retries,
            int connectionTimeout,
            int connectionRequestTimeout,
            int socketTimeout)
            throws MPException {
        if (hashAnnotation.containsKey("method")) {
            throw new MPException("Multiple rest methods found");
        }
        hashAnnotation.put("method", method);
        hashAnnotation.put("path", path);
        hashAnnotation.put("payloadType", payloadType);
        hashAnnotation.put("retries", retries);
        hashAnnotation.put("connectionTimeout", connectionTimeout);
        hashAnnotation.put("connectionRequestTimeout", connectionRequestTimeout);
        hashAnnotation.put("socketTimeout", socketTimeout);
        return hashAnnotation;
    }

    /**
     * Iterates over the methods of a class and returns the one matching the method name passed
     *
     * @param methodName            a String with the name of the method to be recuperated
     * @return                      a AnnotatedMethod that match the method name passed
     * @throws MPException
     */
    private static AnnotatedElement getAnnotatedMethod(Class clazz, String methodName) throws MPException {
        for (Method method : clazz.getDeclaredMethods()) {
            if (method.getName().equals(methodName) && method.getDeclaredAnnotations().length > 0) {
                return method;
            }
        }
        throw new MPException("No annotated method found");
    }

    public String getMarketplaceAccessToken() {
        return marketplaceAccessToken;
    }

    public void setMarketplaceAccessToken(String marketplaceAccessToken) {
        this.marketplaceAccessToken = marketplaceAccessToken;
    }

    protected void addTrackingHeaders(MPRequestOptions requestOptions) {
        Map customHeaders = requestOptions.getCustomHeaders();
        if (MercadoPago.SDK.getPlatformId() != null && !MercadoPago.SDK.getPlatformId().trim().equals("") && !customHeaders.containsKey("x-platform-id")) {
            customHeaders.put("x-platform-id", MercadoPago.SDK.getPlatformId());
        }
        if (MercadoPago.SDK.getCorporationId() != null && !MercadoPago.SDK.getCorporationId().trim().equals("") && !customHeaders.containsKey("x-corporation-id")) {
            customHeaders.put("x-corporation-id", MercadoPago.SDK.getCorporationId());
        }
        if (MercadoPago.SDK.getIntegratorId() != null && !MercadoPago.SDK.getIntegratorId().trim().equals("") && !customHeaders.containsKey("x-integrator-id")) {
            customHeaders.put("x-integrator-id", MercadoPago.SDK.getIntegratorId());
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy