Subversion Repositories mkgmap

Rev

Rev 4491 | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
 * Copyright.
 *
 * Created by Mike Baggaley 2020
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 3 or
 * version 2 as published by the Free Software Foundation.
 *
 * This program 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.
 */


package uk.me.parabola.mkgmap.osmstyle;

import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;

import uk.me.parabola.imgfmt.app.Coord;
import uk.me.parabola.log.Logger;
import uk.me.parabola.mkgmap.general.MapPoint;
import uk.me.parabola.mkgmap.reader.osm.FakeIdGenerator;
import uk.me.parabola.mkgmap.reader.osm.GType;
import uk.me.parabola.mkgmap.reader.osm.Node;
import uk.me.parabola.util.EnhancedProperties;
import uk.me.parabola.util.KdTree;

public class NearbyPoiHandler {
        private static final Logger log = Logger.getLogger(NearbyPoiHandler.class);
        private final Map<MapPoint, Node> data = new LinkedHashMap<>();

        private enum NearbyPoiActOn {
                ACT_ON_ALL, ACT_ON_NAMED, ACT_ON_UNNAMED
        }

        private enum NearbyPoiAction {
                DELETE_POI, DELETE_NAME
        }

        private class NearbyPoiRule {
                int minPoiType = 0;
                int maxPoiType = 0;
                int maxDistance = 0;
                NearbyPoiActOn actOn = NearbyPoiActOn.ACT_ON_ALL;
                NearbyPoiAction action = NearbyPoiAction.DELETE_POI;
               
                @Override
                public String toString() {
                        StringBuilder sb = new StringBuilder();
                        sb.append(GType.formatType(minPoiType));
                        if (minPoiType != maxPoiType) {
                                sb.append('-').append(GType.formatType(maxPoiType));
                        }
                        if (actOn == NearbyPoiActOn.ACT_ON_NAMED)
                                sb.append("/named");
                        if (actOn == NearbyPoiActOn.ACT_ON_UNNAMED)
                                sb.append("/unnamed");
                        sb.append(':').append(maxDistance);
                        if (action == NearbyPoiAction.DELETE_NAME) {
                                sb.append(":delete-name");
                        }
                        return sb.toString();
                }
        }

        private final ArrayList<NearbyPoiRule> nearbyPoiRules = new ArrayList<>();
        private NearbyPoiRule defaultNamedNearbyPoiRule;
        private NearbyPoiRule defaultUnnamedNearbyPoiRule;

        public NearbyPoiHandler(EnhancedProperties props) {
                resetRules();
                String[] rules = props.getProperty("nearby-poi-rules", "").split(",");
                String rulesFileName = props.getProperty("nearby-poi-rules-config", "");
                if (!rulesFileName.isEmpty()) {
                        File file = new File(rulesFileName);
                        try {
                                List<String> fileRules = Files.readAllLines(file.toPath());
                                for (int i = 0; i < fileRules.size(); i++) {
                                        String rule = fileRules.get(i);
                                        int hashPos = rule.indexOf('#');
                                        if (hashPos >= 0)
                                                fileRules.set(i, rule.substring(0, hashPos));
                                }
                                fileRules.addAll(Arrays.asList(rules));
                                rules = fileRules.toArray(rules);
                        } catch (IOException ex) {
                                log.error("Error reading nearby POI rules file", rulesFileName);
                        }
                }
                for (int i = 0; i < rules.length; i++) {
                        String rule = rules[i].replaceAll("\\s+", "");
                        if (!rule.isEmpty()) {
                                parseRule(rule, i);
                        }
                }
                if (!checkRules()) {
                        resetRules();
                }
        }
       
        private void resetRules() {
                nearbyPoiRules.clear();
                defaultNamedNearbyPoiRule = new NearbyPoiRule();
                defaultUnnamedNearbyPoiRule = new NearbyPoiRule();
                defaultNamedNearbyPoiRule.actOn = NearbyPoiActOn.ACT_ON_NAMED;
                defaultUnnamedNearbyPoiRule.actOn = NearbyPoiActOn.ACT_ON_UNNAMED;
        }
       
