|
@@ -3,13 +3,13 @@
|
3
|
3
|
// Authors: Patrick Brosi <brosi@informatik.uni-freiburg.de>
|
4
|
4
|
|
5
|
5
|
#include <vector>
|
|
6
|
+#include "osmfixer/build.h"
|
|
7
|
+#include "osmfixer/index.h"
|
6
|
8
|
#include "osmfixer/server/StatServer.h"
|
7
|
9
|
#include "util/Misc.h"
|
8
|
10
|
#include "util/String.h"
|
9
|
11
|
#include "util/geo/Geo.h"
|
10
|
12
|
#include "util/log/Log.h"
|
11
|
|
-#include "osmfixer/index.h"
|
12
|
|
-#include "osmfixer/build.h"
|
13
|
13
|
|
14
|
14
|
using osmfixer::Params;
|
15
|
15
|
using osmfixer::StatServer;
|
|
@@ -24,10 +24,15 @@ util::http::Answer StatServer::handle(const util::http::Req& req,
|
24
|
24
|
auto cmd = parseUrl(req.url, req.payload, ¶ms);
|
25
|
25
|
|
26
|
26
|
if (cmd == "/") {
|
27
|
|
- a = util::http::Answer("200 OK", std::string(index_html, index_html + sizeof index_html / sizeof index_html[0]));
|
|
27
|
+ a = util::http::Answer(
|
|
28
|
+ "200 OK",
|
|
29
|
+ std::string(index_html,
|
|
30
|
+ index_html + sizeof index_html / sizeof index_html[0]));
|
28
|
31
|
a.params["Content-Type"] = "text/html; charset=utf-8";
|
29
|
32
|
} else if (cmd == "/build.js") {
|
30
|
|
- a = util::http::Answer("200 OK", std::string(build_js, build_js + sizeof build_js / sizeof build_js[0]));
|
|
33
|
+ a = util::http::Answer(
|
|
34
|
+ "200 OK", std::string(build_js, build_js + sizeof build_js /
|
|
35
|
+ sizeof build_js[0]));
|
31
|
36
|
a.params["Content-Type"] = "application/javascript; charset=utf-8";
|
32
|
37
|
} else if (cmd == "/map") {
|
33
|
38
|
a = handleMapReq(params);
|
|
@@ -90,43 +95,49 @@ util::http::Answer StatServer::handleHeatMapReq(const Params& pars) const {
|
90
|
95
|
json << "{\"ok\":[";
|
91
|
96
|
|
92
|
97
|
char sep = ' ';
|
93
|
|
- for (const auto& idx : _idxs) {
|
|
98
|
+ for (const auto& idxPair : _idxs) {
|
|
99
|
+ const auto& idx = idxPair.second;
|
|
100
|
+
|
94
|
101
|
auto hgOk = idx.getHeatGridOk(bbox, z);
|
95
|
102
|
for (auto cluster : hgOk) {
|
96
|
103
|
json << sep;
|
97
|
104
|
sep = ',';
|
98
|
105
|
|
99
|
|
- json << "[" << cluster.latLng.getY() << "," << cluster.latLng.getX() << ","
|
100
|
|
- << cluster.size << "]";
|
|
106
|
+ json << "[" << cluster.latLng.getY() << "," << cluster.latLng.getX()
|
|
107
|
+ << "," << cluster.size << "]";
|
101
|
108
|
}
|
102
|
109
|
}
|
103
|
110
|
|
104
|
111
|
json << "],\"sugg\":[";
|
105
|
112
|
|
106
|
113
|
sep = ' ';
|
107
|
|
- for (const auto& idx : _idxs) {
|
|
114
|
+ for (const auto& idxPair : _idxs) {
|
|
115
|
+ const auto& idx = idxPair.second;
|
|
116
|
+
|
108
|
117
|
auto hgSugg = idx.getHeatGridSugg(bbox, z);
|
109
|
118
|
|
110
|
119
|
for (auto cluster : hgSugg) {
|
111
|
120
|
json << sep;
|
112
|
121
|
sep = ',';
|
113
|
122
|
|
114
|
|
- json << "[" << cluster.latLng.getY() << "," << cluster.latLng.getX() << ","
|
115
|
|
- << cluster.size << "]";
|
|
123
|
+ json << "[" << cluster.latLng.getY() << "," << cluster.latLng.getX()
|
|
124
|
+ << "," << cluster.size << "]";
|
116
|
125
|
}
|
117
|
126
|
}
|
118
|
127
|
|
119
|
128
|
json << "],\"err\":[";
|
120
|
129
|
|
121
|
130
|
sep = ' ';
|
122
|
|
- for (const auto& idx : _idxs) {
|
|
131
|
+ for (const auto& idxPair : _idxs) {
|
|
132
|
+ const auto& idx = idxPair.second;
|
|
133
|
+
|
123
|
134
|
auto hgErr = idx.getHeatGridErr(bbox, z);
|
124
|
135
|
for (auto cluster : hgErr) {
|
125
|
136
|
json << sep;
|
126
|
137
|
sep = ',';
|
127
|
138
|
|
128
|
|
- json << "[" << cluster.latLng.getY() << "," << cluster.latLng.getX() << ","
|
129
|
|
- << cluster.size << "]";
|
|
139
|
+ json << "[" << cluster.latLng.getY() << "," << cluster.latLng.getX()
|
|
140
|
+ << "," << cluster.size << "]";
|
130
|
141
|
}
|
131
|
142
|
}
|
132
|
143
|
|
|
@@ -148,9 +159,6 @@ util::http::Answer StatServer::handleMapReq(const Params& pars) const {
|
148
|
159
|
if (pars.count("cb")) cb = pars.find("cb")->second.c_str();
|
149
|
160
|
auto box = util::split(pars.find("bbox")->second, ',');
|
150
|
161
|
|
151
|
|
- // TODO!
|
152
|
|
- size_t did = 0;
|
153
|
|
-
|
154
|
162
|
if (box.size() != 4) throw std::invalid_argument("Invalid request.");
|
155
|
163
|
|
156
|
164
|
double lat1 = atof(box[0].c_str());
|
|
@@ -169,7 +177,9 @@ util::http::Answer StatServer::handleMapReq(const Params& pars) const {
|
169
|
177
|
json << "{\"stats\":[";
|
170
|
178
|
|
171
|
179
|
char sep = ' ';
|
172
|
|
- for (const auto& idx : _idxs) {
|
|
180
|
+ for (size_t did = 0; did < _idxs.size(); did++) {
|
|
181
|
+ const auto& idx = _idxs[did].second;
|
|
182
|
+
|
173
|
183
|
auto ret = idx.getStations(bbox);
|
174
|
184
|
for (auto stat : ret) {
|
175
|
185
|
json << sep;
|
|
@@ -180,24 +190,28 @@ util::http::Answer StatServer::handleMapReq(const Params& pars) const {
|
180
|
190
|
|
181
|
191
|
json << "], \"groups\":[";
|
182
|
192
|
sep = ' ';
|
183
|
|
- for (const auto& idx : _idxs) {
|
|
193
|
+ for (size_t did = 0; did < _idxs.size(); did++) {
|
|
194
|
+ const auto& idx = _idxs[did].second;
|
|
195
|
+
|
184
|
196
|
auto gret = idx.getGroups(bbox);
|
185
|
197
|
for (auto group : gret) {
|
186
|
198
|
if (group->polyStations.size() == 1 && group->osmid < 2) continue;
|
187
|
199
|
json << sep;
|
188
|
200
|
sep = ',';
|
189
|
|
- printGroup(group, &json);
|
|
201
|
+ printGroup(group, did, &json);
|
190
|
202
|
}
|
191
|
203
|
}
|
192
|
204
|
|
193
|
205
|
json << "], \"su\":[";
|
194
|
206
|
sep = ' ';
|
195
|
|
- for (const auto& idx : _idxs) {
|
|
207
|
+ for (size_t did = 0; did < _idxs.size(); did++) {
|
|
208
|
+ const auto& idx = _idxs[did].second;
|
|
209
|
+
|
196
|
210
|
auto suggs = idx.getSuggestions(bbox);
|
197
|
211
|
for (auto sugg : suggs) {
|
198
|
212
|
json << sep;
|
199
|
213
|
sep = ',';
|
200
|
|
- printSugg(sugg, &json);
|
|
214
|
+ printSugg(sugg, did, &json);
|
201
|
215
|
}
|
202
|
216
|
}
|
203
|
217
|
|
|
@@ -212,13 +226,17 @@ util::http::Answer StatServer::handleMapReq(const Params& pars) const {
|
212
|
226
|
}
|
213
|
227
|
|
214
|
228
|
// _____________________________________________________________________________
|
215
|
|
-void StatServer::printStation(const Station* stat, size_t did, std::ostream* out) const {
|
|
229
|
+void StatServer::printStation(const Station* stat, size_t did,
|
|
230
|
+ std::ostream* out) const {
|
|
231
|
+ const auto& idx = _idxs[did].second;
|
|
232
|
+ const auto& range = _idxs[did].first;
|
|
233
|
+
|
216
|
234
|
auto latLng =
|
217
|
235
|
util::geo::webMercToLatLng<double>(stat->pos.getX(), stat->pos.getY());
|
218
|
|
- (*out) << "{\"id\":" << stat->id << ",\"lat\":" << latLng.getY()
|
219
|
|
- << ",\"lon\":" << latLng.getX();
|
|
236
|
+ (*out) << "{\"id\":" << stat->id + range.sidStart
|
|
237
|
+ << ",\"lat\":" << latLng.getY() << ",\"lon\":" << latLng.getX();
|
220
|
238
|
|
221
|
|
- if (stat->origGroup != stat->group || _idxs[did].getGroup(stat->group)->osmid == 1)
|
|
239
|
+ if (stat->origGroup != stat->group || idx.getGroup(stat->group)->osmid == 1)
|
222
|
240
|
(*out) << ",\"su\":1";
|
223
|
241
|
|
224
|
242
|
if (stat->attrErrs.size())
|
|
@@ -227,15 +245,18 @@ void StatServer::printStation(const Station* stat, size_t did, std::ostream* out
|
227
|
245
|
}
|
228
|
246
|
|
229
|
247
|
// _____________________________________________________________________________
|
230
|
|
-void StatServer::printSugg(const Suggestion* sugg, std::ostream* out) {
|
|
248
|
+void StatServer::printSugg(const Suggestion* sugg, size_t did,
|
|
249
|
+ std::ostream* out) const {
|
|
250
|
+ const auto& range = _idxs[did].first;
|
|
251
|
+
|
231
|
252
|
std::vector<util::geo::DPoint> projArrow;
|
232
|
253
|
|
233
|
254
|
for (auto p : sugg->arrow) {
|
234
|
255
|
projArrow.push_back(util::geo::webMercToLatLng<double>(p.getX(), p.getY()));
|
235
|
256
|
}
|
236
|
257
|
|
237
|
|
- (*out) << "{\"id\":" << sugg->station << ",\"type\":" << sugg->type
|
238
|
|
- << ",\"arrow\":[";
|
|
258
|
+ (*out) << "{\"id\":" << sugg->station + range.suggIdStart
|
|
259
|
+ << ",\"type\":" << sugg->type << ",\"arrow\":[";
|
239
|
260
|
char sep = ' ';
|
240
|
261
|
for (auto p : projArrow) {
|
241
|
262
|
(*out) << sep << "[" << p.getY() << "," << p.getX() << "]";
|
|
@@ -246,14 +267,17 @@ void StatServer::printSugg(const Suggestion* sugg, std::ostream* out) {
|
246
|
267
|
}
|
247
|
268
|
|
248
|
269
|
// _____________________________________________________________________________
|
249
|
|
-void StatServer::printGroup(const Group* group, std::ostream* out) {
|
|
270
|
+void StatServer::printGroup(const Group* group, size_t did,
|
|
271
|
+ std::ostream* out) const {
|
|
272
|
+ const auto& range = _idxs[did].first;
|
|
273
|
+
|
250
|
274
|
std::vector<util::geo::DPoint> projPoly;
|
251
|
275
|
|
252
|
276
|
for (auto p : group->poly.getOuter()) {
|
253
|
277
|
projPoly.push_back(util::geo::webMercToLatLng<double>(p.getX(), p.getY()));
|
254
|
278
|
}
|
255
|
279
|
|
256
|
|
- (*out) << "{\"id\":" << group->id << ",\"poly\":[";
|
|
280
|
+ (*out) << "{\"id\":" << group->id + range.gidStart << ",\"poly\":[";
|
257
|
281
|
char sep = ' ';
|
258
|
282
|
for (auto p : projPoly) {
|
259
|
283
|
(*out) << sep << "[" << p.getY() << "," << p.getX() << "]";
|
|
@@ -290,6 +314,39 @@ std::string StatServer::parseUrl(std::string u, std::string pl,
|
290
|
314
|
}
|
291
|
315
|
|
292
|
316
|
// _____________________________________________________________________________
|
|
317
|
+size_t StatServer::getDidBySid(size_t sid) const {
|
|
318
|
+ for (size_t i = 0; i < _idxs.size(); i++) {
|
|
319
|
+ auto idx = _idxs[i].second;
|
|
320
|
+ auto range = _idxs[i].first;
|
|
321
|
+ if (sid >= range.sidStart && sid <= range.sidEnd) return i;
|
|
322
|
+ }
|
|
323
|
+
|
|
324
|
+ return 0;
|
|
325
|
+}
|
|
326
|
+
|
|
327
|
+// _____________________________________________________________________________
|
|
328
|
+size_t StatServer::getDidBySuggid(size_t suggid) const {
|
|
329
|
+ for (size_t i = 0; i < _idxs.size(); i++) {
|
|
330
|
+ auto idx = _idxs[i].second;
|
|
331
|
+ auto range = _idxs[i].first;
|
|
332
|
+ if (suggid >= range.suggIdStart && suggid <= range.suggIdEnd) return i;
|
|
333
|
+ }
|
|
334
|
+
|
|
335
|
+ return 0;
|
|
336
|
+}
|
|
337
|
+
|
|
338
|
+// _____________________________________________________________________________
|
|
339
|
+size_t StatServer::getDidByGid(size_t gid) const {
|
|
340
|
+ for (size_t i = 0; i < _idxs.size(); i++) {
|
|
341
|
+ auto idx = _idxs[i].second;
|
|
342
|
+ auto range = _idxs[i].first;
|
|
343
|
+ if (gid >= range.gidStart && gid <= range.gidEnd) return i;
|
|
344
|
+ }
|
|
345
|
+
|
|
346
|
+ return 0;
|
|
347
|
+}
|
|
348
|
+
|
|
349
|
+// _____________________________________________________________________________
|
293
|
350
|
util::http::Answer StatServer::handleGroupReq(const Params& pars) const {
|
294
|
351
|
if (pars.count("id") == 0 || pars.find("id")->second.empty())
|
295
|
352
|
throw std::invalid_argument("No ID specified.");
|
|
@@ -298,10 +355,13 @@ util::http::Answer StatServer::handleGroupReq(const Params& pars) const {
|
298
|
355
|
|
299
|
356
|
size_t gid = atol(pars.find("id")->second.c_str());
|
300
|
357
|
|
301
|
|
- // TODO!
|
302
|
|
- size_t did=0;
|
|
358
|
+ size_t did = getDidByGid(gid);
|
|
359
|
+
|
|
360
|
+ const auto& range = _idxs[did].first;
|
|
361
|
+ const auto& idx = _idxs[did].second;
|
|
362
|
+ size_t idxGid = gid - range.gidStart;
|
303
|
363
|
|
304
|
|
- auto group = _idxs[did].getGroup(gid);
|
|
364
|
+ auto group = idx.getGroup(idxGid);
|
305
|
365
|
|
306
|
366
|
if (!group) return util::http::Answer("404 Not Found", "Group not found.");
|
307
|
367
|
|
|
@@ -336,10 +396,10 @@ util::http::Answer StatServer::handleGroupReq(const Params& pars) const {
|
336
|
396
|
for (const auto& sid : group->stations) {
|
337
|
397
|
json << sep;
|
338
|
398
|
sep = ',';
|
339
|
|
- const auto stat = _idxs[did].getStation(sid);
|
340
|
|
- json << "{\"id\":" << sid << ","
|
|
399
|
+ const auto stat = idx.getStation(sid);
|
|
400
|
+ json << "{\"id\":" << sid + range.sidStart << ","
|
341
|
401
|
<< "\"osmid\":" << stat->osmid << ","
|
342
|
|
- << "\"group\":" << stat->group << ","
|
|
402
|
+ << "\"group\":" << stat->group + range.gidStart << ","
|
343
|
403
|
<< "\"orig_group\":" << stat->origGroup << ","
|
344
|
404
|
<< "\"attrs\":{";
|
345
|
405
|
|
|
@@ -381,10 +441,14 @@ util::http::Answer StatServer::handleStatReq(const Params& pars) const {
|
381
|
441
|
if (pars.count("cb")) cb = pars.find("cb")->second.c_str();
|
382
|
442
|
|
383
|
443
|
size_t sid = atol(pars.find("id")->second.c_str());
|
384
|
|
- // TODO!
|
385
|
|
- size_t did=0;
|
386
|
444
|
|
387
|
|
- auto stat = _idxs[did].getStation(sid);
|
|
445
|
+ size_t did = getDidBySid(sid);
|
|
446
|
+
|
|
447
|
+ const auto& range = _idxs[did].first;
|
|
448
|
+ const auto& idx = _idxs[did].second;
|
|
449
|
+ size_t idxSid = sid - range.sidStart;
|
|
450
|
+
|
|
451
|
+ auto stat = idx.getStation(idxSid);
|
388
|
452
|
|
389
|
453
|
if (!stat) return util::http::Answer("404 Not Found", "Station not found.");
|
390
|
454
|
|
|
@@ -421,14 +485,14 @@ util::http::Answer StatServer::handleStatReq(const Params& pars) const {
|
421
|
485
|
for (const auto& err : stat->attrErrs) {
|
422
|
486
|
json << sep;
|
423
|
487
|
sep = ',';
|
424
|
|
- const auto otherStat = _idxs[did].getStation(err.otherId);
|
|
488
|
+ const auto otherStat = idx.getStation(err.otherId);
|
425
|
489
|
json << "{";
|
426
|
490
|
json << "\"attr\":[\"" << err.attr << "\",\"" << stat->attrs.at(err.attr)[0]
|
427
|
491
|
<< "\"]";
|
428
|
492
|
json << ",\"other_attr\":[\"" << err.otherAttr << "\",\""
|
429
|
493
|
<< otherStat->attrs.at(err.otherAttr)[0] << "\"]";
|
430
|
494
|
json << ",\"conf\":" << err.conf;
|
431
|
|
- json << ",\"other\":" << err.otherId;
|
|
495
|
+ json << ",\"other\":" << err.otherId + range.sidStart;
|
432
|
496
|
json << ",\"other_osmid\":" << otherStat->osmid;
|
433
|
497
|
json << "}";
|
434
|
498
|
}
|
|
@@ -440,40 +504,36 @@ util::http::Answer StatServer::handleStatReq(const Params& pars) const {
|
440
|
504
|
char seper = ' ';
|
441
|
505
|
|
442
|
506
|
// suggestions
|
443
|
|
- if (stat->group != stat->origGroup ||
|
444
|
|
- _idxs[did].getGroup(stat->group)->osmid == 1) {
|
445
|
|
- if (_idxs[did].getGroup(stat->origGroup)->osmid < 2) {
|
446
|
|
- if (_idxs[did].getGroup(stat->group)->osmid == 1) {
|
447
|
|
- json << "{\"type\": 1, \"target_gid\":" << stat->group << "}";
|
|
507
|
+ if (stat->group != stat->origGroup || idx.getGroup(stat->group)->osmid == 1) {
|
|
508
|
+ if (idx.getGroup(stat->origGroup)->osmid < 2) {
|
|
509
|
+ if (idx.getGroup(stat->group)->osmid == 1) {
|
|
510
|
+ json << "{\"type\": 1, \"target_gid\":" << stat->group + range.gidStart
|
|
511
|
+ << "}";
|
448
|
512
|
seper = ',';
|
449
|
|
- } else if (_idxs[did].getGroup(stat->group)->osmid > 1) {
|
450
|
|
- json << "{\"type\": 2, \"target_gid\":" << stat->group
|
451
|
|
- << ",\"target_osm_rel_id\":" << _idxs[did].getGroup(stat->group)->osmid
|
|
513
|
+ } else if (idx.getGroup(stat->group)->osmid > 1) {
|
|
514
|
+ json << "{\"type\": 2, \"target_gid\":" << stat->group + range.gidStart
|
|
515
|
+ << ",\"target_osm_rel_id\":" << idx.getGroup(stat->group)->osmid
|
452
|
516
|
<< "}";
|
453
|
517
|
seper = ',';
|
454
|
518
|
}
|
455
|
519
|
} else {
|
456
|
|
- if (_idxs[did].getGroup(stat->group)->osmid == 1 &&
|
457
|
|
- _idxs[did].getGroup(stat->group)->stations.size() > 1) {
|
458
|
|
- json << "{\"type\": 3,\"orig_gid\":" << stat->origGroup
|
459
|
|
- << ",\"orig_osm_rel_id\":"
|
460
|
|
- << _idxs[did].getGroup(stat->origGroup)->osmid
|
461
|
|
- << ",\"target_gid\":" << stat->group << "}";
|
|
520
|
+ if (idx.getGroup(stat->group)->osmid == 1 &&
|
|
521
|
+ idx.getGroup(stat->group)->stations.size() > 1) {
|
|
522
|
+ json << "{\"type\": 3,\"orig_gid\":" << stat->origGroup + range.gidStart
|
|
523
|
+ << ",\"orig_osm_rel_id\":" << idx.getGroup(stat->origGroup)->osmid
|
|
524
|
+ << ",\"target_gid\":" << stat->group + range.gidStart << "}";
|
462
|
525
|
seper = ',';
|
463
|
|
- } else if (_idxs[did].getGroup(stat->group)->osmid > 1) {
|
464
|
|
- json << "{\"type\": 4,\"orig_gid\":" << stat->origGroup
|
465
|
|
- << ",\"orig_osm_rel_id\":"
|
466
|
|
- << _idxs[did].getGroup(stat->origGroup)->osmid
|
467
|
|
- << ",\"target_gid\":" << stat->group
|
468
|
|
- << ",\"target_osm_rel_id\":" << _idxs[did].getGroup(stat->group)->osmid
|
469
|
|
- << "}";
|
|
526
|
+ } else if (idx.getGroup(stat->group)->osmid > 1) {
|
|
527
|
+ json << "{\"type\": 4,\"orig_gid\":" << stat->origGroup + range.gidStart
|
|
528
|
+ << ",\"orig_osm_rel_id\":" << idx.getGroup(stat->origGroup)->osmid
|
|
529
|
+ << ",\"target_gid\":" << stat->group << ",\"target_osm_rel_id\":"
|
|
530
|
+ << idx.getGroup(stat->group)->osmid + range.gidStart << "}";
|
470
|
531
|
seper = ',';
|
471
|
532
|
} else {
|
472
|
|
- json << "{\"type\": 5,\"orig_gid\":" << stat->origGroup
|
473
|
|
- << ",\"orig_osm_rel_id\":"
|
474
|
|
- << _idxs[did].getGroup(stat->origGroup)->osmid
|
475
|
|
- << ",\"target_gid\":" << stat->group
|
476
|
|
- << ",\"target_osm_rel_id\":" << _idxs[did].getGroup(stat->group)->osmid
|
|
533
|
+ json << "{\"type\": 5,\"orig_gid\":" << stat->origGroup + range.gidStart
|
|
534
|
+ << ",\"orig_osm_rel_id\":" << idx.getGroup(stat->origGroup)->osmid
|
|
535
|
+ << ",\"target_gid\":" << stat->group + range.gidStart
|
|
536
|
+ << ",\"target_osm_rel_id\":" << idx.getGroup(stat->group)->osmid
|
477
|
537
|
<< "}";
|
478
|
538
|
seper = ',';
|
479
|
539
|
}
|
|
@@ -481,7 +541,7 @@ util::http::Answer StatServer::handleStatReq(const Params& pars) const {
|
481
|
541
|
}
|
482
|
542
|
|
483
|
543
|
for (auto attrErr : stat->attrErrs) {
|
484
|
|
- const auto otherStat = _idxs[did].getStation(attrErr.otherId);
|
|
544
|
+ const auto otherStat = idx.getStation(attrErr.otherId);
|
485
|
545
|
if (otherStat->osmid == stat->osmid) {
|
486
|
546
|
// fix attributes
|
487
|
547
|
json << seper << "{\"type\": 6,\"attr\":\"" << attrErr.attr << "\""
|