forked from NKAmapper/building2osm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbuilding_merge.py
917 lines (636 loc) · 24.3 KB
/
building_merge.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
#!/usr/bin/env python3
# -*- coding: utf8
# building_merge.py
# Conflates geojson building import file with existing buildings in OSM.
# Usage: building_merge <municipality name> [max Hausdorff distance] [filename.geojson] [-debug].
# geojson file from building2osm must be present in default folder. Other filename is optional parameter.
# Creates OSM file (manual upload to OSM).
import math
import sys
import time
import json
import os.path
import urllib.request, urllib.parse
from xml.etree import ElementTree as ET
version = "0.8.0"
request_header = {"User-Agent": "building2osm/" + version}
overpass_api = "https://overpass-api.de/api/interpreter" # Overpass endpoint
import_folder = "~/Jottacloud/osm/bygninger/" # Folder containing import building files (default folder tried first)
margin_hausdorff = 10.0 # Maximum deviation between polygons (meters)
margin_tagged = 5.0 # Maximum deviation between polygons if building is tagged (meters)
margin_area = 0.4 # At least 40% equal building areas
remove_addr = True # Remove addr tags from buildings
# No warnings when replacing these building tags with each other within same category
similar_buildings = {
'residential': ["house", "detached", "semidetached_house", "terrace", "farm", "apartments", "residential", "cabin", "hut", "bungalow"],
'commercial': ["retail", "commercial", "warehouse", "industrial", "office"],
'farm': ["barn", "farm_auxiliary", "shed", "cabin"]
}
debug = False # Output extra tags for debugging/testing
# Output message to console
def message (text):
sys.stderr.write(text)
sys.stderr.flush()
# Format time
def timeformat (sec):
if sec > 3600:
return "%i:%02i:%02i hours" % (sec / 3600, (sec % 3600) / 60, sec % 60)
elif sec > 60:
return "%i:%02i minutes" % (sec / 60, sec % 60)
else:
return "%i seconds" % sec
# Format decimal number
def format_decimal(number):
if number:
number = "%.1f" % float(number)
return number.rstrip("0").rstrip(".")
else:
return ""
# Compute closest distance from point p3 to line segment [s1, s2].
# Works for short distances.
def line_distance(s1, s2, p3):
x1, y1, x2, y2, x3, y3 = map(math.radians, [s1[0], s1[1], s2[0], s2[1], p3[0], p3[1]])
# Simplified reprojection of latitude
x1 = x1 * math.cos( y1 )
x2 = x2 * math.cos( y2 )
x3 = x3 * math.cos( y3 )
A = x3 - x1
B = y3 - y1
dx = x2 - x1
dy = y2 - y1
dot = (x3 - x1)*dx + (y3 - y1)*dy
len_sq = dx*dx + dy*dy
if len_sq != 0: # in case of zero length line
param = dot / len_sq
else:
param = -1
if param < 0:
x4 = x1
y4 = y1
elif param > 1:
x4 = x2
y4 = y2
else:
x4 = x1 + param * dx
y4 = y1 + param * dy
# Also compute distance from p to segment
x = x4 - x3
y = y4 - y3
distance = 6371000 * math.sqrt( x*x + y*y ) # In meters
'''
# Project back to longitude/latitude
x4 = x4 / math.cos(y4)
lon = math.degrees(x4)
lat = math.degrees(y4)
return (lon, lat, distance)
'''
return distance
# Calculate coordinate area of polygon in square meters
# Simple conversion to planar projection, works for small areas
# < 0: Clockwise
# > 0: Counter-clockwise
# = 0: Polygon not closed
def polygon_area (polygon):
if polygon and polygon[0] == polygon[-1]:
lat_dist = math.pi * 6371009.0 / 180.0
coord = []
for node in polygon:
y = node[1] * lat_dist
x = node[0] * lat_dist * math.cos(math.radians(node[1]))
coord.append((x,y))
area = 0.0
for i in range(len(coord) - 1):
area += (coord[i+1][0] - coord[i][0]) * (coord[i+1][1] + coord[i][1]) # (x2-x1)(y2+y1)
return int(area / 2.0)
else:
return 0
# Calculate center of polygon nodes (simple average method)
# Note: If nodes are skewed to one side, the center will be skewed to the same side
def polygon_center (polygon):
if len(polygon) == 0:
return None
elif len(polygon) == 1:
return polygon[0]
length = len(polygon)
if polygon[0] == polygon[-1]:
length -= 1
x = 0
y = 0
for node in polygon[:length]:
x += node[0]
y += node[1]
x = x / length
y = y / length
return (x, y)
# Calculate centroid of polygon
# Source: https://en.wikipedia.org/wiki/Centroid#Of_a_polygon
def polygon_centroid (polygon):
if polygon[0] == polygon[-1]:
x = 0
y = 0
det = 0
for i in range(len(polygon) - 1):
d = polygon[i][0] * polygon[i+1][1] - polygon[i+1][0] * polygon[i][1]
det += d
x += (polygon[i][0] + polygon[i+1][0]) * d # (x1 + x2) (x1*y2 - x2*y1)
y += (polygon[i][1] + polygon[i+1][1]) * d # (y1 + y2) (x1*y2 - x2*y1)
x = x / (3.0 * det)
y = y / (3.0 * det)
return (x, y)
else:
return None
# Calculate new node with given distance offset in meters
# Works over short distances
def coordinate_offset (node, distance):
m = (1 / ((math.pi / 180.0) * 6378137.0)) # Degrees per meter
latitude = node[1] + (distance * m)
longitude = node[0] + (distance * m) / math.cos( math.radians(node[1]) )
return (longitude, latitude)
# Calculate Hausdorff distance, including reverse.
# Abdel Aziz Taha and Allan Hanbury: "An Efficient Algorithm for Calculating the Exact Hausdorff Distance"
# https://publik.tuwien.ac.at/files/PubDat_247739.pdf
def hausdorff_distance(p1, p2):
N1 = len(p1) - 1
N2 = len(p2) - 1
# Shuffling for small lists disabled
# random.shuffle(p1)
# random.shuffle(p2)
cmax = 0
for i in range(N1):
no_break = True
cmin = 999999.9 # Dummy
for j in range(N2):
d = line_distance(p2[j], p2[j+1], p1[i])
if d < cmax:
no_break = False
break
if d < cmin:
cmin = d
if cmin < 999999.9 and cmin > cmax and no_break:
cmax = cmin
# return cmax
for i in range(N2):
no_break = True
cmin = 999999.9 # Dummy
for j in range(N1):
d = line_distance(p1[j], p1[j+1], p2[i])
if d < cmax:
no_break = False
break
if d < cmin:
cmin = d
if cmin < 999999.9 and cmin > cmax and no_break:
cmax = cmin
return cmax
# Identify municipality name, unless more than one hit
# Returns municipality number, or input parameter if not found
def get_municipality (parameter):
if parameter.isdigit():
return parameter
else:
parameter = parameter
found_id = ""
duplicate = False
for mun_id, mun_name in iter(municipalities.items()):
if parameter.lower() == mun_name.lower():
return mun_id
elif parameter.lower() in mun_name.lower():
if found_id:
duplicate = True
else:
found_id = mun_id
if found_id and not duplicate:
return found_id
else:
return parameter
# Load dict of all municipalities
def load_municipalities():
url = "https://ws.geonorge.no/kommuneinfo/v1/fylkerkommuner?filtrer=fylkesnummer%2Cfylkesnavn%2Ckommuner.kommunenummer%2Ckommuner.kommunenavnNorsk"
file = urllib.request.urlopen(url)
data = json.load(file)
file.close()
for county in data:
for municipality in county['kommuner']:
municipalities[ municipality['kommunenummer'] ] = municipality['kommunenavnNorsk']
# Load buildings from geojson file
def load_import_buildings(filename):
global import_buildings
message ("Loading import buildings ...\n")
message ("\tFilename '%s'\n" % filename)
if not os.path.isfile(filename):
test_filename = os.path.expanduser(import_folder + filename)
if os.path.isfile(test_filename):
filename = test_filename
else:
sys.exit("\t*** File not found\n\n")
file = open(filename)
data = json.load(file)
file.close()
import_buildings = data['features']
# Add polygon center and area
for building in import_buildings:
if "building" not in building['properties']:
message("\t*** No building tag: %s\n" % (json.dumps(building, ensure_ascii=False)))
continue
if building['geometry']['type'] == "Polygon" and len(building['geometry']['coordinates']) == 1:
building['center'] = polygon_center( building['geometry']['coordinates'][0] )
building['area'] = abs(polygon_area( building['geometry']['coordinates'][0] ))
if debug:
building['properties']['AREA'] = str(building['area'])
if "STATUS" in building['properties']:
del building['properties']['STATUS']
if "DATE" in building['properties']:
del building['properties']['DATE']
# Temporary fixes
if "TYPE" in building['properties']:
if "#672 " in building['properties']['TYPE'] or "#673 " in building['properties']['TYPE']:
building['properties']['building'] = "religious"
if "#199 " in building['properties']['TYPE']:
building['properties']['building'] = "residential"
if building['properties']['building'] == "barracks":
building['properties']['building'] = "container"
if building['properties']['building'] == "hotel" and "area" in building and building['area'] < 100:
building['properties']['building'] = "cabin"
if building['properties']['building'] in ["garage", "barn"] and "area" in building and building['area'] < 15:
building['properties']['building'] = "shed"
if building['properties']['building'] == "barn" and "area" in building and building['area'] < 100:
building['properties']['building'] = "farm_auxiliary"
message ("\t%i buildings loaded\n" % len(import_buildings))
# Load existing buildings from OSM Overpass
def load_osm_buildings(municipality_id):
global osm_elements
message ("Loading existing buildings from OSM ...\n")
query = '[out:json][timeout:60];(area[ref=%s][admin_level=7][place=municipality];)->.a;(nwr["building"](area.a););(._;>;<;>;);out center meta;'\
% (municipality_id)
request = urllib.request.Request(overpass_api + "?data=" + urllib.parse.quote(query), headers=request_header)
file = urllib.request.urlopen(request)
data = json.load(file)
file.close()
osm_elements = data['elements']
# Identify members of relations, to exclude from building matching
relation_members = set()
for element in osm_elements:
if element['type'] == "relation":
for member in element['members']:
relation_members.add(member['ref']) # OSM id of element
# Create dict of nodes + list of buildings (ways tagged with building=*)
for element in osm_elements:
if element['type'] == "node":
osm_nodes[ element['id'] ] = element
element['used'] = 0
elif element['type'] == "way":
if "tags" in element and \
"building" in element['tags'] and \
"building:part" not in element['tags'] and \
len(element['nodes']) > 2 and element['nodes'][0] == element['nodes'][-1] and \
not element['id'] in relation_members:
osm_buildings.append(element)
else:
for node_ref in element['nodes']:
if node_ref in osm_nodes:
osm_nodes[ node_ref ]['used'] += 1
# Add polygon center and area
tag_count = 0
for building in osm_buildings:
if "center" in building:
building['center'] = (building['center']['lon'], building['center']['lat'])
if building['type'] == "way":
line = []
for node_ref in building['nodes']:
if node_ref in osm_nodes:
line.append((osm_nodes[ node_ref ]['lon'], osm_nodes[ node_ref ]['lat']))
# if node_ref in osm_nodes:
osm_nodes[ node_ref ]['used'] += 1
building['polygon'] = line
building['area'] = abs(polygon_area(line))
for tag in building['tags']:
if tag not in ["building", "source"] and "addr:" not in tag:
building['tagged'] = True
if "tagged" in building:
tag_count += 1
if debug:
building['tags']['AREA'] = str(building['area'])
message ("\t%i buildings loaded (%i elements)\n" % (len(osm_buildings), len(osm_elements)))
message ("\t%i buildings with tags other than building=*\n" % tag_count)
# Get top contributors
users = {}
for element in osm_elements:
if "tags" in element and "building" in element['tags']:
if element['user'] not in users:
users[ element['user'] ] = 0
users[ element['user'] ] += 1
sorted_users = sorted(users.items(), key=lambda x: x[1], reverse=True)
message ("\tTop contributors:\n")
for i, user in enumerate(sorted_users):
if user[1] > 10 and i < 10 or user[1] >= 100:
message ("\t\t%s (%i)\n" % (user[0], user[1]))
# Create new node with given tag
# Used for debugging centers
def add_node(node, tag):
global osm_id
osm_id -= 1
node_element = {
'type': 'node',
'id': osm_id,
'lat': node[1],
'lon': node[0],
'tags': tag
}
osm_elements.append(node_element)
# Create new way element for OSM
def add_way(coordinates, osm_element):
global osm_id
way_element = {
'type': 'way',
'nodes': [],
'tags': {}
}
for node in coordinates:
node_tuple = (node[0], node[1])
# Either reuse node alreadyimported
if node_tuple in import_nodes:
node_id = import_nodes[ node_tuple ]['id']
# Or create new node
else:
osm_id -= 1
node_id = osm_id
node_element = {
'type': 'node',
'id': node_id,
'lat': node[1],
'lon': node[0],
'tags': {}
}
osm_elements.append(node_element)
import_nodes[ node_tuple ] = node_element
way_element['nodes'].append(node_id)
if osm_element is None:
osm_id -= 1
way_element['id'] = osm_id
osm_elements.append(way_element)
else:
# Delete old nodes if not used anymore, and replace with new nodes
for node_ref in osm_element['nodes']:
if node_ref in osm_nodes:
osm_nodes[ node_ref ]['used'] -= 1
if osm_nodes[ node_ref ]['used'] == 0 and "tags" not in osm_nodes[ node_ref ]:
osm_nodes[ node_ref ]['action'] = "delete"
osm_element['nodes'] = way_element['nodes']
way_element = osm_element
return way_element
def add_building(building, osm_element):
global osm_id
if building['geometry']['type'] == "Point":
return
# Simple polygon
elif len(building['geometry']['coordinates']) == 1:
way_element = add_way(building['geometry']['coordinates'][0], osm_element)
if osm_element is not None and way_element['tags']['building'] != "yes" and \
way_element['tags']['building'] != building['properties']['building'] and \
not (way_element['tags']['building'] in similar_buildings['residential'] and \
building['properties']['building'] in similar_buildings['residential']) and \
not (way_element['tags']['building'] in similar_buildings['commercial'] and \
building['properties']['building'] in similar_buildings['commercial']) and \
not (way_element['tags']['building'] in similar_buildings['farm'] and \
building['properties']['building'] in similar_buildings['farm']):
way_element['tags']['OSM_BUILDING'] = way_element['tags']['building']
for tag in ["building:type", "source", "source:date"] or \
remove_addr and tag in ["addr:street", "addr:housenumber", "addr:city", "addr:country", "addr:place"]:
if tag in way_element['tags']:
del way_element['tags'][ tag ]
way_element['tags'].update(building['properties']) # If merge, update old tags
way_element['center'] = building['center']
way_element['area'] = building['area']
if osm_element is not None:
way_element['action'] = "modify"
# centroid = polygon_centroid(building['geometry']['coordinates'][0])
# add_node(centroid, {'CENTROID': "yes"})
# Multipolygon
else:
relation_element = {
'type': 'relation',
'members': [],
'tags': building['properties']
}
relation_element['tags']['type'] = "multipolygon"
role = "outer"
for patch in building['geometry']['coordinates']:
way_element = add_way(patch, None)
member = {
'type': 'way',
'ref': way_element['id'],
'role': role
}
relation_element['members'].append(member)
role = "inner" # Next patch
osm_id -= 1
relation_element['id'] = osm_id
osm_elements.append(relation_element)
# Do reverse match to verify that two buildings are each others' best match
def reverse_match(import_building):
found_building = None
best_diff = 9999 # Dummy
min_bbox = coordinate_offset(import_building['center'], - 2 * margin_hausdorff) #import_building['area'])
max_bbox = coordinate_offset(import_building['center'], + 2 * margin_hausdorff) #import_building['area'])
for osm_building in osm_buildings:
if "area" in osm_building and "ref:bygningsnr" not in osm_building['tags'] and \
min_bbox[0] < osm_building['center'][0] < max_bbox[0] and \
min_bbox[1] < osm_building['center'][1] < max_bbox[1]: # and "action" not in osm_building and \
diff_haus = hausdorff_distance(import_building['geometry']['coordinates'][0], osm_building['polygon'])
if diff_haus < best_diff:
found_building = osm_building
best_diff = diff_haus
return (found_building, best_diff)
# Merge import with OSM buildings
def merge_buildings():
global import_buildings
message ("Merging buildings ...\n")
message ("\tMaximum Hausdorff difference: %i m (%i m for tagged buildings)\n" % (margin_hausdorff, margin_tagged))
message ("\tMaximum area difference: %i %%\n" % ((1 - margin_area) * 100))
count = len(osm_buildings)
count_merge = 0
count_ref = 0
count_identical = 0
# Remove import buildings which have already been imported
message ("\tDiscover any earlier import ... ")
osm_refs = set()
for osm_element in osm_elements:
if "tags" in osm_element and "ref:bygningsnr" in osm_element['tags']:
for ref in osm_element['tags']['ref:bygningsnr'].split(";"):
osm_refs.add(ref)
count_import = len(import_buildings)
import_buildings = [building for building in import_buildings \
if "ref:bygningsnr" not in building['properties'] or building['properties']['ref:bygningsnr'] not in osm_refs]
count_existing = count_import - len(import_buildings)
message ("%i duplicate 'ref:bygningsnr' found\n" % count_existing)
# Loop osm buildings and attempt to find matching import buildings
message ("\tMatch buildings ...\n")
for osm_building in osm_buildings[:]:
count -= 1
message ("\r\t%i " % count)
found_building = None
best_diff = 9999 # Dummy
# Skip test if ref:bygningsnr exists (building has already been imported)
if "ref:bygningsnr" in osm_building['tags']:
count_ref += 1
continue
# Get bbox for limiting search below
min_bbox = coordinate_offset(osm_building['center'], - 2 * margin_hausdorff) # osm_building['area'])
max_bbox = coordinate_offset(osm_building['center'], + 2 * margin_hausdorff) # osm_building['area'])
for import_building in import_buildings:
if "area" in import_building and \
min_bbox[0] < import_building['center'][0] < max_bbox[0] and \
min_bbox[1] < import_building['center'][1] < max_bbox[1]:
# Calculate Hausdorff distance to identify building with shortest distance
diff_haus = hausdorff_distance(osm_building['polygon'], import_building['geometry']['coordinates'][0])
if diff_haus < 1.0:
count_identical += 1
if debug:
osm_building['tags']['IDENTICAL'] = " %.2f" % diff_haus
if diff_haus < best_diff:
found_building = import_building
best_diff = diff_haus
if found_building is not None:
if debug:
osm_building['tags']['HAUSDORFF'] = " %.2f" % best_diff
# Also check if Hausdorff distance is within given limit (shorter limit for tagged buildings)
if best_diff < margin_hausdorff and "tagged" not in osm_building or best_diff < margin_tagged:
# Also check if both buildings are each others best match
found_reverse, reverse_haus = reverse_match(found_building)
if found_reverse == osm_building and reverse_haus < margin_hausdorff:
# Compare building size
if margin_area < osm_building['area'] / found_building['area'] < 1.0 / margin_area:
add_building(found_building, osm_building)
import_buildings.remove(found_building)
count_merge += 1
elif debug:
osm_building['tags']['SIZE'] = "%.1f" % (osm_building['area'] / found_building['area'])
# Add remaining import buildings which were not matched
count_add = 0
for building in import_buildings:
if building['geometry']['type'] == "Polygon" and "building" in building['properties']:
add_building(building, None)
count_add += 1
message ("\r\tMerged %i buildings from OSM (%i%%)\n" % (count_merge, 100.0 * count_merge / len(osm_buildings)))
if count_ref > 0:
message ("\tSkipped %i already imported buildings in OSM (%i%%)\n" % (count_ref, 100.0 * count_ref / len(osm_buildings)))
message ("\tRemaining %i buildings from OSM not merged (%i%%)\n" % \
(len(osm_buildings) - count_merge - count_ref, 100 - 100.0 * (count_merge + count_ref) / len(osm_buildings)))
message ("\tAdded %i new buildings from import file\n" % count_add)
message ("\t%i buildings had less than 1 meter offset\n" % count_identical)
# Indent XML output
def indent_tree(elem, level=0):
i = "\n" + level*" "
if len(elem):
if not elem.text or not elem.text.strip():
elem.text = i + " "
if not elem.tail or not elem.tail.strip():
elem.tail = i
for elem in elem:
indent_tree(elem, level+1)
if not elem.tail or not elem.tail.strip():
elem.tail = i
else:
if level and (not elem.tail or not elem.tail.strip()):
elem.tail = i
# Generate one osm tag for output
def tag_property (osm_element, tag_key, tag_value):
tag_value = tag_value.strip()
if tag_value:
osm_element.append(ET.Element("tag", k=tag_key, v=tag_value))
def set_attributes (element, data):
if "user" in data:
element.set('version', str(data['version']))
element.set('user', data['user'])
element.set('uid', str(data['uid']))
element.set('timestamp', data['timestamp'])
element.set('changeset', str(data['changeset']))
element.set('visible', 'true')
if "action" in data:
element.set('action', data['action'])
else:
element.set('action', 'modify')
element.set('visible', 'true')
# Output result
def save_file(filename):
message ("Saving file ...\n")
message ("\tFilename '%s'\n" % filename)
count = 0
osm_root = ET.Element("osm", version="0.6", generator="building_merge", upload="false")
# First output all start/end nodes
for element in osm_elements:
if element['type'] == "node":
osm_node = ET.Element("node", id=str(element['id']), lat=str(element['lat']), lon=str(element['lon']))
set_attributes(osm_node, element)
osm_root.append(osm_node)
count += 1
if "tags" in element:
for key, value in iter(element['tags'].items()):
tag_property (osm_node, key, value)
elif element['type'] == "way":
osm_way = ET.Element("way", id=str(element['id']))
set_attributes(osm_way, element)
osm_root.append(osm_way)
count += 1
if "tags" in element:
for key, value in iter(element['tags'].items()):
tag_property (osm_way, key, value)
for node_ref in element['nodes']:
osm_way.append(ET.Element("nd", ref=str(node_ref)))
elif element['type'] == "relation":
osm_relation = ET.Element("relation", id=str(element['id']))
set_attributes(osm_relation, element)
osm_root.append(osm_relation)
count += 1
if "tags" in element:
for key, value in iter(element['tags'].items()):
tag_property (osm_relation, key, value)
for member in element['members']:
osm_relation.append(ET.Element("member", type=member['type'], ref=str(member['ref']), role=member['role']))
# Produce OSM/XML file
osm_tree = ET.ElementTree(osm_root)
indent_tree(osm_root)
osm_tree.write(filename, encoding="utf-8", method="xml", xml_declaration=True)
message ("\t%i elements saved\n" % count)
# Main program
if __name__ == '__main__':
start_time = time.time()
message ("\n*** building_merge %s ***\n\n" % version)
municipalities = {}
import_buildings = []
osm_buildings = []
osm_elements = []
import_nodes = {}
osm_nodes = {}
osm_id = -1000
# Parse parameters
if len(sys.argv) < 2:
message ("Please provide municipality number or name\n\n")
sys.exit()
if len(sys.argv) > 2 and sys.argv[2].isdigit():
factor = margin_tagged / margin_hausdorff
margin_hausdorff = int(sys.argv[2])
margin_tagged = margin_hausdorff * factor
if "-debug" in sys.argv:
debug = True
# Get municipality
load_municipalities()
municipality_query = sys.argv[1]
municipality_id = get_municipality(municipality_query)
if municipality_id is None or municipality_id not in municipalities:
sys.exit("Municipality '%s' not found\n" % municipality_query)
message ("Municipality: %s %s\n\n" % (municipality_id, municipalities[ municipality_id ]))
# Get filename
filename = "bygninger_%s_%s.geojson" % (municipality_id, municipalities[ municipality_id ].replace(" ", "_"))
for arg in sys.argv[2:]:
if ".geojson" in arg:
filename = arg
# Process
load_import_buildings(filename)
load_osm_buildings(municipality_id)
if len(import_buildings) > 0 and len(osm_buildings) > 0:
merge_buildings()
filename = filename.replace(".geojson", "") + "_merged.osm"
save_file(filename)
used_time = time.time() - start_time
message("Done in %s (%i buildings per second)\n\n" % (timeformat(used_time), len(osm_buildings) / used_time))