        private boolean checkRules() {
                // check for conflicting overlaps in intervals
                Map<Integer,NearbyPoiRule> test = new HashMap<>();
                for (NearbyPoiRule r : nearbyPoiRules) {
                        int i = r.minPoiType;
                        while (i <= r.maxPoiType) {
                                if ((i & 0xff) > 0x1f)
                                        i = ((i >> 8) + 1) << 8;
                                NearbyPoiRule old = test.put(i, r);
                                if (old != null
                                                && (r.maxDistance != old.maxDistance || r.action != old.action || r.actOn != old.actOn)) {
                                        log.error("Different rules match for", GType.formatType(i));
                                        log.error(old);
                                        log.error(r);
                                        log.error("NearbyPoiHandler is disabled, only identical POI will be removed");
                                        return false;
                                }
                                i++;
                        }              
                       
                }
                return true;
        }
       
        private void parseRule(String rule, int i) {
                NearbyPoiRule nearbyPoiRule = new NearbyPoiRule();
                boolean valid = true;
                String[] ruleParts = rule.split(":", 4);
                String part1 = ruleParts[0];
                int slashPos = part1.indexOf('/');
                if (slashPos > 0) {
                        String part1Suffix = part1.substring(slashPos + 1);
                        part1 = part1.substring(0, slashPos);
                        switch (part1Suffix.toLowerCase()) {
                        case "named":
                                nearbyPoiRule.actOn = NearbyPoiActOn.ACT_ON_NAMED;
                                break;
       
                        case "unnamed":
                                nearbyPoiRule.actOn = NearbyPoiActOn.ACT_ON_UNNAMED;
                                break;
       
                        case "all":
                                break;
       
                        default:
                                valid = false;
                                log.error("Invalid Act On value", part1Suffix, "in nearby poi rule", i + 1, rule, "- 'all', 'named' or 'unnamed' expected.");
                                break;
                        }
                }
                if (!"*".equals(part1)) {
                        String[] poiRange = part1.split("-", 2);
                        try {
                                nearbyPoiRule.minPoiType = Integer.decode(poiRange[0]);
                                if (poiRange.length == 1)
                                        nearbyPoiRule.maxPoiType = nearbyPoiRule.minPoiType;
                                else {
                                        nearbyPoiRule.maxPoiType = Integer.decode(poiRange[1]);
                                        if (nearbyPoiRule.maxPoiType < nearbyPoiRule.minPoiType) {
                                                valid = false;
                                                log.error("Invalid POI range", part1, "in nearby poi rule", i + 1, rule);
                                        }
                                }
                        } catch (Exception ex) {
                                valid = false;
                                log.error("Invalid POI type", part1, "in nearby poi rule", i + 1, rule);
                        }
                }
                if (ruleParts.length > 1) {
                        try {
                                nearbyPoiRule.maxDistance = Integer.parseInt(ruleParts[1]);
                        } catch (Exception ex) {
                                valid = false;
                                log.error("Invalid maximum distance", ruleParts[1], "in nearby poi rule", i + 1, rule);
                        }
                        if (ruleParts.length > 2) {
                                switch (ruleParts[2].toLowerCase()) {
                                case "delete-poi":
                                        break;
       
                                case "delete-name":
                                        nearbyPoiRule.action = NearbyPoiAction.DELETE_NAME;
                                        break;
       
                                default:
                                        valid = false;
                                        log.error("Invalid Action value", ruleParts[2], "in nearby poi rule", i + 1, rule,
                                                        "- 'delete-poi', 'delete-name' or'merge-at-mid-point' expected.");
                                        break;
                                }
                        }
                        if (ruleParts.length > 3)
                                log.warn("Unexpected text", ruleParts[3], "in nearby poi rule", i + 1, rule);
                        if (valid) {
                                if ("*".equals(part1)) {
                                        if (nearbyPoiRule.actOn != NearbyPoiActOn.ACT_ON_UNNAMED)
                                                defaultNamedNearbyPoiRule = nearbyPoiRule;
                                        if (nearbyPoiRule.actOn != NearbyPoiActOn.ACT_ON_NAMED)
                                                defaultUnnamedNearbyPoiRule = nearbyPoiRule;
                                } else {
                                        int index;
                                        int count = nearbyPoiRules.size();
                                        for (index = 0; index < count; index++) {
                                                NearbyPoiRule ruleAtIndex = nearbyPoiRules.get(index);
                                                if ((nearbyPoiRule.minPoiType < ruleAtIndex.minPoiType) ||
                                                        ((nearbyPoiRule.minPoiType == ruleAtIndex.minPoiType) &&
                                                         (nearbyPoiRule.maxDistance < ruleAtIndex.maxDistance))){
                                                        nearbyPoiRules.add(index, nearbyPoiRule);
                                                        break;
                                                }
                                        }
                                        if (index == count)
                                                nearbyPoiRules.add(nearbyPoiRule);
                                }
                        }
                } else {
                        log.error("Maximum distance expected for POI type", part1, "in nearby poi rule", i + 1, rule);
                }
        }

        /**
         * Add POI and corresponding node for later processing.
         * @param mp the POI
         * @param node the node
         */

        public void add(final MapPoint mp, final Node node) {
                data.put(mp, node);
        }

        /**
         *
         * @return unmodifiable Set of all POI
         */

        public Set<MapPoint> getAllPOI() {
                return Collections.unmodifiableSet(data.keySet());
        }

        /**
         * @return Collection of deduplicated POI
         */

        public Collection<MapPoint> deDuplicate() {
                Set<MapPoint> allPOI = data.keySet();
                if (allPOI.size() < 2)
                        return allPOI; // nothing to do
                Comparator<MapPoint> typeAndName = Comparator.comparingInt(MapPoint::getType)
                                .thenComparing(mp -> mp.getName() == null ? "" : mp.getName());
                List<MapPoint> sorted = allPOI.stream().sorted(typeAndName).collect(Collectors.toList());
                Set<MapPoint> toKeep = new HashSet<>();

                int first = 0;
                int last = 0;
                int n = sorted.size();
                while (first < n) {
                        while (last < n && (0 == typeAndName.compare(sorted.get(first), sorted.get(last)))) {
                                last++;
                        }
                        toKeep.addAll(reduce(sorted.subList(first, last)));
                        first = last;
                }
                if (toKeep.size() < allPOI.size())
                        return allPOI.stream().filter(toKeep::contains).collect(Collectors.toList());
                else
                        return allPOI;
        }
       
        /**
         * @param points list of points with equal type and name
         * @return possibly reduced list
         */

        private Collection<MapPoint> reduce(List<MapPoint> points) {
                if (points.size() == 1)
                        return points;
                int type = points.get(0).getType();
                String name = points.get(0).getName();
                boolean isNamed = name != null && !name.isEmpty();
                for (NearbyPoiRule rule : nearbyPoiRules) {
                        if (rule.minPoiType > type)
                                break; // rules are in ascending order of poiType, so we can
                                                // stop trying other rules

                        if ((rule.minPoiType <= type) && (rule.maxPoiType >= type)
                                        && ((rule.actOn == NearbyPoiActOn.ACT_ON_ALL)
                                                        || ((rule.actOn == NearbyPoiActOn.ACT_ON_NAMED) && isNamed)
                                                        || ((rule.actOn == NearbyPoiActOn.ACT_ON_UNNAMED) && !isNamed))) {
                                return applyRule(points, rule);
                        }
                }

                NearbyPoiRule defaultRule = isNamed ? defaultNamedNearbyPoiRule : defaultUnnamedNearbyPoiRule;
                return applyRule(points, defaultRule);
        }

        private Collection<MapPoint> applyRule(List<MapPoint> points, NearbyPoiRule rule) {
                List<MapPoint> toKeep = new ArrayList<>();
                Map<MapPoint, Set<MapPoint>> groupsMap = buildGroups(points, rule.maxDistance, toKeep);

                while (!groupsMap.isEmpty()) {
                       
                        // find  cloud with the highest number of POI
                        Set<MapPoint> biggestCloud = groupsMap.values().stream().max(Comparator.comparingInt(Set::size)).orElse(null);
                        if (biggestCloud == null || biggestCloud.isEmpty())
                                break;
                       
                        final Coord middle = calcMiddle(biggestCloud);
                        final Set<MapPoint> done = new HashSet<>(biggestCloud);
                        removeSimpleDuplicates(biggestCloud);
                       
                        // select point that is closest to the middle
                        MapPoint bestPoint = biggestCloud.stream()
                                        .min(Comparator.comparingDouble(mp -> middle.distance(mp.getLocation())))
                                        .orElse(biggestCloud.iterator().next()); // should not happen, stream is not empty
                       
                        performAction(rule.action, bestPoint, biggestCloud, toKeep);
                       
                        // remove the processed points, they may also appear in other clouds
                        groupsMap.entrySet().forEach(e -> e.getValue().removeAll(done));
                        groupsMap.entrySet().removeIf(e -> e.getValue().isEmpty());
                }
               
                return rule.action != NearbyPoiAction.DELETE_NAME ? toKeep : points;
        }

        private void removeSimpleDuplicates(Set<MapPoint> biggestCloud) {
                Set<Coord> locations = new HashSet<>();
                Iterator<MapPoint> iter = biggestCloud.iterator();
                while (iter.hasNext()) {
                        MapPoint mp = iter.next();
                        if (!locations.add(mp.getLocation())) {
                                if (log.isInfoEnabled()) {
                                        log.info("Removed duplicate", getLogInfo(mp));
                                }
                                iter.remove();
                        }
                }
        }

        private static Map<MapPoint, Set<MapPoint>> buildGroups(List<MapPoint> points, int maxDistance, List<MapPoint> toKeep) {
                final KdTree<MapPoint> kdTree = new KdTree<>();
                points.forEach(kdTree::add);
                Map<MapPoint, Set<MapPoint>> groupsMap = new LinkedHashMap<>();
                for (MapPoint mp : points) {
                        Set<MapPoint> set = kdTree.findClosePoints(mp, maxDistance);
                        if (set.size() <= 1) {
                                toKeep.add(mp); // no other point is close
                        } else {
                                // collect cloud
                                groupsMap.put(mp, set);
                        }
                }
                return groupsMap;
        }

        private void performAction(NearbyPoiAction action, MapPoint bestPoint, Set<MapPoint> biggestCloud,
                        List<MapPoint> toKeep) {
                if (action == NearbyPoiAction.DELETE_NAME) {
                        final MapPoint midPoint = bestPoint;
                        biggestCloud.stream().filter(mp -> mp != midPoint).forEach(mp -> {
                                if (log.isInfoEnabled()) {
                                        double dist = mp.getLocation().distance(bestPoint.getLocation());
                                        log.info(String.format("Removed name from nearby(<%d m)", (long) Math.ceil(dist)), getLogInfo(mp));
                                }
                                mp.setName(null);
                        });
                        return;
                }
                // this is the point that will be displayed
                toKeep.add(bestPoint);
                if (log.isInfoEnabled()) {
                        logRemoval(biggestCloud, bestPoint);
                }
        }

        private void logRemoval(Set<MapPoint> biggestCloud, MapPoint bestPoint) {
                for (MapPoint mp : biggestCloud) {
                        if (mp != bestPoint) {
                                double dist = mp.getLocation().distance(bestPoint.getLocation());
                                log.info(String.format("Removed nearby (<%d m)", (long) Math.ceil(dist)), getLogInfo(mp));
                        }
                }
        }

        private static Coord calcMiddle(final Set<MapPoint> points) {
                // calculate centre of cloud
                double lat = 0;
                double lon = 0;
                final int n = points.size();
                for (MapPoint mp : points) {
                        Coord p = mp.getLocation();
                        lat += (double) p.getHighPrecLat() / n;
                        lon += (double) p.getHighPrecLon() / n;
                }
                return Coord.makeHighPrecCoord((int) Math.round(lat), (int) Math.round(lon));
        }

        private String getLogInfo(final MapPoint mp) {
                StringBuilder sb = new StringBuilder();
                String name = mp.getName();
                sb.append("POI with type " + GType.formatType(mp.getType()));
                sb.append(" \"");
                if (name != null && !name.isEmpty())
                        sb.append(mp.getName());
                sb.append('"');
                Node n = data.get(mp);
                if (n != null) {
                        sb.append(" for element ");
                        if (FakeIdGenerator.isFakeId(n.getId())) {
                                sb.append("generated from ").append(n.getOriginalId());
                        } else {
                                sb.append(n.toBrowseURL()).append(" at ").append(n.getLocation().toOSMURL());
                        }
                }
                return sb.toString();
        }
       
}