com.thinkbiganalytics.jobrepo.rest.controller.NifiFeedProcessorStatisticsRestControllerV2.java Source code

Java tutorial

Introduction

Here is the source code for com.thinkbiganalytics.jobrepo.rest.controller.NifiFeedProcessorStatisticsRestControllerV2.java

Source

package com.thinkbiganalytics.jobrepo.rest.controller;

/*-
 * #%L
 * thinkbig-job-repository-controller
 * %%
 * Copyright (C) 2017 ThinkBig Analytics
 * %%
 * 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.
 * #L%
 */

import com.thinkbiganalytics.jobrepo.security.OperationsAccessControl;
import com.thinkbiganalytics.metadata.api.MetadataAccess;
import com.thinkbiganalytics.metadata.api.jobrepo.nifi.NifiFeedProcessorErrors;
import com.thinkbiganalytics.metadata.api.jobrepo.nifi.NifiFeedProcessorStatisticsProvider;
import com.thinkbiganalytics.metadata.api.jobrepo.nifi.NifiFeedProcessorStats;
import com.thinkbiganalytics.metadata.api.jobrepo.nifi.NifiFeedStatisticsProvider;
import com.thinkbiganalytics.metadata.api.jobrepo.nifi.NifiFeedStats;
import com.thinkbiganalytics.metadata.jobrepo.nifi.provenance.NifiStatsJmsReceiver;
import com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NiFiFeedProcessorErrorsContainer;
import com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NiFiFeedProcessorStatsContainer;
import com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStatsTransform;
import com.thinkbiganalytics.rest.model.LabelValue;
import com.thinkbiganalytics.security.AccessController;

import org.joda.time.DateTime;
import org.springframework.beans.factory.annotation.Autowired;

import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

import javax.inject.Inject;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;

import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import io.swagger.annotations.ApiResponse;
import io.swagger.annotations.ApiResponses;

@Api(tags = "Operations Manager - Feeds", produces = "application/json")
@Path("/v2/provenance-stats")
public class NifiFeedProcessorStatisticsRestControllerV2 {

    /**
     * Maximum number of processor statistics to be returned to client, otherwise server would do
     * unnecessary aggregations and clients may get overloaded anyway
     **/
    private static final Integer MAX_DATA_POINTS = 6400;

    @Inject
    private MetadataAccess metadataAccess;

    @Inject
    private AccessController accessController;

    @Autowired
    private NifiFeedProcessorStatisticsProvider statsProvider;

    @Inject
    NifiFeedStatisticsProvider nifiFeedStatisticsProvider;

    @Inject
    NifiStatsJmsReceiver nifiStatsJmsReceiver;

    @GET
    @Path("/all")
    @Produces(MediaType.APPLICATION_JSON)
    @ApiOperation("Gets the provenance statistics for all feeds.")
    @ApiResponses(@ApiResponse(code = 200, message = "Returns the provenance stats.", response = com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStats.class, responseContainer = "List"))
    public Response findStats() {
        this.accessController.checkPermission(AccessController.SERVICES, OperationsAccessControl.ACCESS_OPS);
        return metadataAccess.read(() -> {
            List<? extends NifiFeedProcessorStats> list = statsProvider
                    .findWithinTimeWindow(DateTime.now().minusDays(1), DateTime.now());
            List<com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStats> model = NifiFeedProcessorStatsTransform
                    .toModel(list);
            return Response.ok(model).build();
        });
    }

    @GET
    @Path("/{feedName}/processor-duration")
    @Produces(MediaType.APPLICATION_JSON)
    @ApiOperation("Returns the list of stats for each processor within the given timeframe relative to now")
    @ApiResponses(@ApiResponse(code = 200, message = "Returns the list of stats for each processor within the given timeframe relative to now", response = com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStats.class, responseContainer = "List"))
    public Response findStats(@PathParam("feedName") String feedName, @QueryParam("from") Long fromMillis,
            @QueryParam("to") Long toMillis) {
        this.accessController.checkPermission(AccessController.SERVICES, OperationsAccessControl.ACCESS_OPS);
        final DateTime endTime = getToDateTime(toMillis);
        final DateTime startTime = getFromDateTime(fromMillis);
        return metadataAccess.read(() -> {
            NiFiFeedProcessorStatsContainer statsContainer = new NiFiFeedProcessorStatsContainer(startTime,
                    endTime);
            List<? extends NifiFeedProcessorStats> list = statsProvider.findFeedProcessorStatisticsByProcessorName(
                    feedName, statsContainer.getStartTime(), statsContainer.getEndTime());
            List<? extends NifiFeedProcessorErrors> errors = statsProvider.findFeedProcessorErrors(feedName,
                    statsContainer.getStartTime(), statsContainer.getEndTime());
            List<com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStats> model = NifiFeedProcessorStatsTransform
                    .toModel(list);
            statsContainer.setStats(model);
            return Response.ok(statsContainer).build();
        });
    }

