| | 1 | // License: GPL. See LICENSE file for details. |
| | 2 | package org.openstreetmap.josm.data.validation.tests; |
| | 3 | |
| | 4 | import static org.openstreetmap.josm.tools.I18n.tr; |
| | 5 | |
| | 6 | import java.util.ArrayList; |
| | 7 | import java.util.Collection; |
| | 8 | import java.util.HashSet; |
| | 9 | import java.util.LinkedList; |
| | 10 | import java.util.LinkedHashSet; |
| | 11 | import java.util.List; |
| | 12 | import java.util.Map; |
| | 13 | import java.util.Vector; |
| | 14 | |
| | 15 | import org.openstreetmap.josm.Main; |
| | 16 | import org.openstreetmap.josm.command.ChangeCommand; |
| | 17 | import org.openstreetmap.josm.command.Command; |
| | 18 | import org.openstreetmap.josm.command.DeleteCommand; |
| | 19 | import org.openstreetmap.josm.command.SequenceCommand; |
| | 20 | import org.openstreetmap.josm.data.coor.LatLon; |
| | 21 | import org.openstreetmap.josm.data.osm.Node; |
| | 22 | import org.openstreetmap.josm.data.osm.OsmPrimitive; |
| | 23 | import org.openstreetmap.josm.data.osm.OsmPrimitiveType; |
| | 24 | import org.openstreetmap.josm.data.osm.Relation; |
| | 25 | import org.openstreetmap.josm.data.osm.RelationMember; |
| | 26 | import org.openstreetmap.josm.data.osm.Way; |
| | 27 | import org.openstreetmap.josm.gui.progress.ProgressMonitor; |
| | 28 | import org.openstreetmap.josm.data.validation.Severity; |
| | 29 | import org.openstreetmap.josm.data.validation.Test; |
| | 30 | import org.openstreetmap.josm.data.validation.TestError; |
| | 31 | import org.openstreetmap.josm.tools.MultiMap; |
| | 32 | /** |
| | 33 | * Tests if there are duplicate relations |
| | 34 | */ |
| | 35 | public class DuplicateRelation extends Test |
| | 36 | { |
| | 37 | |
| | 38 | public class RelMember { |
| | 39 | private String role; |
| | 40 | private OsmPrimitiveType type; |
| | 41 | private Map<String, String> tags; |
| | 42 | private List<LatLon> coor; |
| | 43 | private long rel_id; |
| | 44 | |
| | 45 | @Override |
| | 46 | public int hashCode() { |
| | 47 | return role.hashCode()+(int)rel_id+tags.hashCode()+type.hashCode()+coor.hashCode(); |
| | 48 | } |
| | 49 | |
| | 50 | @Override |
| | 51 | public boolean equals(Object obj) { |
| | 52 | if (!(obj instanceof RelMember)) return false; |
| | 53 | RelMember rm = (RelMember) obj; |
| | 54 | return rm.role.equals(role) && rm.type.equals(type) && rm.rel_id==rel_id && rm.tags.equals(tags) && rm.coor.equals(coor); |
| | 55 | } |
| | 56 | |
| | 57 | public RelMember(RelationMember src) { |
| | 58 | role=src.getRole(); |
| | 59 | type=src.getType(); |
| | 60 | rel_id=0; |
| | 61 | coor=new ArrayList<LatLon>(); |
| | 62 | |
| | 63 | if (src.isNode()) { |
| | 64 | Node r=src.getNode(); |
| | 65 | tags=r.getKeys(); |
| | 66 | coor=new ArrayList<LatLon>(1); |
| | 67 | coor.add(r.getCoor()); |
| | 68 | } |
| | 69 | if (src.isWay()) { |
| | 70 | Way r=src.getWay(); |
| | 71 | tags=r.getKeys(); |
| | 72 | List<Node> wNodes = r.getNodes(); |
| | 73 | coor=new ArrayList<LatLon>(wNodes.size()); |
| | 74 | for (int i=0;i<wNodes.size();i++) { |
| | 75 | coor.add(wNodes.get(i).getCoor()); |
| | 76 | } |
| | 77 | } |
| | 78 | if (src.isRelation()) { |
| | 79 | Relation r=src.getRelation(); |
| | 80 | tags=r.getKeys(); |
| | 81 | rel_id=r.getId(); |
| | 82 | coor=new ArrayList<LatLon>(); |
| | 83 | } |
| | 84 | } |
| | 85 | } |
| | 86 | |
| | 87 | private class RelationMembers { |
| | 88 | public List<RelMember> members; |
| | 89 | public RelationMembers(List<RelationMember> _members) { |
| | 90 | members=new ArrayList<RelMember>(_members.size()); |
| | 91 | for (int i=0;i<_members.size();i++) { |
| | 92 | members.add(new RelMember(_members.get(i))); |
| | 93 | } |
| | 94 | } |
| | 95 | @Override |
| | 96 | public int hashCode() { |
| | 97 | return members.hashCode(); |
| | 98 | } |
| | 99 | @Override |
| | 100 | public boolean equals(Object obj) { |
| | 101 | if (!(obj instanceof RelationMembers)) return false; |
| | 102 | RelationMembers rm = (RelationMembers) obj; |
| | 103 | return rm.members.equals(members); |
| | 104 | } |
| | 105 | } |
| | 106 | |
| | 107 | private class RelationPair { |
| | 108 | public RelationMembers members; |
| | 109 | public Map<String, String> keys; |
| | 110 | public RelationPair(List<RelationMember> _members,Map<String, String> _keys) { |
| | 111 | members=new RelationMembers(_members); |
| | 112 | keys=_keys; |
| | 113 | } |
| | 114 | @Override |
| | 115 | public int hashCode() { |
| | 116 | return members.hashCode()+keys.hashCode(); |
| | 117 | } |
| | 118 | @Override |
| | 119 | public boolean equals(Object obj) { |
| | 120 | if (!(obj instanceof RelationPair)) return false; |
| | 121 | RelationPair rp = (RelationPair) obj; |
| | 122 | return rp.members.equals(members) && rp.keys.equals(keys); |
| | 123 | } |
| | 124 | } |
| | 125 | |
| | 126 | protected static int DUPLICATE_RELATION = 1901; |
| | 127 | protected static int SAME_RELATION = 1902; |
| | 128 | |
| | 129 | /** MultiMap of all relations */ |
| | 130 | MultiMap<RelationPair, OsmPrimitive> relations; |
| | 131 | |
| | 132 | /** MultiMap of all relations, regardless of keys */ |
| | 133 | MultiMap<List<RelationMember>, OsmPrimitive> relations_nokeys; |
| | 134 | |
| | 135 | /** |
| | 136 | * Constructor |
| | 137 | */ |
| | 138 | public DuplicateRelation() |
| | 139 | { |
| | 140 | super(tr("Duplicated relations")+".", |
| | 141 | tr("This test checks that there are no relations with same tags and same members with same roles.")); |
| | 142 | } |
| | 143 | |
| | 144 | |
| | 145 | @Override |
| | 146 | public void startTest(ProgressMonitor monitor) |
| | 147 | { |
| | 148 | super.startTest(monitor); |
| | 149 | relations = new MultiMap<RelationPair, OsmPrimitive>(1000); |
| | 150 | relations_nokeys = new MultiMap<List<RelationMember>, OsmPrimitive>(1000); |
| | 151 | } |
| | 152 | |
| | 153 | @Override |
| | 154 | public void endTest() |
| | 155 | { |
| | 156 | super.endTest(); |
| | 157 | for(LinkedHashSet<OsmPrimitive> duplicated : relations.values() ) |
| | 158 | { |
| | 159 | if( duplicated.size() > 1) |
| | 160 | { |
| | 161 | TestError testError = new TestError(this, Severity.ERROR, tr("Duplicated relations"), DUPLICATE_RELATION, duplicated); |
| | 162 | errors.add( testError ); |
| | 163 | } |
| | 164 | } |
| | 165 | relations = null; |
| | 166 | for(LinkedHashSet<OsmPrimitive> duplicated : relations_nokeys.values() ) |
| | 167 | { |
| | 168 | if( duplicated.size() > 1) |
| | 169 | { |
| | 170 | TestError testError = new TestError(this, Severity.WARNING, tr("Relations with same members"), SAME_RELATION, duplicated); |
| | 171 | errors.add( testError ); |
| | 172 | } |
| | 173 | } |
| | 174 | relations_nokeys = null; |
| | 175 | } |
| | 176 | |
| | 177 | @Override |
| | 178 | public void visit(Relation r) |
| | 179 | { |
| | 180 | if( !r.isUsable() ) |
| | 181 | return; |
| | 182 | List<RelationMember> rMembers=r.getMembers(); |
| | 183 | Map<String, String> rkeys=r.getKeys(); |
| | 184 | rkeys.remove("created_by"); |
| | 185 | RelationPair rKey=new RelationPair(rMembers,rkeys); |
| | 186 | relations.put(rKey, r); |
| | 187 | relations_nokeys.put(rMembers, r); |
| | 188 | } |
| | 189 | |
| | 190 | /** |
| | 191 | * Fix the error by removing all but one instance of duplicate relations |
| | 192 | */ |
| | 193 | @Override |
| | 194 | public Command fixError(TestError testError) |
| | 195 | { |
| | 196 | if (testError.getCode() == SAME_RELATION) return null; |
| | 197 | Collection<? extends OsmPrimitive> sel = testError.getPrimitives(); |
| | 198 | HashSet<Relation> rel_fix = new HashSet<Relation>(); |
| | 199 | |
| | 200 | for (OsmPrimitive osm : sel) |
| | 201 | if (osm instanceof Relation) |
| | 202 | rel_fix.add((Relation)osm); |
| | 203 | |
| | 204 | if( rel_fix.size() < 2 ) |
| | 205 | return null; |
| | 206 | |
| | 207 | long idToKeep = 0; |
| | 208 | Relation relationToKeep = rel_fix.iterator().next(); |
| | 209 | // Only one relation will be kept - the one with lowest positive ID, if such exist |
| | 210 | // or one "at random" if no such exists. Rest of the relations will be deleted |
| | 211 | for (Relation w: rel_fix) { |
| | 212 | if (!w.isNew()) { |
| | 213 | if (idToKeep == 0 || w.getId() < idToKeep) { |
| | 214 | idToKeep = w.getId(); |
| | 215 | relationToKeep = w; |
| | 216 | } |
| | 217 | } |
| | 218 | } |
| | 219 | |
| | 220 | // Find the relation that is member of one or more relations. (If any) |
| | 221 | Relation relationWithRelations = null; |
| | 222 | List<Relation> rel_ref = null; |
| | 223 | for (Relation w : rel_fix) { |
| | 224 | List<Relation> rel = OsmPrimitive.getFilteredList(w.getReferrers(), Relation.class); |
| | 225 | if (!rel.isEmpty()) { |
| | 226 | if (relationWithRelations != null) |
| | 227 | throw new AssertionError("Cannot fix duplicate relations: More than one relation is member of another relation."); |
| | 228 | relationWithRelations = w; |
| | 229 | rel_ref = rel; |
| | 230 | } |
| | 231 | } |
| | 232 | |
| | 233 | Collection<Command> commands = new LinkedList<Command>(); |
| | 234 | |
| | 235 | // Fix relations. |
| | 236 | if (relationWithRelations != null && relationToKeep != relationWithRelations) { |
| | 237 | for (Relation rel : rel_ref) { |
| | 238 | Relation newRel = new Relation(rel); |
| | 239 | for (int i = 0; i < newRel.getMembers().size(); ++i) { |
| | 240 | RelationMember m = newRel.getMember(i); |
| | 241 | if (relationWithRelations.equals(m.getMember())) { |
| | 242 | newRel.setMember(i, new RelationMember(m.getRole(), relationToKeep)); |
| | 243 | } |
| | 244 | } |
| | 245 | commands.add(new ChangeCommand(rel, newRel)); |
| | 246 | } |
| | 247 | } |
| | 248 | |
| | 249 | //Delete all relations in the list |
| | 250 | rel_fix.remove(relationToKeep); |
| | 251 | commands.add(new DeleteCommand(rel_fix)); |
| | 252 | return new SequenceCommand(tr("Delete duplicate relations"), commands); |
| | 253 | } |
| | 254 | |
| | 255 | @Override |
| | 256 | public boolean isFixable(TestError testError) |
| | 257 | { |
| | 258 | if (!(testError.getTester() instanceof DuplicateRelation)) |
| | 259 | return false; |
| | 260 | |
| | 261 | if (testError.getCode() == SAME_RELATION) return false; |
| | 262 | |
| | 263 | // We fix it only if there is no more than one relation that is relation member. |
| | 264 | Collection<? extends OsmPrimitive> sel = testError.getPrimitives(); |
| | 265 | HashSet<Relation> relations = new HashSet<Relation>(); |
| | 266 | |
| | 267 | for (OsmPrimitive osm : sel) |
| | 268 | if (osm instanceof Relation) |
| | 269 | relations.add((Relation)osm); |
| | 270 | |
| | 271 | if (relations.size() < 2) |
| | 272 | return false; |
| | 273 | |
| | 274 | int relationsWithRelations = 0; |
| | 275 | for (Relation w : relations) { |
| | 276 | List<Relation> rel = OsmPrimitive.getFilteredList(w.getReferrers(), Relation.class); |
| | 277 | if (!rel.isEmpty()) { |
| | 278 | ++relationsWithRelations; |
| | 279 | } |
| | 280 | } |
| | 281 | return (relationsWithRelations <= 1); |
| | 282 | } |
| | 283 | } |