Add detection for negative tagging in layers, fix issue with 'key!=' being interpreted as 'key!=*', add tests

This commit is contained in:
pietervdvn 2022-03-31 02:44:23 +02:00
parent 360f83f04f
commit b5e289764e
10 changed files with 49 additions and 3 deletions

View file

@ -238,4 +238,8 @@ export class And extends TagsFilter {
return new And(newAnds) return new And(newAnds)
} }
isNegative(): boolean {
return !this.and.some(t => !t.isNegative());
}
} }

View file

@ -59,4 +59,8 @@ export default class ComparingTag implements TagsFilter {
optimize(): TagsFilter | boolean { optimize(): TagsFilter | boolean {
return this; return this;
} }
isNegative(): boolean {
return true;
}
} }

View file

@ -167,6 +167,10 @@ export class Or extends TagsFilter {
return new Or(newOrs) return new Or(newOrs)
} }
isNegative(): boolean {
return this.or.some(t => t.isNegative());
}
} }

View file

@ -197,4 +197,8 @@ export class RegexTag extends TagsFilter {
optimize(): TagsFilter | boolean { optimize(): TagsFilter | boolean {
return this; return this;
} }
isNegative(): boolean {
return this.invert;
}
} }

View file

@ -89,4 +89,8 @@ export default class SubstitutingTag implements TagsFilter {
optimize(): TagsFilter | boolean { optimize(): TagsFilter | boolean {
return this; return this;
} }
isNegative(): boolean {
return false;
}
} }

View file

@ -120,4 +120,8 @@ export class Tag extends TagsFilter {
optimize(): TagsFilter | boolean { optimize(): TagsFilter | boolean {
return this; return this;
} }
isNegative(): boolean {
return false;
}
} }

View file

@ -172,7 +172,7 @@ export class TagUtils {
* *
* TagUtils.Tag("key=value") // => new Tag("key", "value") * TagUtils.Tag("key=value") // => new Tag("key", "value")
* TagUtils.Tag("key=") // => new Tag("key", "") * TagUtils.Tag("key=") // => new Tag("key", "")
* TagUtils.Tag("key!=") // => new RegexTag("key", "^..*$", true) * TagUtils.Tag("key!=") // => new RegexTag("key", "^..*$")
* TagUtils.Tag("key!=value") // => new RegexTag("key", /^value$/, true) * TagUtils.Tag("key!=value") // => new RegexTag("key", /^value$/, true)
* TagUtils.Tag("vending~.*bicycle_tube.*") // => new RegexTag("vending", /^.*bicycle_tube.*$/) * TagUtils.Tag("vending~.*bicycle_tube.*") // => new RegexTag("vending", /^.*bicycle_tube.*$/)
* TagUtils.Tag("x!~y") // => new RegexTag("x", /^y$/, true) * TagUtils.Tag("x!~y") // => new RegexTag("x", /^y$/, true)
@ -298,6 +298,7 @@ export class TagUtils {
} }
if (split[1] === "") { if (split[1] === "") {
split[1] = "..*" split[1] = "..*"
return new RegexTag(split[0], /^..*$/)
} }
return new RegexTag( return new RegexTag(
split[0], split[0],

View file

@ -33,5 +33,21 @@ export abstract class TagsFilter {
*/ */
abstract optimize(): TagsFilter | boolean; abstract optimize(): TagsFilter | boolean;
/**
* Returns 'true' if the tagsfilter might select all features (i.e. the filter will return everything from OSM, except a few entries).
*
* A typical negative tagsfilter is 'key!=value'
*
* import {RegexTag} from "./RegexTag";
* import {Tag} from "./Tag";
* import {And} from "./And";
* import {Or} from "./Or";
*
* new Tag("key","value").isNegative() // => false
* new And([new RegexTag("key","value", true)]).isNegative() // => true
* new Or([new RegexTag("key","value", true), new Tag("x","y")]).isNegative() // => true
* new And([new RegexTag("key","value", true), new Tag("x","y")]).isNegative() // => false
*/
abstract isNegative(): boolean
} }

View file

@ -2,7 +2,7 @@ import {Utils} from "../Utils";
export default class Constants { export default class Constants {
public static vNumber = "0.17.0"; public static vNumber = "0.17.1";
public static ImgurApiKey = '7070e7167f0a25a' public static ImgurApiKey = '7070e7167f0a25a'
public static readonly mapillary_client_token_v4 = "MLY|4441509239301885|b40ad2d3ea105435bd40c7e76993ae85" public static readonly mapillary_client_token_v4 = "MLY|4441509239301885|b40ad2d3ea105435bd40c7e76993ae85"

View file

@ -101,6 +101,10 @@ export default class LayerConfig extends WithContextLoader {
context + "source.osmTags" context + "source.osmTags"
); );
if(Constants.priviliged_layers.indexOf(this.id) < 0 && osmTags.isNegative()){
throw context + "The source states tags which give a very wide selection: it only uses negative expressions, which will result in too much and unexpected data. Add at least one required tag. The tags are:\n\t"+osmTags.asHumanString(false, false, {});
}
if (json.source["geoJsonSource"] !== undefined) { if (json.source["geoJsonSource"] !== undefined) {
throw context + "Use 'geoJson' instead of 'geoJsonSource'"; throw context + "Use 'geoJson' instead of 'geoJsonSource'";
} }
@ -108,6 +112,7 @@ export default class LayerConfig extends WithContextLoader {
if (json.source["geojson"] !== undefined) { if (json.source["geojson"] !== undefined) {
throw context + "Use 'geoJson' instead of 'geojson' (the J is a capital letter)"; throw context + "Use 'geoJson' instead of 'geojson' (the J is a capital letter)";
} }
this.source = new SourceConfig( this.source = new SourceConfig(
{ {