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

com.cognite.client.servicesV1.response.TSPointsResponseParser 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.cognite.client.servicesV1.util.DurationParser;
import com.fasterxml.jackson.databind.JsonNode;
import com.google.auto.value.AutoValue;
import com.google.common.base.Preconditions;

import java.io.IOException;
import java.time.Instant;
import java.util.Optional;

@AutoValue
public abstract class TSPointsResponseParser extends DefaultResponseParser {
    private static final int DEFAULT_PARAMETER_LIMIT = 100;

    private static final String END_KEY = "end";
    private static final String GRANULARITY_KEY = "granularity";

    private final DurationParser durationParser = DurationParser.builder().build();

    public static TSPointsResponseParser.Builder builder() {
        return new com.cognite.client.servicesV1.response.AutoValue_TSPointsResponseParser.Builder()
                .setRequest(Request.create());
    }

    public abstract TSPointsResponseParser.Builder toBuilder();

    public abstract Request getRequest();

    public TSPointsResponseParser withRequest(Request parameters) {
        Preconditions.checkNotNull(parameters, "Request parameters cannot be null");

        return toBuilder().setRequest(parameters).build();
    }

    /**
     * Extract the next cursor from a results json payload.
     *
     * @param json The results json payload
     * @return
     * @throws IOException
     */
    @Override
    public Optional extractNextCursor(String json) throws Exception {
        LOG.info("Extracting next cursor from TS datapoints payload.");

        JsonNode dataitems = objectMapper.readTree(json).path("items");
        if (dataitems.isArray()) {
            LOG.debug("Extracting next cursor. Found items in Json array.");
            JsonNode datapoints = dataitems.get(0).path("datapoints");
            if (datapoints.isArray()) {
                LOG.debug("Extracting next cursor. Found datapoints in items Json array.");
                // first check the length of the json response to see if we received *limit* number of points
                final int limit = (Integer) getRequest().getRequestParameters()
                                .getOrDefault("limit", DEFAULT_PARAMETER_LIMIT);
                LOG.debug("Extracting next cursor. Limit parameter in request: {}.", limit);
                if (datapoints.size() >= limit) {
                    LOG.debug("Extracting next cursor. Limit parameter and datapoints size are equal.");
                    // get the time of the last datapoint
                    JsonNode lastPoint = datapoints.get(datapoints.size() - 1);
                    long lastTimestamp = lastPoint.get("timestamp").longValue();
                    LOG.debug("Extracting next cursor. Last datapoint timestamp: {}.", lastTimestamp);
                    long endTimestamp = Instant.now().toEpochMilli();
                    // check if we have an end time (otherwise it was now)
                    if (getRequest().getRequestParameters().containsKey(END_KEY)) {
                        LOG.debug("Extracting next cursor. Request contains end key: {}",
                                getRequest().getRequestParameters().containsKey(END_KEY));

                        // if end is String, we need to parse it
                        if (getRequest().getRequestParameters().get(END_KEY) instanceof String) {
                            LOG.debug("Extracting next cursor. Trying to parse end key");
                            endTimestamp = System.currentTimeMillis() - durationParser
                                    .parseDuration((String) getRequest().getRequestParameters().get(END_KEY))
                                    .toMillis();

                        } else if (getRequest().getRequestParameters().get(END_KEY) instanceof Number) {
                            LOG.debug("Extracting next cursor. End key matched epoch timestamp");
                            endTimestamp = (Long) getRequest().getRequestParameters().get(END_KEY);
                        } else {
                            // no compatible type.
                            LOG.error("Parameter end is not a compatible type: "
                                    + getRequest().getRequestParameters().get(END_KEY).getClass().getCanonicalName());
                            throw new Exception("Parameter end is not a compatible type: "
                                    + getRequest().getRequestParameters().get(END_KEY).getClass().getCanonicalName());
                        }
                    }
                    // if we do have an end time, check that the latest point is not past it
                    if (lastTimestamp + 1 >= endTimestamp) {
                        LOG.debug("Extracting next cursor. Last timestamp > end key. No next cursor.");
                        // we have gotten all points, return that there is no nextCursor
                        return Optional.empty();
                    }

                    // we are missing datapoints, return the next expected Timestamp
                    LOG.debug("Extracting next cursor. Need to fetch more datapoints. Building next cursor.");
                    long nextDelta = 1;

                    // Check if this is an aggregation
                    if (getRequest().getRequestParameters().containsKey(GRANULARITY_KEY)) {
                        LOG.debug("Extracting next cursor. Request is an aggregation request: {}",
                                getRequest().getRequestParameters().get(GRANULARITY_KEY));
                        // Parse the granularity specification
                        if (getRequest().getRequestParameters().get(GRANULARITY_KEY) instanceof String) {
                            LOG.debug("Extracting next cursor. Trying to parse the granularity key.");
                            String granularityString = (String) getRequest().getRequestParameters().get(GRANULARITY_KEY);
                            nextDelta = durationParser.parseDuration(granularityString).toMillis();

                        } else {
                            // no compatible type.
                            LOG.error("Parameter " + GRANULARITY_KEY + " is not a compatible type: "
                                    + getRequest().getRequestParameters().get(GRANULARITY_KEY)
                                            .getClass().getCanonicalName());
                            throw new Exception("Parameter " + GRANULARITY_KEY + " is not a compatible type: "
                                    + getRequest().getRequestParameters().get(GRANULARITY_KEY)
                                    .getClass().getCanonicalName());
                        }
                    }

                    return Optional.of(Long.toString(lastTimestamp + nextDelta));
                }
            }
        }
        // we have gotten all points, return that there is no nextCursor
        LOG.info("Next cursor not found in Json payload: \r\n" + json
                .substring(0, Math.min(MAX_LENGTH_JSON_LOG, json.length())));
        return Optional.empty();
    }

    @AutoValue.Builder
    public abstract static class Builder {
        public abstract TSPointsResponseParser.Builder setRequest(Request value);

        public abstract TSPointsResponseParser build();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy