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

com.cognite.client.servicesV1.response.RequestParametersResponseParser Maven / Gradle / Ivy

/*
 * Copyright (c) 2020 Cognite AS
 *
 * 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.cognite.client.servicesV1.response;

import com.cognite.client.Request;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.auto.value.AutoValue;
import com.google.common.collect.ImmutableList;
import org.apache.commons.lang3.RandomStringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.io.Serializable;
import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Optional;

/**
 * Parses a Json payload and generates a {@link Request} object containing the extracted parameters.
 *
 * The parser will look for Json parameters that match the attribute map. The parameters
 * are added to the {@code RequestParameters} object as root parameters.
 *
 * The attribute map defines {@code json node} -> {@code RequestParameters root node}. The {@code key} specifies which
 * json node to extract, and the corresponding {@code value} specifies the {@link Request} root parameter name.
 * Example:
 * {@code {"jobId", "id"}} will extract the json value from {@code root.jobId} and maps it to the {@link Request}
 * parameter "id".
 *
 * If you specify '*' as a key in the attribute map, all root attributes that are value nodes (i.e. not container
 * nodes) will be added to the {@link Request}.
 *
 * A cursor is never returned from this parser.
 */
@AutoValue
public abstract class RequestParametersResponseParser implements ResponseParser, Serializable {
    private static final int MAX_LENGTH_JSON_LOG = 500;

    private final Logger LOG = LoggerFactory.getLogger(this.getClass());
    private final ObjectMapper objectMapper = new ObjectMapper();
    private final String instanceId = RandomStringUtils.randomAlphanumeric(6);

    static Builder builder() {
        return new AutoValue_RequestParametersResponseParser.Builder();
    }

    public static RequestParametersResponseParser of(Map attributeMap) {
        return RequestParametersResponseParser.builder()
                .setAttributeMap(attributeMap)
                .setItemMode(false)
                .build();
    }

    abstract Builder toBuilder();

    abstract Map getAttributeMap();
    abstract boolean isItemMode();

    /**
     * Enables/disables item mode.
     *
     * In item mode, the {@link Request} will be populated with values within an items array. This is a
     * common request pattern for performing per-item requests towards the Cognite API.
     *
     * Example:
     * Given the attribute map {@code {"jobId", "id"}}.
     * In non-item mode this will extract the json value
     * from {@code root.jobId} and map it to the {@link Request} parameter {@code root.id}.
     *
     * In item mode, this will extract the json value from {@code root.jobId} and map it to {@code root.items[n].id}
     *
     * Default is {@code false}.
     * @param enable
     * @return
     */
    public RequestParametersResponseParser enableItemMode(boolean enable) {
        return toBuilder().setItemMode(enable).build();
    }

    /**
     * Extract the next cursor from a response body.
     *
     * @param payload The response body
     * @return
     * @throws Exception
     */
    public Optional extractNextCursor(byte[] payload) throws Exception {
        return Optional.empty();
    }

    /**
     * Extract the main items from a results json payload. Returns the entire payload as a json string.
     *
     * @param payload The results byte payload
     * @return
     * @throws IOException
     */
    public ImmutableList extractItems(byte[] payload) throws Exception {
        String loggingPrefix = "RequestParametersResponseParser.extractItems - " + instanceId + " - ";
        String json = parseToString(payload);
        Request requestParameters = Request.create();

        LOG.debug(loggingPrefix + "Parsing Json payload: \r\n" + json
                .substring(0, Math.min(MAX_LENGTH_JSON_LOG, json.length())));
        LOG.debug(loggingPrefix + "Start extracting {} parameters", getAttributeMap().size());
        // Check if the input string is valid json. Will throw an exception if it cannot be parsed.
        objectMapper.readTree(json);

        if (isItemMode()) {
            Map item = new HashMap<>();
            for (Map.Entry entry : getAttributeMap().entrySet()) {
                if (entry.getKey().equalsIgnoreCase("*")) {
                    item = extractAllRootNodes(json, item);
                } else {
                    item = extractNode(json, entry.getKey(), item, entry.getValue());
                }
            }
            requestParameters = requestParameters
                    .withItems(ImmutableList.of(item));
        } else {
            for (Map.Entry entry : getAttributeMap().entrySet()) {
                if (entry.getKey().equalsIgnoreCase("*")) {
                    requestParameters = extractAllRootNodes(json, requestParameters);
                } else {
                    requestParameters = extractNode(json, entry.getKey(), requestParameters, entry.getValue());
                }
            }
        }

        LOG.debug(loggingPrefix + "Built request parameters: {}", requestParameters);
        return ImmutableList.of(requestParameters);
    }

    /*
    Extracts all root nodes and puts them on the root of the {@link RequestParameters}.
     */
    private Request extractAllRootNodes(String json,
                                        Request requestParameters) throws Exception {
        Request returnValue = requestParameters;

        for (Iterator it = objectMapper.readTree(json).fieldNames(); it.hasNext(); ) {
            String nodeName = it.next();
            returnValue = extractNode(json, nodeName, returnValue, nodeName);
        }

        return returnValue;
    }

    /*
    Extracts all root nodes and puts them on a Map. Will be a building block for item based requests.
     */
    private Map extractAllRootNodes(String json,
                                                    Map map) throws Exception {

        for (Iterator it = objectMapper.readTree(json).fieldNames(); it.hasNext(); ) {
            String nodeName = it.next();
            extractNode(json, nodeName, map, nodeName);
        }

        return map;
    }

    /*
    Extract the value from the specified json node and add it as a parameter to the RequestParameters.
     */
    private Request extractNode(String json,
                                String path,
                                Request requestParameters,
                                String parameterName) throws Exception {
        JsonNode node = objectMapper.readTree(json).path(path);

        if (node.isIntegralNumber()) {
            return requestParameters.withRootParameter(parameterName, node.longValue());
        } else if (node.isFloatingPointNumber()) {
            return requestParameters.withRootParameter(parameterName, node.doubleValue());
        } else if (node.isBoolean()) {
            return requestParameters.withRootParameter(parameterName, node.booleanValue());
        } else if (node.isTextual()) {
            return requestParameters.withRootParameter(parameterName, node.textValue());
        } else {
            return requestParameters;
        }
    }

    /*
    Extract the value from the specified json node and add it as a parameter to the map.
    This supports building an item with the required parameters (i.e. item mode).
     */
    private Map extractNode(String json,
                                            String path,
                                            Map map,
                                            String parameterName) throws Exception {
        JsonNode node = objectMapper.readTree(json).path(path);

        if (node.isIntegralNumber()) {
            map.put(parameterName, node.longValue());
        } else if (node.isFloatingPointNumber()) {
            map.put(parameterName, node.doubleValue());
        } else if (node.isBoolean()) {
            map.put(parameterName, node.booleanValue());
        } else if (node.isTextual()) {
            map.put(parameterName, node.textValue());
        }

        return map;
    }

    private String parseToString(byte[] payload) {
        return new String(payload, StandardCharsets.UTF_8);
    }

    @AutoValue.Builder
    public abstract static class Builder {
        abstract Builder setAttributeMap(Map value);
        abstract Builder setItemMode(boolean value);

        public abstract RequestParametersResponseParser build();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy