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

com.cognite.client.servicesV1.parser.RawParser 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.parser;

import com.cognite.client.dto.RawRow;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.protobuf.Int64Value;
import com.google.protobuf.Struct;
import com.google.protobuf.Value;
import com.google.protobuf.util.JsonFormat;

import java.util.HashMap;
import java.util.Map;

import static com.cognite.client.servicesV1.ConnectorConstants.MAX_LOG_ELEMENT_LENGTH;

/**
 * This class contains a set of methods to help parsing file objects between Cognite api representations
 * (json and proto) and typed objects.
 */
public class RawParser {
    static final String logPrefix = "RawParser - ";
    static final ObjectMapper objectMapper = new ObjectMapper();

    /**
     * Parses an raw row json string to {@link RawRow} proto object.
     *
     * @return
     * @throws Exception
     */
    public static RawRow parseRawRow(String dbName, String dbTable, String rowJson) throws Exception {
        Preconditions.checkNotNull(dbName, "dbName cannot be null");
        Preconditions.checkNotNull(dbTable, "dbTable cannot be null");
        Preconditions.checkNotNull(rowJson, "rowJson cannot be null");
        String logItemExerpt = rowJson.substring(0, Math.min(rowJson.length() - 1, MAX_LOG_ELEMENT_LENGTH));

        JsonNode root = objectMapper.readTree(rowJson);
        RawRow.Builder rowBuilder = RawRow.newBuilder()
                .setDbName(dbName)
                .setTableName(dbTable);

        // A raw row must contain key, last updated time and columns.
        if (root.path("key").isTextual()) {
            rowBuilder.setKey(root.get("key").textValue());
        } else {
            throw new Exception(logPrefix
                    + "Unable to parse attribute: key. Item exerpt: "
                    + logItemExerpt);
        }
        if (root.path("lastUpdatedTime").isIntegralNumber()) {
            rowBuilder.setLastUpdatedTime(root.get("lastUpdatedTime").longValue());
        } else {
            throw new Exception(logPrefix
                    + "Unable to parse attribute: lastUpdatedTime. Item exerpt: "
                    + logItemExerpt);
        }
        if (root.path("columns").isObject()) {
            Struct.Builder structBuilder = Struct.newBuilder();
            JsonFormat.parser().merge(objectMapper.writeValueAsString(root.get("columns")), structBuilder);
            rowBuilder.setColumns(structBuilder.build());
        } else {
            throw new Exception(logPrefix
                    + "Unable to parse attribute: columns. Item exerpt: "
                    + logItemExerpt);
        }

       return rowBuilder.build();
    }

    /**
     * Builds a request insert item object from RawRow.
     *
     * An insert item object creates a new raw row data object in the Cognite system.
     *
     * @param element
     * @return
     */
    public static Map toRequestInsertItem(RawRow element) {
        ImmutableMap.Builder mapBuilder = ImmutableMap.builder();
        mapBuilder.put("key", element.getKey());
        if (element.hasColumns()) {
            mapBuilder.put("columns", RawParser.parseStructToMap(element.getColumns()));
        }
        return mapBuilder.build();
    }

    /**
     * Parses the main proto Struct (Json object equivalent) into a POJO representation.
     */
    private static Map parseStructToMap(Struct input) {
        Map map = new HashMap<>();
        for (Map.Entry entry : input.getFieldsMap().entrySet()) {
            map.put(entry.getKey(), RawParser.parseValueToObject(entry.getValue()));
        }
        return map;
    }

    /**
     * Parses the proto Value to POJO. This is needed to translate the proto Json representation into the Java object
     * representation used by the request parameters object.
     * @param input
     * @return
     */
    private static Object parseValueToObject(Value input) {
        if (input.getKindCase() == Value.KindCase.NULL_VALUE) {
            return null;
        }
        if (input.getKindCase() == Value.KindCase.BOOL_VALUE) {
            return Boolean.valueOf(input.getBoolValue());
        }
        if (input.getKindCase() == Value.KindCase.STRING_VALUE) {
            return String.valueOf(input.getStringValue());
        }
        if (input.getKindCase() == Value.KindCase.NUMBER_VALUE) {
            return Double.valueOf(input.getNumberValue());
        }
        if (input.getKindCase() == Value.KindCase.LIST_VALUE) {
            ImmutableList.Builder listBuilder = ImmutableList.builder();
            for (Value listElement : input.getListValue().getValuesList()) {
                if (listElement.getKindCase() != Value.KindCase.NULL_VALUE) {
                    // Nulls are skipped.
                    listBuilder.add(RawParser.parseValueToObject(listElement));
                }
            }
            return listBuilder.build();
        }
        if (input.getKindCase() == Value.KindCase.STRUCT_VALUE) {
            return RawParser.parseStructToMap(input.getStructValue());
        }
        // If no compatible type is found (should not happen!) then just return null
        return null;
    }
}