forked from MapComplete/MapComplete
Fix regression and add tests, add overpass link in layer documentation
This commit is contained in:
parent
f03544c468
commit
abc4a08b3a
5 changed files with 201 additions and 130 deletions
|
@ -4,6 +4,8 @@ import {Utils} from "../../Utils";
|
||||||
import {UIEventSource} from "../UIEventSource";
|
import {UIEventSource} from "../UIEventSource";
|
||||||
import {BBox} from "../BBox";
|
import {BBox} from "../BBox";
|
||||||
import * as osmtogeojson from "osmtogeojson";
|
import * as osmtogeojson from "osmtogeojson";
|
||||||
|
// @ts-ignore
|
||||||
|
import {Tag} from "../Tags/Tag"; // used in doctest
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Interfaces overpass to get all the latest data
|
* Interfaces overpass to get all the latest data
|
||||||
|
@ -16,21 +18,19 @@ export class Overpass {
|
||||||
private _includeMeta: boolean;
|
private _includeMeta: boolean;
|
||||||
private _relationTracker: RelationsTracker;
|
private _relationTracker: RelationsTracker;
|
||||||
|
|
||||||
|
|
||||||
constructor(filter: TagsFilter,
|
constructor(filter: TagsFilter,
|
||||||
extraScripts: string[],
|
extraScripts: string[],
|
||||||
interpreterUrl: string,
|
interpreterUrl: string,
|
||||||
timeout: UIEventSource<number>,
|
timeout?: UIEventSource<number>,
|
||||||
relationTracker: RelationsTracker,
|
relationTracker?: RelationsTracker,
|
||||||
includeMeta = true) {
|
includeMeta = true) {
|
||||||
this._timeout = timeout;
|
this._timeout = timeout ?? new UIEventSource<number>(90);
|
||||||
this._interpreterUrl = interpreterUrl;
|
this._interpreterUrl = interpreterUrl;
|
||||||
const optimized = filter.optimize()
|
const optimized = filter.optimize()
|
||||||
if(optimized === true || optimized === false){
|
if(optimized === true || optimized === false){
|
||||||
throw "Invalid filter: optimizes to true of false"
|
throw "Invalid filter: optimizes to true of false"
|
||||||
}
|
}
|
||||||
this._filter = optimized
|
this._filter = optimized
|
||||||
console.log("Overpass filter is",this._filter)
|
|
||||||
this._extraScripts = extraScripts;
|
this._extraScripts = extraScripts;
|
||||||
this._includeMeta = includeMeta;
|
this._includeMeta = includeMeta;
|
||||||
this._relationTracker = relationTracker
|
this._relationTracker = relationTracker
|
||||||
|
@ -51,23 +51,45 @@ export class Overpass {
|
||||||
console.warn("No features for", json)
|
console.warn("No features for", json)
|
||||||
}
|
}
|
||||||
|
|
||||||
self._relationTracker.RegisterRelations(json)
|
self._relationTracker?.RegisterRelations(json)
|
||||||
const geojson = osmtogeojson.default(json);
|
const geojson = osmtogeojson.default(json);
|
||||||
const osmTime = new Date(json.osm3s.timestamp_osm_base);
|
const osmTime = new Date(json.osm3s.timestamp_osm_base);
|
||||||
return [geojson, osmTime];
|
return [geojson, osmTime];
|
||||||
}
|
}
|
||||||
|
|
||||||
buildQuery(bbox: string): string {
|
/**
|
||||||
|
* 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 {
|
||||||
const filters = this._filter.asOverpass()
|
const filters = this._filter.asOverpass()
|
||||||
let filter = ""
|
let filter = ""
|
||||||
for (const filterOr of filters) {
|
for (const filterOr of filters) {
|
||||||
filter += 'nwr' + filterOr + ';'
|
if(pretty){
|
||||||
|
filter += " "
|
||||||
|
}
|
||||||
|
filter += 'nwr' + filterOr + postCall + ';'
|
||||||
|
if(pretty){
|
||||||
|
filter+="\n"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
for (const extraScript of this._extraScripts) {
|
for (const extraScript of this._extraScripts) {
|
||||||
filter += '(' + extraScript + ');';
|
filter += '(' + extraScript + ');';
|
||||||
}
|
}
|
||||||
const query =
|
return`[out:json][timeout:${this._timeout.data}]${bbox};(${filter});out body;${this._includeMeta ? 'out meta;' : ''}>;out skel qt;`
|
||||||
`[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)
|
||||||
return `${this._interpreterUrl}?data=${encodeURIComponent(query)}`
|
return `${this._interpreterUrl}?data=${encodeURIComponent(query)}`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,6 +47,9 @@ export class RegexTag extends TagsFilter {
|
||||||
*
|
*
|
||||||
* // A wildcard regextag should only give the key
|
* // A wildcard regextag should only give the key
|
||||||
* new RegexTag("a", /^..*$/).asOverpass() // => [ `["a"]` ]
|
* new RegexTag("a", /^..*$/).asOverpass() // => [ `["a"]` ]
|
||||||
|
*
|
||||||
|
* // A regextag with a regex key should give correct output
|
||||||
|
* new RegexTag(/a.*x/, /^..*$/).asOverpass() // => [ `[~"a.*x"~\"^..*$\"]` ]
|
||||||
*/
|
*/
|
||||||
asOverpass(): string[] {
|
asOverpass(): string[] {
|
||||||
const inv =this.invert ? "!" : ""
|
const inv =this.invert ? "!" : ""
|
||||||
|
|
|
@ -181,6 +181,7 @@ export class TagUtils {
|
||||||
* TagUtils.Tag("survey:date:={_date:now}") // => new SubstitutingTag("survey:date", "{_date:now}")
|
* TagUtils.Tag("survey:date:={_date:now}") // => new SubstitutingTag("survey:date", "{_date:now}")
|
||||||
* TagUtils.Tag("xyz!~\\[\\]") // => new RegexTag("xyz", /^\[\]$/, true)
|
* TagUtils.Tag("xyz!~\\[\\]") // => new RegexTag("xyz", /^\[\]$/, true)
|
||||||
* TagUtils.Tag("tags~(^|.*;)amenity=public_bookcase($|;.*)") // => new RegexTag("tags", /(^|.*;)amenity=public_bookcase($|;.*)/)
|
* TagUtils.Tag("tags~(^|.*;)amenity=public_bookcase($|;.*)") // => new RegexTag("tags", /(^|.*;)amenity=public_bookcase($|;.*)/)
|
||||||
|
* TagUtils.Tag("service:bicycle:.*~~*") // => new RegexTag(/^service:bicycle:.*$/, /^..*$/)
|
||||||
*/
|
*/
|
||||||
public static Tag(json: AndOrTagConfigJson | string, context: string = ""): TagsFilter {
|
public static Tag(json: AndOrTagConfigJson | string, context: string = ""): TagsFilter {
|
||||||
try {
|
try {
|
||||||
|
@ -219,9 +220,21 @@ export class TagUtils {
|
||||||
if (json === undefined) {
|
if (json === undefined) {
|
||||||
throw `Error while parsing a tag: 'json' is undefined in ${context}. Make sure all the tags are defined and at least one tag is present in a complex expression`
|
throw `Error while parsing a tag: 'json' is undefined in ${context}. Make sure all the tags are defined and at least one tag is present in a complex expression`
|
||||||
}
|
}
|
||||||
if (typeof (json) == "string") {
|
if (typeof (json) != "string") {
|
||||||
const tag = json as string;
|
if (json.and !== undefined && json.or !== undefined) {
|
||||||
|
throw `Error while parsing a TagConfig: got an object where both 'and' and 'or' are defined`
|
||||||
|
}
|
||||||
|
if (json.and !== undefined) {
|
||||||
|
return new And(json.and.map(t => TagUtils.Tag(t, context)));
|
||||||
|
}
|
||||||
|
if (json.or !== undefined) {
|
||||||
|
return new Or(json.or.map(t => TagUtils.Tag(t, context)));
|
||||||
|
}
|
||||||
|
throw "At " + context + ": unrecognized tag"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const tag = json as string;
|
||||||
for (const [operator, comparator] of TagUtils.comparators) {
|
for (const [operator, comparator] of TagUtils.comparators) {
|
||||||
if (tag.indexOf(operator) >= 0) {
|
if (tag.indexOf(operator) >= 0) {
|
||||||
const split = Utils.SplitFirst(tag, operator);
|
const split = Utils.SplitFirst(tag, operator);
|
||||||
|
@ -265,8 +278,8 @@ export class TagUtils {
|
||||||
split[1] = "..*"
|
split[1] = "..*"
|
||||||
}
|
}
|
||||||
return new RegexTag(
|
return new RegexTag(
|
||||||
split[0],
|
new RegExp("^"+split[0]+"$"),
|
||||||
split[1]
|
new RegExp("^"+ split[1]+"$")
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
if (tag.indexOf("!:=") >= 0) {
|
if (tag.indexOf("!:=") >= 0) {
|
||||||
|
@ -281,9 +294,9 @@ export class TagUtils {
|
||||||
if (tag.indexOf("!=") >= 0) {
|
if (tag.indexOf("!=") >= 0) {
|
||||||
const split = Utils.SplitFirst(tag, "!=");
|
const split = Utils.SplitFirst(tag, "!=");
|
||||||
if (split[1] === "*") {
|
if (split[1] === "*") {
|
||||||
throw "At "+context+": invalid tag "+tag+". To indicate a missing tag, use '"+split[0]+"!=' instead"
|
throw "At " + context + ": invalid tag " + tag + ". To indicate a missing tag, use '" + split[0] + "!=' instead"
|
||||||
}
|
}
|
||||||
if(split[1] === "") {
|
if (split[1] === "") {
|
||||||
split[1] = "..*"
|
split[1] = "..*"
|
||||||
}
|
}
|
||||||
return new RegexTag(
|
return new RegexTag(
|
||||||
|
@ -326,19 +339,6 @@ export class TagUtils {
|
||||||
return new Tag(split[0], split[1])
|
return new Tag(split[0], split[1])
|
||||||
}
|
}
|
||||||
throw `Error while parsing tag '${tag}' in ${context}: no key part and value part were found`
|
throw `Error while parsing tag '${tag}' in ${context}: no key part and value part were found`
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
if (json.and !== undefined && json.or !== undefined) {
|
|
||||||
throw `Error while parsing a TagConfig: got an object where both 'and' and 'or' are defined`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (json.and !== undefined) {
|
|
||||||
return new And(json.and.map(t => TagUtils.Tag(t, context)));
|
|
||||||
}
|
|
||||||
if (json.or !== undefined) {
|
|
||||||
return new Or(json.or.map(t => TagUtils.Tag(t, context)));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static GetCount(key: string, value?: string) {
|
private static GetCount(key: string, value?: string) {
|
||||||
|
|
|
@ -417,11 +417,46 @@ class PreparePersonalTheme extends DesugaringStep<LayoutConfigJson> {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class WarnForUnsubstitutedLayersInTheme extends DesugaringStep<LayoutConfigJson>{
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super("Generates a warning if a theme uses an unsubstituted layer", ["layers"],"WarnForUnsubstitutedLayersInTheme");
|
||||||
|
}
|
||||||
|
|
||||||
|
convert(json: LayoutConfigJson, context: string): { result: LayoutConfigJson; errors?: string[]; warnings?: string[]; information?: string[] } {
|
||||||
|
if(json.hideFromOverview === true){
|
||||||
|
return {result: json}
|
||||||
|
}
|
||||||
|
const warnings = []
|
||||||
|
for (const layer of json.layers) {
|
||||||
|
if(typeof layer === "string"){
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if(layer["builtin"] !== undefined){
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if(layer["source"]["geojson"] !== undefined){
|
||||||
|
// We turn a blind eye for import layers
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
const wrn = "The theme "+json.id+" has an inline layer: "+layer["id"]+". This is discouraged."
|
||||||
|
warnings.push(wrn)
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
result: json,
|
||||||
|
warnings
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
export class PrepareTheme extends Fuse<LayoutConfigJson> {
|
export class PrepareTheme extends Fuse<LayoutConfigJson> {
|
||||||
constructor(state: DesugaringContext) {
|
constructor(state: DesugaringContext) {
|
||||||
super(
|
super(
|
||||||
"Fully prepares and expands a theme",
|
"Fully prepares and expands a theme",
|
||||||
new PreparePersonalTheme(state),
|
new PreparePersonalTheme(state),
|
||||||
|
// new WarnForUnsubstitutedLayersInTheme(),
|
||||||
new OnEveryConcat("layers", new SubstituteLayer(state)),
|
new OnEveryConcat("layers", new SubstituteLayer(state)),
|
||||||
new SetDefault("socialImage", "assets/SocialImage.png", true),
|
new SetDefault("socialImage", "assets/SocialImage.png", true),
|
||||||
// We expand all tagrenderings first...
|
// We expand all tagrenderings first...
|
||||||
|
|
|
@ -24,6 +24,9 @@ import {Utils} from "../../Utils";
|
||||||
import {TagsFilter} from "../../Logic/Tags/TagsFilter";
|
import {TagsFilter} from "../../Logic/Tags/TagsFilter";
|
||||||
import Table from "../../UI/Base/Table";
|
import Table from "../../UI/Base/Table";
|
||||||
import FilterConfigJson from "./Json/FilterConfigJson";
|
import FilterConfigJson from "./Json/FilterConfigJson";
|
||||||
|
import {And} from "../../Logic/Tags/And";
|
||||||
|
import {Overpass} from "../../Logic/Osm/Overpass";
|
||||||
|
import Constants from "../Constants";
|
||||||
|
|
||||||
export default class LayerConfig extends WithContextLoader {
|
export default class LayerConfig extends WithContextLoader {
|
||||||
|
|
||||||
|
@ -236,10 +239,9 @@ export default class LayerConfig extends WithContextLoader {
|
||||||
const hasCenterRendering = this.mapRendering.some(r => r.location.has("centroid") || r.location.has("start") || r.location.has("end"))
|
const hasCenterRendering = this.mapRendering.some(r => r.location.has("centroid") || r.location.has("start") || r.location.has("end"))
|
||||||
|
|
||||||
if (this.lineRendering.length === 0 && this.mapRendering.length === 0) {
|
if (this.lineRendering.length === 0 && this.mapRendering.length === 0) {
|
||||||
console.log(json.mapRendering)
|
|
||||||
throw("The layer " + this.id + " does not have any maprenderings defined and will thus not show up on the map at all. If this is intentional, set maprenderings to 'null' instead of '[]'")
|
throw("The layer " + this.id + " does not have any maprenderings defined and will thus not show up on the map at all. If this is intentional, set maprenderings to 'null' instead of '[]'")
|
||||||
} else if (!hasCenterRendering && this.lineRendering.length === 0 && !this.source.geojsonSource?.startsWith("https://api.openstreetmap.org/api/0.6/notes.json")) {
|
} else if (!hasCenterRendering && this.lineRendering.length === 0 && !this.source.geojsonSource?.startsWith("https://api.openstreetmap.org/api/0.6/notes.json")) {
|
||||||
throw "The layer " + this.id + " might not render ways. This might result in dropped information (at "+context+")"
|
throw "The layer " + this.id + " might not render ways. This might result in dropped information (at " + context + ")"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -251,10 +253,10 @@ export default class LayerConfig extends WithContextLoader {
|
||||||
|
|
||||||
this.tagRenderings = (Utils.NoNull(json.tagRenderings) ?? []).map((tr, i) => new TagRenderingConfig(<TagRenderingConfigJson>tr, this.id + ".tagRenderings[" + i + "]"))
|
this.tagRenderings = (Utils.NoNull(json.tagRenderings) ?? []).map((tr, i) => new TagRenderingConfig(<TagRenderingConfigJson>tr, this.id + ".tagRenderings[" + i + "]"))
|
||||||
|
|
||||||
if(json.filter !== undefined && json.filter !== null && json.filter["sameAs"] !== undefined){
|
if (json.filter !== undefined && json.filter !== null && json.filter["sameAs"] !== undefined) {
|
||||||
this.filterIsSameAs = json.filter["sameAs"]
|
this.filterIsSameAs = json.filter["sameAs"]
|
||||||
this.filters = []
|
this.filters = []
|
||||||
}else{
|
} else {
|
||||||
this.filters = (<FilterConfigJson[]>json.filter ?? []).map((option, i) => {
|
this.filters = (<FilterConfigJson[]>json.filter ?? []).map((option, i) => {
|
||||||
return new FilterConfig(option, `${context}.filter-[${i}]`)
|
return new FilterConfig(option, `${context}.filter-[${i}]`)
|
||||||
});
|
});
|
||||||
|
@ -317,7 +319,7 @@ export default class LayerConfig extends WithContextLoader {
|
||||||
return mapRendering.GetBaseIcon(this.GetBaseTags())
|
return mapRendering.GetBaseIcon(this.GetBaseTags())
|
||||||
}
|
}
|
||||||
|
|
||||||
public GetBaseTags(): any{
|
public GetBaseTags(): any {
|
||||||
return TagUtils.changeAsProperties(this.source.osmTags.asChange({id: "node/-1"}))
|
return TagUtils.changeAsProperties(this.source.osmTags.asChange({id: "node/-1"}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -367,7 +369,7 @@ export default class LayerConfig extends WithContextLoader {
|
||||||
extraProps.push(new Combine(["This layer will automatically load ", new Link(dep.neededLayer, "./" + dep.neededLayer + ".md"), " into the layout as it depends on it: ", dep.reason, "(" + dep.context + ")"]))
|
extraProps.push(new Combine(["This layer will automatically load ", new Link(dep.neededLayer, "./" + dep.neededLayer + ".md"), " into the layout as it depends on it: ", dep.reason, "(" + dep.context + ")"]))
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const revDep of Utils.Dedup( layerIsNeededBy?.get(this.id) ?? [])) {
|
for (const revDep of Utils.Dedup(layerIsNeededBy?.get(this.id) ?? [])) {
|
||||||
extraProps.push(new Combine(["This layer is needed as dependency for layer", new Link(revDep, "#" + revDep)]))
|
extraProps.push(new Combine(["This layer is needed as dependency for layer", new Link(revDep, "#" + revDep)]))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -412,6 +414,15 @@ export default class LayerConfig extends WithContextLoader {
|
||||||
iconImg = `<img src='https://mapcomplete.osm.be/${icon}' height="100px"> `
|
iconImg = `<img src='https://mapcomplete.osm.be/${icon}' height="100px"> `
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let overpassLink: BaseUIElement = undefined;
|
||||||
|
if (Constants.priviliged_layers.indexOf(this.id) < 0) {
|
||||||
|
try {
|
||||||
|
overpassLink = new Link("Execute on overpass", Overpass.AsOverpassTurboLink(<TagsFilter> new And(neededTags).optimize()))
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Could not generate overpasslink for " + this.id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return new Combine([
|
return new Combine([
|
||||||
new Combine([
|
new Combine([
|
||||||
new Title(this.id, 1),
|
new Title(this.id, 1),
|
||||||
|
@ -427,7 +438,7 @@ export default class LayerConfig extends WithContextLoader {
|
||||||
new Title("Basic tags for this layer", 2),
|
new Title("Basic tags for this layer", 2),
|
||||||
"Elements must have the all of following tags to be shown on this layer:",
|
"Elements must have the all of following tags to be shown on this layer:",
|
||||||
new List(neededTags.map(t => t.asHumanString(true, false, {}))),
|
new List(neededTags.map(t => t.asHumanString(true, false, {}))),
|
||||||
|
overpassLink,
|
||||||
new Title("Supported attributes", 2),
|
new Title("Supported attributes", 2),
|
||||||
quickOverview,
|
quickOverview,
|
||||||
...this.tagRenderings.map(tr => tr.GenerateDocumentation())
|
...this.tagRenderings.map(tr => tr.GenerateDocumentation())
|
||||||
|
|
Loading…
Reference in a new issue