    @GET
    @Path("/{feedName}/processor-errors")
    @Produces(MediaType.APPLICATION_JSON)
    @ApiOperation("Returns the list of stats for each processor within the given timeframe relative to now")
    @ApiResponses(@ApiResponse(code = 200, message = "Returns the list of stats for each processor within the given timeframe relative to now", response = com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStats.class, responseContainer = "List"))
    public Response findFeedProcessorErrors(@PathParam("feedName") String feedName,
            @QueryParam("from") Long fromMillis, @QueryParam("to") Long toMillis,
            @QueryParam("after") Long timestamp) {
        this.accessController.checkPermission(AccessController.SERVICES, OperationsAccessControl.ACCESS_OPS);

        final DateTime endTime = getToDateTime(toMillis);
        final DateTime startTime = getFromDateTime(fromMillis);

        NiFiFeedProcessorErrorsContainer container = new NiFiFeedProcessorErrorsContainer(startTime, endTime);
        List<? extends NifiFeedProcessorErrors> errors = null;
        if (nifiStatsJmsReceiver.isPersistErrors()) {
            errors = metadataAccess.read(() -> {
                if (timestamp != null && timestamp != 0L) {
                    return statsProvider.findFeedProcessorErrorsAfter(feedName, new DateTime(timestamp));
                } else {
                    return statsProvider.findFeedProcessorErrors(feedName, startTime, endTime);
                }
            });
        } else {
            if (timestamp != null && timestamp != 0L) {
                errors = nifiStatsJmsReceiver.getErrorsForFeed(feedName, timestamp);
            } else {
                errors = nifiStatsJmsReceiver.getErrorsForFeed(feedName, startTime.getMillis(),
                        endTime.getMillis());
            }
        }
        List<com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStatsErrors> errorsModel = NifiFeedProcessorStatsTransform
                .toErrorsModel(errors);
        container.setErrors(errorsModel);
        return Response.ok(container).build();

    }

    @GET
    @Path("/{feedName}")
    @Produces(MediaType.APPLICATION_JSON)
    @ApiOperation("Gets the statistics for the specified feed.")
    @ApiResponses(@ApiResponse(code = 200, message = "Returns the feed statistics.", response = com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStats.class, responseContainer = "List"))
    public Response findFeedStats(@PathParam("feedName") String feedName, @QueryParam("from") Long fromMillis,
            @QueryParam("to") Long toMillis) {
        this.accessController.checkPermission(AccessController.SERVICES, OperationsAccessControl.ACCESS_OPS);

        final DateTime endTime = getToDateTime(toMillis);
        final DateTime startTime = getFromDateTime(fromMillis);

        return metadataAccess.read(() -> {
            NiFiFeedProcessorStatsContainer statsContainer = new NiFiFeedProcessorStatsContainer(startTime,
                    endTime);
            NifiFeedStats feedStats = nifiFeedStatisticsProvider.findLatestStatsForFeed(feedName);

            List<? extends NifiFeedProcessorStats> list = statsProvider.findForFeedStatisticsGroupedByTime(feedName,
                    statsContainer.getStartTime(), statsContainer.getEndTime());
            List<com.thinkbiganalytics.metadata.rest.jobrepo.nifi.NifiFeedProcessorStats> model = NifiFeedProcessorStatsTransform
                    .toModel(list);

            statsContainer.setStats(model);
            if (feedStats != null) {
                statsContainer.setRunningFlows(feedStats.getRunningFeedFlows());
            } else {
                //calc diff from finished - started
                Long started = model.stream().mapToLong(s -> s.getJobsStarted()).sum();
                Long finished = model.stream().mapToLong(s -> s.getJobsFinished()).sum();
                Long running = started - finished;
                if (running < 0) {
                    running = 0L;
                }
                statsContainer.setRunningFlows(running);
            }
            return Response.ok(statsContainer).build();
        });
    }

    private DateTime getToDateTime(Long toMillis) {
        DateTime toDate;
        if (toMillis == null) {
            toDate = DateTime.now();
        } else {
            toDate = new DateTime(toMillis);
        }
        return toDate;
    }

    private DateTime getFromDateTime(Long fromMillis) {
        DateTime fromDate;
        if (fromMillis == null) {
            fromDate = DateTime.now().minusMinutes(5);
        } else {
            fromDate = new DateTime(fromMillis);
        }
        return fromDate;
    }

    @GET
    @Path("/time-frame-options")
    @Produces(MediaType.APPLICATION_JSON)
    @ApiOperation("Gets the default time frame options.")
    @ApiResponses(@ApiResponse(code = 200, message = "Returns the time frame options.", response = LabelValue.class, responseContainer = "List"))
    public Response getTimeFrameOptions() {
        List<LabelValue> vals = Arrays.stream(NifiFeedProcessorStatisticsProvider.TimeFrame.values())
                .map(timeFrame -> {
                    LabelValue label = new LabelValue(timeFrame.getDisplayName(), timeFrame.name());
                    Map<String, Object> properties = new HashMap<>(1);
                    properties.put("millis", timeFrame.getMillis());
                    label.setProperties(properties);
                    return label;
                }).collect(Collectors.toList());
        return Response.ok(vals).build();
    }
}