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

com.conveyal.gtfs.stats.PatternStats Maven / Gradle / Ivy

package com.conveyal.gtfs.stats;

import com.conveyal.gtfs.GTFSFeed;
import com.conveyal.gtfs.model.Pattern;
import com.conveyal.gtfs.model.StopTime;
import com.conveyal.gtfs.model.Trip;
import gnu.trove.list.TDoubleList;
import gnu.trove.list.TIntList;
import gnu.trove.list.array.TDoubleArrayList;
import gnu.trove.list.array.TIntArrayList;
import org.mapdb.Fun;

import java.time.LocalDate;
import java.time.LocalTime;
import java.util.Collection;
import java.util.List;
import java.util.Spliterator;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;

/**
 * Created by landon on 9/2/16.
 */
public class PatternStats {

    private GTFSFeed feed = null;
    private FeedStats stats = null;

    public PatternStats (GTFSFeed f, FeedStats fs) {
        feed = f;
        stats = fs;
    }

    /**
     * Gets the pattern speed for a given pattern for a specified date and time window.
     * @param pattern_id pattern ID
     * @param date service date
     * @param from beginning of time window
     * @param to end of time window
     * @return
     */
    public double getPatternSpeed (String pattern_id, LocalDate date, LocalTime from, LocalTime to) {

        List trips = getTripsForDate(pattern_id, date);

        return getAverageSpeedForTrips(trips, from, to);
    }

    /**
     * Get average speed for set of trips that begin within the time window in meters per second.
     * @param trips
     * @param from
     * @param to
     * @return avg. speed (meters per second)
     */
    public double getAverageSpeedForTrips (Collection trips, LocalTime from, LocalTime to) {
        TDoubleList speeds = new TDoubleArrayList();

        for (Trip trip : trips) {
            StopTime firstStopTime = feed.stop_times.ceilingEntry(Fun.t2(trip.trip_id, null)).getValue();
            LocalTime tripBeginTime = LocalTime.ofSecondOfDay(firstStopTime.departure_time % 86399); // convert 24hr+ seconds to 0 - 86399

            // skip trip if begin time is before or after specified time period
            if (tripBeginTime.isAfter(to) || tripBeginTime.isBefore(from)) {
                continue;
            }
            // TODO: swap straight lines for actual geometry?
            double speed = feed.getTripSpeed(trip.trip_id, true);

            if (!Double.isNaN(speed)) {
                speeds.add(speed);
            }
        }

        if (speeds.isEmpty()) return -1;

        return speeds.sum() / speeds.size();
    }

    /**
     * Get earliest departure time for a set of trips.
     * @param trips
     * @return earliest departure time
     */
    public LocalTime getStartTimeForTrips (Collection trips) {
        int earliestDeparture = Integer.MAX_VALUE;

        for (Trip trip : trips) {
            StopTime st = feed.getOrderedStopTimesForTrip(trip.trip_id).iterator().next();
            int dep = st.departure_time;

            // these trips begin on the next day, so we need to cast them to 0 - 86399
            if (dep > 86399) {
                dep = dep % 86399;
            }

            if (dep <= earliestDeparture) {
                earliestDeparture = dep;
            }
        }
        return LocalTime.ofSecondOfDay(earliestDeparture);
    }

    /**
     * Get last arrival time for a set of trips.
     * @param trips
     * @return last arrival time (if arrival occurs after midnight, time is expressed in terms of following day, e.g., 2:00 AM)
     */
    public LocalTime getEndTimeForTrips (Collection trips) {
        int latestArrival = Integer.MIN_VALUE;

        for (Trip trip : trips) {
            StopTime st = feed.getOrderedStopTimesForTrip(trip.trip_id).iterator().next();

            if (st.arrival_time >= latestArrival) {
                latestArrival = st.arrival_time;
            }
        }

        // return end time as 2:00 am if last arrival occurs after midnight
        return LocalTime.ofSecondOfDay(latestArrival % 86399);
    }

    /**
     * Get total revenue time (in seconds) for set of trips.
     * @param trips
     * @return total revenue time (in seconds)
     */
    public long getTotalRevenueTimeForTrips (Collection trips) {
        TIntList times = new TIntArrayList();
        for (Trip trip : trips) {
            StopTime first;
            StopTime last;
            Spliterator stopTimes = feed.getOrderedStopTimesForTrip(trip.trip_id).spliterator();;

            first = StreamSupport.stream(stopTimes, false)
                    .findFirst()
                    .orElse(null);

            last = StreamSupport.stream(stopTimes, false)
                    .reduce((a, b) -> b)
                    .orElse(null);

            if (last != null && first != null) {
                // revenue time should not include layovers at termini
                int time = last.arrival_time - first.departure_time;

                times.add(time);
            }
        }

        return times.sum();
    }

    /**
     * Get total revenue distance (in meters) for set of trips.
     * @param trips
     * @return total trip distance (in meters)
     */
    public double getTotalDistanceForTrips (Collection trips) {
        TDoubleList distances = new TDoubleArrayList();
        for (Trip trip : trips) {
            distances.add(feed.getTripDistance(trip.trip_id, false));
        }

        return distances.sum();
    }

    /**
     * Get distance for a pattern. Uses the first trip associated with the pattern.
     * @param pattern_id
     * @return distance (in meters)
     */
    public double getPatternDistance (String pattern_id) {
        Pattern pattern = feed.patterns.get(pattern_id);

        return feed.getTripDistance(pattern.associatedTrips.iterator().next(), false);
    }

    /**
     * Get average stop spacing for a pattern.
     * @param pattern_id
     * @return avg. stop spacing (in meters)
     */
    public double getAverageStopSpacing (String pattern_id) {
        Pattern pattern = feed.patterns.get(pattern_id);
        return getPatternDistance(pattern_id) / pattern.orderedStops.size();
    }

    /**
     *
     * @param pattern_id pattern ID
     * @param date service date
     * @param from beginning of time window
     * @param to end of time window
     * @return
     */
    public int getHeadwayForPattern (String pattern_id, LocalDate date, LocalTime from, LocalTime to) {

        List tripsForPattern = getTripsForDate(pattern_id, date);

        String commonStop = stats.route.getCommonStopForTrips(tripsForPattern);
        if (commonStop == null) return -1;

        return stats.stop.getStopHeadwayForTrips(commonStop, tripsForPattern, from, to);
    }

    /**
     *
     * @param pattern_id pattern ID
     * @param date service date
     * @return list of trips
     */
    public long getTripCountForDate (String pattern_id, LocalDate date) {
        return getTripsForDate(pattern_id, date).size();
    }

    /**
     *
     * @param pattern_id pattern ID
     * @param date service date
     * @return list of trips
     */
    public List getTripsForDate (String pattern_id, LocalDate date) {
        Pattern pattern = feed.patterns.get(pattern_id);
        if (pattern == null) return null;

        return stats.getTripsForDate(date).stream()
                .filter(trip -> pattern.associatedTrips.contains(trip.trip_id))
                .collect(Collectors.toList());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy