import { Conversion, DesugaringStep, Fuse } from "./Conversion" import { LayerConfigJson } from "../Json/LayerConfigJson" import LayerConfig from "../LayerConfig" import { Utils } from "../../../Utils" import Constants from "../../Constants" import { Translation } from "../../../UI/i18n/Translation" import { ThemeConfigJson } from "../Json/ThemeConfigJson" import ThemeConfig from "../ThemeConfig" import { TagRenderingConfigJson } from "../Json/TagRenderingConfigJson" import { TagUtils } from "../../../Logic/Tags/TagUtils" import { And } from "../../../Logic/Tags/And" import FilterConfigJson from "../Json/FilterConfigJson" import { QuestionableTagRenderingConfigJson } from "../Json/QuestionableTagRenderingConfigJson" import Validators from "../../../UI/InputElement/Validators" import TagRenderingConfig from "../TagRenderingConfig" import { parse as parse_html } from "node-html-parser" import PresetConfig from "../PresetConfig" import { TagsFilter } from "../../../Logic/Tags/TagsFilter" import { Translatable } from "../Json/Translatable" import { ConversionContext } from "./ConversionContext" import PointRenderingConfigJson from "../Json/PointRenderingConfigJson" import { PrevalidateLayer } from "./PrevalidateLayer" import { AvailableRasterLayers } from "../../RasterLayers" import { eliCategory } from "../../RasterLayerProperties" export class ValidateLanguageCompleteness extends DesugaringStep { private readonly _languages: string[] constructor(...languages: string[]) { super( "Checks that the given object is fully translated in the specified languages", [], "ValidateLanguageCompleteness" ) this._languages = languages ?? ["en"] } convert(obj: ThemeConfig, context: ConversionContext): ThemeConfig { const origLayers = obj.layers obj.layers = [...obj.layers].filter((l) => l["id"] !== "favourite") const translations = Translation.ExtractAllTranslationsFrom(obj) for (const neededLanguage of this._languages) { translations .filter( (t) => t.tr.translations[neededLanguage] === undefined && t.tr.translations["*"] === undefined ) .forEach((missing) => { context .enter(missing.context.split(".")) .err( `The theme ${obj.id} should be translation-complete for ` + neededLanguage + ", but it lacks a translation for " + missing.context + ".\n\tThe known translation is " + missing.tr.textFor("en") ) }) } obj.layers = origLayers return obj } } export class DoesImageExist extends DesugaringStep { private readonly _knownImagePaths: Set private readonly _ignore?: Set private readonly doesPathExist: (path: string) => boolean = undefined constructor( knownImagePaths: Set, checkExistsSync: (path: string) => boolean = undefined, ignore?: Set ) { super("Checks if an image exists", [], "DoesImageExist") this._ignore = ignore this._knownImagePaths = knownImagePaths this.doesPathExist = checkExistsSync } convert(image: string, context: ConversionContext): string { if (this._ignore?.has(image)) { return image } if (image.indexOf("{") >= 0) { context.debug("Ignoring image with { in the path: " + image) return image } if (image === "assets/SocialImage.png") { return image } if (image.match(/[a-z]*/)) { if (Constants.defaultPinIcons.indexOf(image) >= 0) { // This is a builtin img, e.g. 'checkmark' or 'crosshair' return image } } if (image.startsWith("<") && image.endsWith(">")) { // This is probably HTML, you're on your own here return image } if (Utils.isEmoji(image)) { return image } if (!this._knownImagePaths.has(image)) { if (this.doesPathExist === undefined || image.indexOf("nsi/logos/") >= 0) { // pass } else if (!this.doesPathExist(image)) { context.err( `Image with path ${image} does not exist.\n Check for typo's and missing directories in the path. ` ) } else { context.err( `Image with path ${image} is not attributed (but it exists); execute 'npm run query:licenses' to add the license information and/or run 'npm run generate:licenses' to compile all the license info` ) } } return image } } class OverrideShadowingCheck extends DesugaringStep { constructor() { super( "Checks that an 'overrideAll' does not override a single override", [], "OverrideShadowingCheck" ) } convert(json: ThemeConfigJson, context: ConversionContext): ThemeConfigJson { const overrideAll = json.overrideAll if (overrideAll === undefined) { return json } const withOverride = json.layers.filter((l) => l["override"] !== undefined) for (const layer of withOverride) { for (const key in overrideAll) { if (key.endsWith("+") || key.startsWith("+")) { // This key will _add_ to the list, not overwrite it - so no warning is needed continue } if ( layer["override"][key] !== undefined || layer["override"]["=" + key] !== undefined ) { const w = "The override of layer " + JSON.stringify(layer["builtin"]) + " has a shadowed property: " + key + " is overriden by overrideAll of the theme" context.err(w) } } } return json } } class MiscThemeChecks extends DesugaringStep { constructor() { super("Miscelleanous checks on the theme", [], "MiscThemesChecks") } convert(json: ThemeConfigJson, context: ConversionContext): ThemeConfigJson { if (json.id !== "personal" && (json.layers === undefined || json.layers.length === 0)) { context.err("The theme " + json.id + " has no 'layers' defined") } if (!Array.isArray(json.layers)) { context .enter("layers") .err( "The 'layers'-field should be an array, but it is not. Did you pase a layer identifier and forget to add the '[' and ']'?" ) } if (json.socialImage === "") { context.warn("Social image for theme " + json.id + " is the emtpy string") } if (json["clustering"]) { context.warn("Obsolete field `clustering` is still around") } if (json.layers === undefined) { context.err("This theme has no layers defined") } else { for (let i = 0; i < json.layers.length; i++) { const l = json.layers[i] if (l["override"]?.["source"] === undefined) { continue } if (l["override"]?.["source"]?.["geoJson"]) { continue // We don't care about external data as we won't cache it anyway } if (l["override"]["id"] !== undefined) { continue } context .enters("layers", i) .err("A layer which changes the source-tags must also change the ID") } } if (json["overideAll"]) { context .enter("overideAll") .err( "'overrideAll' is spelled with _two_ `r`s. You only wrote a single one of them." ) } if ( json.defaultBackgroundId && ![AvailableRasterLayers.osmCartoProperties.id, ...eliCategory].find( (l) => l === json.defaultBackgroundId ) ) { const background = json.defaultBackgroundId const match = AvailableRasterLayers.globalLayers.find( (l) => l.properties.id === background ) if (!match) { const suggestions = Utils.sortedByLevenshteinDistance( background, AvailableRasterLayers.globalLayers, (l) => l.properties.id ) context.enter("defaultBackgroundId").warn( "The default background layer with id", background, "does not exist or is not a global layer. Perhaps you meant one of:", suggestions .slice(0, 5) .map((l) => l.properties.id) .join(", "), "If you want to use a certain category of background image, use", AvailableRasterLayers.globalLayers.join(", ") ) } } return json } } export class PrevalidateTheme extends Fuse { constructor() { super( "Various consistency checks on the raw JSON", new MiscThemeChecks(), new OverrideShadowingCheck() ) } } export class DetectConflictingAddExtraTags extends DesugaringStep { constructor() { super( "The `if`-part in a mapping might set some keys. Those keys are not allowed to be set in the `addExtraTags`, as this might result in conflicting values", [], "DetectConflictingAddExtraTags" ) } convert(json: TagRenderingConfigJson, context: ConversionContext): TagRenderingConfigJson { if (!(json.mappings?.length > 0)) { return json } try { const tagRendering = new TagRenderingConfig(json, context.path.join(".")) for (let i = 0; i < tagRendering.mappings.length; i++) { const mapping = tagRendering.mappings[i] if (!mapping.addExtraTags) { continue } const keysInMapping = new Set(mapping.if.usedKeys()) const keysInAddExtraTags = mapping.addExtraTags.map((t) => t.key) const duplicateKeys = keysInAddExtraTags.filter((k) => keysInMapping.has(k)) if (duplicateKeys.length > 0) { context .enters("mappings", i) .err( "AddExtraTags overrides a key that is set in the `if`-clause of this mapping. Selecting this answer might thus first set one value (needed to match as answer) and then override it with a different value, resulting in an unsaveable question. The offending `addExtraTags` is " + duplicateKeys.join(", ") ) } } return json } catch (e) { context.err("Could not check for conflicting extra tags due to: " + e) return undefined } } } export class DetectNonErasedKeysInMappings extends DesugaringStep { constructor() { super( "A tagRendering might set a freeform key (e.g. `name` and have an option that _should_ erase this name, e.g. `noname=yes`). Under normal circumstances, every mapping/freeform should affect all touched keys", [], "DetectNonErasedKeysInMappings" ) } convert( json: QuestionableTagRenderingConfigJson, context: ConversionContext ): QuestionableTagRenderingConfigJson { if (json.multiAnswer) { // No need to check this here, this has its own validation return json } if (!json.question) { // No need to check the writable tags, as this cannot write return json } function addAll(keys: { forEach: (f: (s: string) => void) => void }, addTo: Set) { keys?.forEach((k) => addTo.add(k)) } const freeformKeys: Set = new Set() if (json.freeform) { freeformKeys.add(json.freeform.key) for (const tag of json.freeform.addExtraTags ?? []) { const tagParsed = TagUtils.Tag(tag) addAll(tagParsed.usedKeys(), freeformKeys) } } const mappingKeys: Set[] = [] for (const mapping of json.mappings ?? []) { if (mapping.hideInAnswer === true) { mappingKeys.push(undefined) continue } const thisMappingKeys: Set = new Set() addAll(TagUtils.Tag(mapping.if).usedKeys(), thisMappingKeys) for (const tag of mapping.addExtraTags ?? []) { addAll(TagUtils.Tag(tag).usedKeys(), thisMappingKeys) } mappingKeys.push(thisMappingKeys) } const neededKeys = new Set() addAll(freeformKeys, neededKeys) for (const mappingKey of mappingKeys) { addAll(mappingKey, neededKeys) } neededKeys.delete("fixme") // gets a free pass if (json.freeform) { for (const neededKey of neededKeys) { if (!freeformKeys.has(neededKey)) { context .enters("freeform") .warn( "The freeform block does not modify the key `" + neededKey + "` which is set in a mapping. Use `addExtraTags` to overwrite it" ) } } } for (let i = 0; i < json.mappings?.length; i++) { const mapping = json.mappings[i] if (mapping.hideInAnswer === true) { continue } const keys = mappingKeys[i] for (const neededKey of neededKeys) { if (!keys.has(neededKey)) { context .enters("mappings", i) .warn( "This mapping does not modify the key `" + neededKey + "` which is set in a mapping or by the freeform block. Use `addExtraTags` to overwrite it" ) } } } return json } } export class DetectMappingsShadowedByCondition extends DesugaringStep { private readonly _forceError: boolean constructor(forceError: boolean = false) { super( "Checks that, if the tagrendering has a condition, that a mapping is not contradictory to it, i.e. that there are no dead mappings", [], "DetectMappingsShadowedByCondition" ) this._forceError = forceError } /** * * const validator = new DetectMappingsShadowedByCondition(true) * const ctx = ConversionContext.construct([],["test"]) * validator.convert({ * condition: "count>0", * mappings:[ * { * if: "count=0", * then:{ * en: "No count" * } * } * ] * }, ctx) * ctx.hasErrors() // => true */ convert(json: TagRenderingConfigJson, context: ConversionContext): TagRenderingConfigJson { if (!json.condition && !json.metacondition) { return json } if (!json.mappings || json.mappings?.length == 0) { return json } let conditionJson = json.condition ?? json.metacondition if (json.condition !== undefined && json.metacondition !== undefined) { conditionJson = { and: [json.condition, json.metacondition] } } const condition = TagUtils.Tag(conditionJson, context.path.join(".")) for (let i = 0; i < json.mappings.length; i++) { const mapping = json.mappings[i] const tagIf = TagUtils.Tag(mapping.if, context.path.join(".")) const optimized = new And([tagIf, condition]).optimize() if (optimized === false) { const msg = "Detected a conflicting mapping and condition. The mapping requires tags " + tagIf.asHumanString() + ", yet this can never happen because the set condition requires " + condition.asHumanString() const ctx = context.enters("mappings", i) if (this._forceError) { ctx.err(msg) } else { ctx.warn(msg) } } } return undefined } } export class DetectShadowedMappings extends DesugaringStep { private readonly _calculatedTagNames: string[] constructor(layerConfig?: LayerConfigJson) { super("Checks that the mappings don't shadow each other", [], "DetectShadowedMappings") this._calculatedTagNames = DetectShadowedMappings.extractCalculatedTagNames(layerConfig) } /** * * DetectShadowedMappings.extractCalculatedTagNames({calculatedTags: ["_abc:=js()"]}) // => ["_abc"] * DetectShadowedMappings.extractCalculatedTagNames({calculatedTags: ["_abc=js()"]}) // => ["_abc"] */ private static extractCalculatedTagNames( layerConfig?: LayerConfigJson | { calculatedTags: string[] } ) { return ( layerConfig?.calculatedTags?.map((ct) => { if (ct.indexOf(":=") >= 0) { return ct.split(":=")[0] } return ct.split("=")[0] }) ?? [] ) } /** * * // should detect a simple shadowed mapping * const tr = {mappings: [ * { * if: {or: ["key=value", "x=y"]}, * then: "Case A" * }, * { * if: "key=value", * then: "Shadowed" * } * ] * } * const context = ConversionContext.test() * const r = new DetectShadowedMappings().convert(tr, context); * context.getAll("error").length // => 1 * context.getAll("error")[0].message.indexOf("The mapping key=value is fully matched by a previous mapping (namely 0)") >= 0 // => true * * const tr = {mappings: [ * { * if: {or: ["key=value", "x=y"]}, * then: "Case A" * }, * { * if: {and: ["key=value", "x=y"]}, * then: "Shadowed" * } * ] * } * const context = ConversionContext.test() * const r = new DetectShadowedMappings().convert(tr, context); * context.getAll("error").length // => 1 * context.getAll("error")[0].message.indexOf("The mapping key=value & x=y is fully matched by a previous mapping (namely 0)") >= 0 // => true */ convert(json: TagRenderingConfigJson, context: ConversionContext): TagRenderingConfigJson { if (json.mappings === undefined || json.mappings.length === 0) { return json } const defaultProperties = {} for (const calculatedTagName of this._calculatedTagNames) { defaultProperties[calculatedTagName] = "some_calculated_tag_value_for_" + calculatedTagName } const parsedConditions = json.mappings.map((m, i) => { const c = context.enters("mappings", i) const ifTags = TagUtils.Tag(m.if, c.enter("if")) const hideInAnswer = m["hideInAnswer"] if (hideInAnswer !== undefined && hideInAnswer !== false && hideInAnswer !== true) { const conditionTags = TagUtils.Tag(hideInAnswer) // Merge the condition too! return new And([conditionTags, ifTags]) } return ifTags }) for (let i = 0; i < json.mappings.length; i++) { if (!parsedConditions[i]?.isUsableAsAnswer()) { // There is no straightforward way to convert this mapping.if into a properties-object, so we simply skip this one // Yes, it might be shadowed, but running this check is to difficult right now continue } const keyValues = parsedConditions[i].asChange(defaultProperties) const properties = {} keyValues.forEach(({ k, v }) => { properties[k] = v }) for (let j = 0; j < i; j++) { const doesMatch = parsedConditions[j].matchesProperties(properties) if ( doesMatch && json.mappings[j]["hideInAnswer"] === true && json.mappings[i]["hideInAnswer"] !== true ) { context.warn( `Mapping ${i} is shadowed by mapping ${j}. However, mapping ${j} has 'hideInAnswer' set, which will result in a different rendering in question-mode.` ) } else if (doesMatch) { // The current mapping is shadowed! context.err(`Mapping ${i} is shadowed by mapping ${j} and will thus never be shown: The mapping ${parsedConditions[i].asHumanString( false, false, {} )} is fully matched by a previous mapping (namely ${j}), which matches: ${parsedConditions[j].asHumanString(false, false, {})}. To fix this problem, you can try to: - Move the shadowed mapping up - Do you want to use a different text in 'question mode'? Add 'hideInAnswer=true' to the first mapping - Use "addExtraTags": ["key=value", ...] in order to avoid a different rendering (e.g. [{"if": "fee=no", "then": "Free to use", "hideInAnswer":true}, {"if": {"and":["fee=no","charge="]}, "then": "Free to use"}] can be replaced by [{"if":"fee=no", "then": "Free to use", "addExtraTags": ["charge="]}] `) } } } return json } } export class ValidatePossibleLinks extends DesugaringStep> { constructor() { super( "Given a possible set of translations, validates that does have `rel='noopener'` set", [], "ValidatePossibleLinks" ) } public isTabnabbingProne(str: string): boolean { const p = parse_html(str) const links = Array.from(p.getElementsByTagName("a")) if (links.length == 0) { return false } for (const link of Array.from(links)) { if (link.getAttribute("target") !== "_blank") { continue } const rel = new Set(link.getAttribute("rel")?.split(" ") ?? []) if (rel.has("noopener")) { continue } const source = link.getAttribute("href") if (source.startsWith("http")) { // No variable part - we assume the link is safe continue } return true } return false } convert( json: string | Record, context: ConversionContext ): string | Record { if (typeof json === "string") { if (this.isTabnabbingProne(json)) { context.err( "The string " + json + " has a link targeting `_blank`, but it doesn't have `rel='noopener'` set. This gives rise to reverse tabnapping" ) } } else { for (const k in json) { if (this.isTabnabbingProne(json[k])) { context.err( `The translation for ${k} '${json[k]}' has a link targeting \`_blank\`, but it doesn't have \`rel='noopener'\` set. This gives rise to reverse tabnapping` ) } } } return json } } export class CheckTranslation extends DesugaringStep { public static readonly allowUndefined: CheckTranslation = new CheckTranslation(true) public static readonly noUndefined: CheckTranslation = new CheckTranslation() private readonly _allowUndefined: boolean constructor(allowUndefined: boolean = false) { super( "Checks that a translation is valid and internally consistent", ["*"], "CheckTranslation" ) this._allowUndefined = allowUndefined } convert(json: Translatable, context: ConversionContext): Translatable { if (json === undefined || json === null) { if (!this._allowUndefined) { context.err("Expected a translation, but got " + json) } return json } if (typeof json === "string") { return json } const keys = Object.keys(json) if (keys.length === 0) { context.err("No actual values are given in this translation, it is completely empty") return json } const en = json["en"] if (!en && json["*"] === undefined) { const msg = "Received a translation without english version" context.warn(msg) } for (const key of keys) { const lng = json[key] if (lng === "") { context.enter(lng).err("Got an empty string in translation for language " + key) } // TODO validate that all subparts are here } return json } } export class ValidateLayerConfig extends DesugaringStep { private readonly validator: ValidateLayer constructor( path: string, isBuiltin: boolean, doesImageExist: DoesImageExist, studioValidations: boolean = false, skipDefaultLayers: boolean = false ) { super("Thin wrapper around 'ValidateLayer", [], "ValidateLayerConfig") this.validator = new ValidateLayer( path, isBuiltin, doesImageExist, studioValidations, skipDefaultLayers ) } convert(json: LayerConfigJson, context: ConversionContext): LayerConfigJson { const prepared = this.validator.convert(json, context) if (!prepared) { context.err("Preparing layer failed") return undefined } return prepared?.raw } } export class ValidatePointRendering extends DesugaringStep { constructor() { super("Various checks for pointRenderings", [], "ValidatePOintRendering") } convert(json: PointRenderingConfigJson, context: ConversionContext): PointRenderingConfigJson { if (json.marker === undefined && json.label === undefined) { context.err(`A point rendering should define at least an marker or a label`) } if (json["markers"]) { context .enter("markers") .err( `Detected a field 'markerS' in pointRendering. It is written as a singular case` ) } if (json.marker && !Array.isArray(json.marker)) { context.enter("marker").err("The marker in a pointRendering should be an array") } if (!(json.location?.length > 0)) { context .enter("location") .err( "A pointRendering should have at least one 'location' to defined where it should be rendered. " ) } return json } } export class ValidateLayer extends Conversion< LayerConfigJson, { parsed: LayerConfig; raw: LayerConfigJson } > { private readonly _skipDefaultLayers: boolean private readonly _prevalidation: PrevalidateLayer constructor( path: string, isBuiltin: boolean, doesImageExist: DoesImageExist, studioValidations: boolean = false, skipDefaultLayers: boolean = false ) { super("Doesn't change anything, but emits warnings and errors", [], "ValidateLayer") this._prevalidation = new PrevalidateLayer( path, isBuiltin, doesImageExist, studioValidations ) this._skipDefaultLayers = skipDefaultLayers } convert( json: LayerConfigJson, context: ConversionContext ): { parsed: LayerConfig; raw: LayerConfigJson } { context = context.inOperation(this.name) if (typeof json === "string") { context.err( `Not a valid layer: the layerConfig is a string. 'npm run generate:layeroverview' might be needed` ) return undefined } if (this._skipDefaultLayers && Constants.added_by_default.indexOf(json.id) >= 0) { return { parsed: undefined, raw: json } } this._prevalidation.convert(json, context.inOperation(this._prevalidation.name)) if (context.hasErrors()) { return undefined } let layerConfig: LayerConfig try { layerConfig = new LayerConfig(json, "validation", true) } catch (e) { console.error("Could not parse layer due to", e) context.err("Could not parse layer due to: " + e) return undefined } for (let i = 0; i < (layerConfig.calculatedTags ?? []).length; i++) { const [_, code, __] = layerConfig.calculatedTags[i] try { new Function("feat", "return " + code + ";") } catch (e) { context .enters("calculatedTags", i) .err( `Invalid function definition: the custom javascript is invalid:${e}. The offending javascript code is:\n ${code}` ) } } for (let i = 0; i < layerConfig.titleIcons.length; i++) { const titleIcon = layerConfig.titleIcons[i] if (titleIcon.render === "icons.defaults") { context.enters("titleIcons", i).err("Detected a literal 'icons.defaults'") } if (titleIcon.render === "icons.rating") { context.enters("titleIcons", i).err("Detected a literal 'icons.rating'") } } for (let i = 0; i < json.presets?.length; i++) { const preset = json.presets[i] if ( preset.snapToLayer === undefined && preset.maxSnapDistance !== undefined && preset.maxSnapDistance !== null ) { context .enters("presets", i, "maxSnapDistance") .err("A maxSnapDistance is given, but there is no layer given to snap to") } } if (json["doCount"]) { context.enters("doCount").err("Use `isCounted` instead of `doCount`") } if (json.source) { const src = json.source if (src["isOsmCache"] !== undefined) { context.enters("source").err("isOsmCache is deprecated") } if (src["maxCacheAge"] !== undefined) { context .enters("source") .err("maxCacheAge is deprecated; it is " + src["maxCacheAge"]) } } if (json.allowMove?.["enableAccuraccy"] !== undefined) { context .enters("allowMove", "enableAccuracy") .err( "`enableAccuracy` is written with two C in the first occurrence and only one in the last" ) } if (typeof json.allowMove === "object") { if (!(json.allowMove.enableRelocation || json.allowMove.enableImproveAccuracy)) { context.warn( "MoveConfig: At least one default move reason should be allowed (at " + context + "); both 'enableImproveAccuracy and enableRelocation are falsy. If you don't want to allow moving points, set `allowMove: false` instead. Full config is: " + JSON.stringify(json.allowMove) ) } } return { raw: json, parsed: layerConfig } } } export class ValidateFilter extends DesugaringStep { constructor() { super("Detect common errors in the filters", [], "ValidateFilter") } convert(filter: FilterConfigJson, context: ConversionContext): FilterConfigJson { if (typeof filter === "string") { // Calling another filter, we skip return filter } if (filter === undefined) { context.err("Trying to validate a filter, but this filter is undefined") return undefined } for (const option of filter.options) { for (let i = 0; i < option.fields?.length ?? 0; i++) { const field = option.fields[i] const type = field.type ?? "string" if (Validators.availableTypes.find((t) => t === type) === undefined) { context .enters("fields", i) .err( `Invalid filter: ${type} is not a valid textfield type.\n\tTry one of ${Array.from( Validators.availableTypes ).join(",")}` ) } } } return filter } } export class DetectDuplicateFilters extends DesugaringStep<{ layers: LayerConfigJson[] themes: ThemeConfigJson[] }> { constructor() { super( "Tries to detect layers where a shared filter can be used (or where similar filters occur)", [], "DetectDuplicateFilters" ) } convert( json: { layers: LayerConfigJson[]; themes: ThemeConfigJson[] }, context: ConversionContext ): { layers: LayerConfigJson[]; themes: ThemeConfigJson[] } { const { layers, themes } = json const perOsmTag = new Map< string, { layer: LayerConfigJson theme: ThemeConfigJson | undefined filter: FilterConfigJson }[] >() for (const layer of layers) { this.addLayerFilters(layer, perOsmTag) } for (const theme of themes) { if (theme.id === "personal") { continue } for (const layer of theme.layers) { if (typeof layer === "string") { continue } if (layer["builtin"] !== undefined) { continue } this.addLayerFilters(layer, perOsmTag, theme) } } // At this point, we have gathered all filters per tag - time to find duplicates perOsmTag.forEach((value, key) => { if (value.length <= 1) { // Seen this key just once, it is unique return } let msg = "Possible duplicate filter: " + key for (const { filter, layer, theme } of value) { let id = "" if (theme !== undefined) { id = theme.id + ":" } msg += `\n - ${id}${layer.id}.${filter.id}` } context.warn(msg) }) return json } /** * Add all filter options into 'perOsmTag' */ private addLayerFilters( layer: LayerConfigJson, perOsmTag: Map< string, { layer: LayerConfigJson theme: ThemeConfigJson | undefined filter: FilterConfigJson }[] >, theme?: ThemeConfigJson | undefined ): void { if (layer.filter === undefined || layer.filter === null) { return } if (layer.filter["sameAs"] !== undefined) { return } for (const filter of <(string | FilterConfigJson)[]>layer.filter) { if (typeof filter === "string") { continue } if (filter["#"]?.indexOf("ignore-possible-duplicate") >= 0) { continue } for (const option of filter.options) { if (option.osmTags === undefined) { continue } const key = JSON.stringify(option.osmTags) if (!perOsmTag.has(key)) { perOsmTag.set(key, []) } perOsmTag.get(key).push({ layer, filter, theme, }) } } } } export class DetectDuplicatePresets extends DesugaringStep { constructor() { super( "Detects mappings which have identical (english) names or identical mappings.", ["presets"], "DetectDuplicatePresets" ) } convert(json: ThemeConfig, context: ConversionContext): ThemeConfig { const presets: PresetConfig[] = [].concat(...json.layers.map((l) => l.presets)) const enNames = presets.map((p) => p.title.textFor("en")) if (new Set(enNames).size != enNames.length) { const dups = Utils.Duplicates(enNames) const layersWithDup = json.layers.filter((l) => l.presets.some((p) => dups.indexOf(p.title.textFor("en")) >= 0) ) const layerIds = layersWithDup.map((l) => l.id) context.err( `This theme has multiple presets which are named:${dups}, namely layers ${layerIds.join( ", " )} this is confusing for contributors and is probably the result of reusing the same layer multiple times. Use \`{"override": {"=presets": []}}\` to remove some presets` ) } const optimizedTags = presets.map((p) => new And(p.tags).optimize()) for (let i = 0; i < presets.length; i++) { const presetATags = optimizedTags[i] const presetA = presets[i] for (let j = i + 1; j < presets.length; j++) { const presetBTags = optimizedTags[j] const presetB = presets[j] if ( Utils.SameObject(presetATags, presetBTags) && Utils.sameList( presetA.preciseInput.snapToLayers, presetB.preciseInput.snapToLayers ) ) { context.err( `This theme has multiple presets with the same tags: ${presetATags.asHumanString( false, false, {} )}, namely the preset '${presets[i].title.textFor("en")}' and '${presets[ j ].title.textFor("en")}'` ) } } } return json } } export class ValidateThemeEnsemble extends Conversion< ThemeConfig[], Map< string, { tags: TagsFilter foundInTheme: string[] isCounted: boolean } > > { constructor() { super( "Validates that all themes together are logical, i.e. no duplicate ids exists within (overriden) themes", [], "ValidateThemeEnsemble" ) } convert( json: ThemeConfig[], context: ConversionContext ): Map< string, { tags: TagsFilter foundInTheme: string[] isCounted: boolean } > { const idToSource = new Map< string, { tags: TagsFilter; foundInTheme: string[]; isCounted: boolean } >() for (const theme of json) { if (theme.id === "personal") { continue } for (const layer of theme.layers) { if (typeof layer.source === "string") { continue } if (Constants.isPriviliged(layer)) { continue } if (!layer.source) { console.log(theme, layer, layer.source) context.enters(theme.id, "layers", "source", layer.id).err("No source defined") continue } if (layer.source.geojsonSource) { continue } const id = layer.id const tags = layer.source.osmTags if (!idToSource.has(id)) { idToSource.set(id, { tags, foundInTheme: [theme.id], isCounted: layer.doCount }) continue } const oldTags = idToSource.get(id).tags const oldTheme = idToSource.get(id).foundInTheme if (oldTags.shadows(tags) && tags.shadows(oldTags)) { // All is good, all is well oldTheme.push(theme.id) idToSource.get(id).isCounted ||= layer.doCount continue } context.err( [ "The layer with id '" + id + "' is found in multiple themes with different tag definitions:", "\t In theme " + oldTheme + ":\t" + oldTags.asHumanString(false, false, {}), "\tIn theme " + theme.id + ":\t" + tags.asHumanString(false, false, {}), ].join("\n") ) } } return idToSource } }