2021-04-18 14:24:30 +02:00
|
|
|
import {TagsFilter} from "../Tags/TagsFilter";
|
2021-09-21 02:10:42 +02:00
|
|
|
import RelationsTracker from "./RelationsTracker";
|
2021-07-03 22:24:12 +02:00
|
|
|
import {Utils} from "../../Utils";
|
2021-08-23 15:48:42 +02:00
|
|
|
import {UIEventSource} from "../UIEventSource";
|
2021-09-29 16:55:05 +02:00
|
|
|
import {BBox} from "../BBox";
|
2022-01-14 13:58:37 +01:00
|
|
|
import * as osmtogeojson from "osmtogeojson";
|
2022-04-28 00:37:30 +02:00
|
|
|
|
2020-08-30 01:13:18 +02:00
|
|
|
/**
|
|
|
|
* Interfaces overpass to get all the latest data
|
|
|
|
*/
|
2020-06-24 00:35:19 +02:00
|
|
|
export class Overpass {
|
2021-03-25 15:19:44 +01:00
|
|
|
private _filter: TagsFilter
|
2021-09-29 16:55:05 +02:00
|
|
|
private readonly _interpreterUrl: string;
|
2021-08-23 15:48:42 +02:00
|
|
|
private readonly _timeout: UIEventSource<number>;
|
2021-03-25 15:19:44 +01:00
|
|
|
private readonly _extraScripts: string[];
|
2021-04-25 13:25:03 +02:00
|
|
|
private _includeMeta: boolean;
|
2021-09-21 02:10:42 +02:00
|
|
|
private _relationTracker: RelationsTracker;
|
2021-09-28 17:30:48 +02:00
|
|
|
|
2021-10-03 01:38:57 +02:00
|
|
|
constructor(filter: TagsFilter,
|
|
|
|
extraScripts: string[],
|
2021-09-29 16:55:05 +02:00
|
|
|
interpreterUrl: string,
|
2022-03-24 19:59:46 +01:00
|
|
|
timeout?: UIEventSource<number>,
|
|
|
|
relationTracker?: RelationsTracker,
|
2021-08-23 15:48:42 +02:00
|
|
|
includeMeta = true) {
|
2022-03-24 19:59:46 +01:00
|
|
|
this._timeout = timeout ?? new UIEventSource<number>(90);
|
2021-08-23 15:48:42 +02:00
|
|
|
this._interpreterUrl = interpreterUrl;
|
2022-03-13 01:27:19 +01:00
|
|
|
const optimized = filter.optimize()
|
|
|
|
if(optimized === true || optimized === false){
|
|
|
|
throw "Invalid filter: optimizes to true of false"
|
|
|
|
}
|
|
|
|
this._filter = optimized
|
2021-03-20 23:45:52 +01:00
|
|
|
this._extraScripts = extraScripts;
|
2021-04-25 13:25:03 +02:00
|
|
|
this._includeMeta = includeMeta;
|
2021-09-21 02:10:42 +02:00
|
|
|
this._relationTracker = relationTracker
|
2020-06-24 00:35:19 +02:00
|
|
|
}
|
|
|
|
|
2021-09-29 16:55:05 +02:00
|
|
|
public async queryGeoJson(bounds: BBox): Promise<[any, Date]> {
|
2020-07-20 20:15:21 +02:00
|
|
|
|
2021-09-29 16:55:05 +02:00
|
|
|
let query = this.buildQuery("[bbox:" + bounds.getSouth() + "," + bounds.getWest() + "," + bounds.getNorth() + "," + bounds.getEast() + "]")
|
2020-10-12 01:25:27 +02:00
|
|
|
|
2021-09-21 02:10:42 +02:00
|
|
|
const self = this;
|
2021-09-22 16:07:56 +02:00
|
|
|
const json = await Utils.downloadJson(query)
|
2021-11-07 16:34:51 +01:00
|
|
|
|
2021-09-28 17:30:48 +02:00
|
|
|
if (json.elements.length === 0 && json.remark !== undefined) {
|
|
|
|
console.warn("Timeout or other runtime error while querying overpass", json.remark);
|
|
|
|
throw `Runtime error (timeout or similar)${json.remark}`
|
|
|
|
}
|
2021-11-07 16:34:51 +01:00
|
|
|
if (json.elements.length === 0) {
|
|
|
|
console.warn("No features for", json)
|
2021-09-22 16:07:56 +02:00
|
|
|
}
|
2021-05-10 23:51:03 +02:00
|
|
|
|
2022-03-24 19:59:46 +01:00
|
|
|
self._relationTracker?.RegisterRelations(json)
|
2022-01-14 13:58:37 +01:00
|
|
|
const geojson = osmtogeojson.default(json);
|
2021-09-22 16:07:56 +02:00
|
|
|
const osmTime = new Date(json.osm3s.timestamp_osm_base);
|
|
|
|
return [geojson, osmTime];
|
2020-06-24 00:35:19 +02:00
|
|
|
}
|
2021-03-25 15:19:44 +01:00
|
|
|
|
2022-03-24 19:59:46 +01:00
|
|
|
/**
|
2022-04-26 10:31:43 +02:00
|
|
|
* Constructs the actual script
|
|
|
|
*
|
2022-04-28 00:37:30 +02:00
|
|
|
* import {Tag} from "../Tags/Tag";
|
|
|
|
*
|
2022-03-24 19:59:46 +01:00
|
|
|
* new Overpass(new Tag("key","value"), [], "").buildScript("{{bbox}}") // => `[out:json][timeout:90]{{bbox}};(nwr["key"="value"];);out body;out meta;>;out skel qt;`
|
|
|
|
*/
|
|
|
|
public buildScript(bbox: string, postCall: string = "", pretty = false): string {
|
2021-03-25 15:19:44 +01:00
|
|
|
const filters = this._filter.asOverpass()
|
|
|
|
let filter = ""
|
|
|
|
for (const filterOr of filters) {
|
2022-03-24 19:59:46 +01:00
|
|
|
if(pretty){
|
|
|
|
filter += " "
|
|
|
|
}
|
|
|
|
filter += 'nwr' + filterOr + postCall + ';'
|
|
|
|
if(pretty){
|
|
|
|
filter+="\n"
|
|
|
|
}
|
2021-03-25 15:19:44 +01:00
|
|
|
}
|
|
|
|
for (const extraScript of this._extraScripts) {
|
|
|
|
filter += '(' + extraScript + ');';
|
|
|
|
}
|
2022-03-24 19:59:46 +01:00
|
|
|
return`[out:json][timeout:${this._timeout.data}]${bbox};(${filter});out body;${this._includeMeta ? 'out meta;' : ''}>;out skel qt;`
|
|
|
|
}
|
|
|
|
|
|
|
|
public buildQuery(bbox: string): string {
|
|
|
|
const query = this.buildScript(bbox)
|
2021-09-29 16:55:05 +02:00
|
|
|
return `${this._interpreterUrl}?data=${encodeURIComponent(query)}`
|
2021-03-25 15:19:44 +01:00
|
|
|
}
|
2022-03-24 19:59:46 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Little helper method to quickly open overpass-turbo in the browser
|
|
|
|
*/
|
|
|
|
public static AsOverpassTurboLink(tags: TagsFilter){
|
|
|
|
const overpass = new Overpass(tags, [], "", undefined, undefined, false)
|
|
|
|
const script = overpass.buildScript("","({{bbox}})", true)
|
|
|
|
const url = "http://overpass-turbo.eu/?Q="
|
|
|
|
return url + encodeURIComponent(script)
|
|
|
|
}
|
2020-07-29 13:16:21 +02:00
|
|
|
}
|