-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathrun_candidates.rb
executable file
·123 lines (105 loc) · 4.02 KB
/
run_candidates.rb
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
#!/usr/bin/env ruby
# Generates the list of candidate entities in the tracker database
require 'pg'
require 'yaml'
require 'xml/libxml'
# The candidate entities are any of those touched by a decliner, plus any other
# entities touched by someone who has made a "questionable" agreement, as decided
# by the community. The full logic on which changesets can be excluded from processing etc
# are defined elsewhere - this candidate list is only an optimisation, not a decision-making
# step. Running this with "select osm_id as id from nodes;" should have no impact on results.
SCALE = 10000000
auth = YAML.load(File.open('auth.yaml'))
tracker_auth = auth['tracker']
@tracker_conn = PGconn.open( :dbname => tracker_auth['dbname'] )
dbauth = auth['database']
@dbconn = PGconn.open( :dbname => dbauth['dbname'] )
res = @tracker_conn.exec("select 1 from pg_type where typname = 'candidate_status'")
if res.num_tuples == 0
@tracker_conn.exec("CREATE TYPE candidate_status as enum ('unprocessed', 'processed', 'failed')")
end
res = @tracker_conn.exec("select 1 from pg_type where typname = 'entity_type'")
if res.num_tuples == 0
@tracker_conn.exec("CREATE TYPE entity_type as enum ('node', 'way', 'relation')")
end
@tracker_conn.exec("create table if not exists candidates (type entity_type, osm_id bigint, status candidate_status default 'unprocessed', lat real, lon real)")
@tracker_conn.exec("truncate table candidates")
parser = XML::Reader.io(File.open('additional_users.xml', "r"))
@additional_uids = []
while parser.read
next unless ["user"].include? parser.name
@additional_uids << parser['uid']
end
def lat_lon_for_entity(type, id)
lat = nil
lon = nil
case type
when 'node'
r = @dbconn.query(NODE_LOCATION_SQL % {:id => id})
if r.num_tuples > 0
lat, lon = r[0]['latitude'].to_f / SCALE, r[0]['longitude'].to_f / SCALE
end
when 'way'
r = @dbconn.query(WAY_LOCATION_SQL % {:id => id})
if r.num_tuples > 0
lat, lon = r[0]['latitude'].to_f / SCALE, r[0]['longitude'].to_f / SCALE
end
when 'relation'
r = @dbconn.query(RELATION_LOCATION_BY_NODE % {:id => id})
if r.num_tuples > 0
lat, lon = r[0]['latitude'].to_f / SCALE, r[0]['longitude'].to_f / SCALE
else
r2 = @dbconn.query(RELATION_LOCATION_BY_WAY % {:id => id})
if r2.num_tuples > 0
lat, lon = r2[0]['latitude'].to_f / SCALE, r2[0]['longitude'].to_f / SCALE
end
end
end
return lat, lon
end
EACH_SQL = \
'SELECT %{type}s.%{type}_id AS id
FROM %{type}s
INNER JOIN changesets ON %{type}s.changeset_id = changesets.id
INNER JOIN users ON changesets.user_id = users.id
WHERE users.terms_agreed is NULL
OR users.id in (%{uid_list})
GROUP BY %{type}s.%{type}_id;'
NODE_LOCATION_SQL = \
'SELECT latitude, longitude FROM current_nodes
WHERE id = %{id}'
WAY_LOCATION_SQL = \
'SELECT latitude, longitude
FROM way_nodes
JOIN nodes on way_nodes.node_id = nodes.node_id
WHERE way_id = %{id}
LIMIT 1'
RELATION_LOCATION_BY_NODE = \
"SELECT latitude, longitude
FROM relation_members
JOIN nodes on relation_members.member_id = nodes.node_id
WHERE relation_members.member_type = 'Node'
AND relation_id = %{id}
LIMIT 1"
RELATION_LOCATION_BY_WAY = \
"SELECT latitude, longitude
FROM relation_members
JOIN way_nodes on relation_members.member_id = way_nodes.way_id
JOIN nodes on way_nodes.node_id = nodes.node_id
WHERE relation_members.member_type = 'Way'
AND relation_id = %{id}
LIMIT 1"
out = @tracker_conn.transaction do |tracker_conn|
tracker_conn.exec("copy candidates FROM STDIN WITH csv")
['node', 'way', 'relation'].each do |type|
res = @dbconn.query(EACH_SQL % {:type => type, :uid_list => @additional_uids.join(",")})
puts "#{res.num_tuples} #{type}s"
res.each do |r|
id = r['id']
lat, lon = lat_lon_for_entity(type, id)
line = "#{type},#{id},unprocessed,#{lat},#{lon}\n"
tracker_conn.put_copy_data(line)
end
end
tracker_conn.put_copy_end
end