/** * Copyright (C) 2013 - 2015 the enviroCar community * * This file is part of the enviroCar app. * * The enviroCar app is free software: you can redistribute it and/or * modify it under the terms of the GNU General Public License as published * by the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * The enviroCar app is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General * Public License for more details. * * You should have received a copy of the GNU General Public License along * with the enviroCar app. If not, see http://www.gnu.org/licenses/. */ package org.envirocar.core.utils; import org.envirocar.core.entity.Measurement; import org.envirocar.core.entity.Track; import org.envirocar.core.exception.NoMeasurementsException; import org.envirocar.core.logging.Logger; import org.envirocar.core.util.Util; import java.util.ArrayList; import java.util.List; /** * TODO JavaDoc * * @author dewall */ public class TrackUtils { private static final Logger LOG = Logger.getLogger(TrackUtils.class); private static final double OBFUSCATION_DISTANCE_KM = 0.25; private static final int OBFUSCATION_TIME_MS = 60000; /** * resolve all not obfuscated measurements of a track. * <p> * This returns all measurements, if obfuscation is disabled. Otherwise * measurements within the first and last minute and those within the start/end * radius of 250 m are ignored (only if they are in the beginning/end of the track). * * @param track * @return */ public static Track getObfuscatedTrack(Track track) throws NoMeasurementsException { Track result = track.carbonCopy(); result.setMeasurements(getNonObfuscatedMeasurements(track)); return result; } private static List<Measurement> getNonObfuscatedMeasurements(Track track) throws NoMeasurementsException { List<Measurement> measurements = track.getMeasurements(); List<Measurement> nonPrivateMeasurements = new ArrayList<Measurement>(); try { int first = determineFirstNonObfuscatedIndex(measurements, track); int last = determineLastNonObfuscatedIndex(measurements, track); if (first == -1 || last == -1) { LOG.warn("Could not determine first/last non-obfuscated measurements."); throw new NoMeasurementsException("No obfuscated measurements available."); } for (int i = first; i <= last; i++) { nonPrivateMeasurements.add(measurements.get(i)); } return nonPrivateMeasurements; } catch (NoMeasurementsException e) { LOG.warn("Could not obfuscate track"); throw e; } } private static int determineFirstNonObfuscatedIndex(List<Measurement> measurements, Track track) throws NoMeasurementsException { for (int i = 0; i < measurements.size(); i++) { Measurement m = measurements.get(i); if (!isTemporalObfuscated(m, track) && !isSpatialObfuscated(m, track)) { return i; } } return -1; } private static int determineLastNonObfuscatedIndex(List<Measurement> measurements, Track track) throws NoMeasurementsException { for (int i = measurements.size()-1; i >= 0; i--) { Measurement m = measurements.get(i); if (!isTemporalObfuscated(m, track) && !isSpatialObfuscated(m, track)) { return i; } } return -1; } private static boolean isSpatialObfuscated(Measurement measurement, Track track) throws NoMeasurementsException { return (Util.getDistance(track.getFirstMeasurement(), measurement) <= OBFUSCATION_DISTANCE_KM) || (Util.getDistance(track.getLastMeasurement(), measurement) <= OBFUSCATION_DISTANCE_KM); } private static boolean isTemporalObfuscated(Measurement measurement, Track track) throws NoMeasurementsException { return (measurement.getTime() - track.getStartTime() <= OBFUSCATION_TIME_MS || track.getEndTime() - measurement.getTime() <= OBFUSCATION_TIME_MS); } }