forked from MapComplete/MapComplete
		
	Wire 'doesImageExist' everywhere, fixes #928
This commit is contained in:
		
							parent
							
								
									a1ea9afac9
								
							
						
					
					
						commit
						b64a873c40
					
				
					 2 changed files with 50 additions and 61 deletions
				
			
		|  | @ -45,14 +45,15 @@ class ValidateLanguageCompleteness extends DesugaringStep<any> { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export class DoesImageExist extends DesugaringStep<string>{ | export class DoesImageExist extends DesugaringStep<string> { | ||||||
| 
 | 
 | ||||||
|     private readonly _knownImagePaths: Set<string>; |     private readonly _knownImagePaths: Set<string>; | ||||||
|     public static doesPathExist : (path: string) => boolean = undefined; |     private readonly doesPathExist: (path: string) => boolean = undefined; | ||||||
| 
 | 
 | ||||||
|     constructor(knownImagePaths: Set<string>) { |     constructor(knownImagePaths: Set<string>, checkExistsSync: (path: string) => boolean = undefined) { | ||||||
|         super("Checks if an image exists", [], "DoesImageExist"); |         super("Checks if an image exists", [], "DoesImageExist"); | ||||||
|         this._knownImagePaths = knownImagePaths; |         this._knownImagePaths = knownImagePaths; | ||||||
|  |         this.doesPathExist = checkExistsSync; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     convert(image: string, context: string): { result: string; errors?: string[]; warnings?: string[]; information?: string[] } { |     convert(image: string, context: string): { result: string; errors?: string[]; warnings?: string[]; information?: string[] } { | ||||||
|  | @ -76,11 +77,11 @@ export class DoesImageExist extends DesugaringStep<string>{ | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (this._knownImagePaths !== undefined && !this._knownImagePaths.has(image)) { |         if (this._knownImagePaths !== undefined && !this._knownImagePaths.has(image)) { | ||||||
|             if(DoesImageExist.doesPathExist === undefined){ |             if (this.doesPathExist === undefined) { | ||||||
|                 errors.push(`Image with path ${image} not found or not attributed; it is used in ${context}`) |                 errors.push(`Image with path ${image} not found or not attributed; it is used in ${context}`) | ||||||
|             }else if(!DoesImageExist.doesPathExist(image)){ |             } else if (!this.doesPathExist(image)) { | ||||||
|                 errors.push(`Image with path ${image} does not exist; it is used in ${context}.\n     Check for typo's and missing directories in the path.`) |                 errors.push(`Image with path ${image} does not exist; it is used in ${context}.\n     Check for typo's and missing directories in the path.`) | ||||||
|             }else{ |             } else { | ||||||
|                 errors.push(`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`) |                 errors.push(`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`) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | @ -98,17 +99,16 @@ class ValidateTheme extends DesugaringStep<LayoutConfigJson> { | ||||||
|      * @private |      * @private | ||||||
|      */ |      */ | ||||||
|     private readonly _path?: string; |     private readonly _path?: string; | ||||||
|     private readonly knownImagePaths: Set<string>; |  | ||||||
|     private readonly _isBuiltin: boolean; |     private readonly _isBuiltin: boolean; | ||||||
|     private _sharedTagRenderings: Map<string, any>; |     private _sharedTagRenderings: Map<string, any>; | ||||||
|     private readonly _validateImage : DesugaringStep<string>; |     private readonly _validateImage: DesugaringStep<string>; | ||||||
|     constructor(knownImagePaths: Set<string>, path: string, isBuiltin: boolean, sharedTagRenderings: Map<string, any>) { | 
 | ||||||
|  |     constructor(doesImageExist: DoesImageExist, path: string, isBuiltin: boolean, sharedTagRenderings: Map<string, any>) { | ||||||
|         super("Doesn't change anything, but emits warnings and errors", [], "ValidateTheme"); |         super("Doesn't change anything, but emits warnings and errors", [], "ValidateTheme"); | ||||||
|         this.knownImagePaths = knownImagePaths; |         this._validateImage = doesImageExist; | ||||||
|         this._path = path; |         this._path = path; | ||||||
|         this._isBuiltin = isBuiltin; |         this._isBuiltin = isBuiltin; | ||||||
|         this._sharedTagRenderings = sharedTagRenderings; |         this._sharedTagRenderings = sharedTagRenderings; | ||||||
|         this._validateImage = new DoesImageExist(this.knownImagePaths); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     convert(json: LayoutConfigJson, context: string): { result: LayoutConfigJson; errors: string[], warnings: string[], information: string[] } { |     convert(json: LayoutConfigJson, context: string): { result: LayoutConfigJson; errors: string[], warnings: string[], information: string[] } { | ||||||
|  | @ -179,9 +179,7 @@ class ValidateTheme extends DesugaringStep<LayoutConfigJson> { | ||||||
|             if (theme.id !== filename) { |             if (theme.id !== filename) { | ||||||
|                 errors.push("Theme ids should be the same as the name.json, but we got id: " + theme.id + " and filename " + filename + " (" + this._path + ")") |                 errors.push("Theme ids should be the same as the name.json, but we got id: " + theme.id + " and filename " + filename + " (" + this._path + ")") | ||||||
|             } |             } | ||||||
|             if (!this.knownImagePaths.has(theme.icon)) { |             this._validateImage.convertJoin(theme.icon, context + ".icon", errors, warnings, information); | ||||||
|                 errors.push("The theme image " + theme.icon + " is not attributed or not saved locally") |  | ||||||
|             } |  | ||||||
|             const dups = Utils.Dupiclates(json.layers.map(layer => layer["id"])) |             const dups = Utils.Dupiclates(json.layers.map(layer => layer["id"])) | ||||||
|             if (dups.length > 0) { |             if (dups.length > 0) { | ||||||
|                 errors.push(`The theme ${json.id} defines multiple layers with id ${dups.join(", ")}`) |                 errors.push(`The theme ${json.id} defines multiple layers with id ${dups.join(", ")}`) | ||||||
|  | @ -195,7 +193,7 @@ class ValidateTheme extends DesugaringStep<LayoutConfigJson> { | ||||||
| 
 | 
 | ||||||
|                 // The first key in the the title-field must be english, otherwise the title in the loading page will be the different language
 |                 // The first key in the the title-field must be english, otherwise the title in the loading page will be the different language
 | ||||||
|                 const targetLanguage = theme.title.SupportedLanguages()[0] |                 const targetLanguage = theme.title.SupportedLanguages()[0] | ||||||
|                 if(targetLanguage !== "en"){ |                 if (targetLanguage !== "en") { | ||||||
|                     warnings.push(`TargetLanguage is not 'en' for public theme ${theme.id}, it is ${targetLanguage}. Move 'en' up in the title of the theme and set it as the first key`) |                     warnings.push(`TargetLanguage is not 'en' for public theme ${theme.id}, it is ${targetLanguage}. Move 'en' up in the title of the theme and set it as the first key`) | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|  | @ -221,10 +219,10 @@ class ValidateTheme extends DesugaringStep<LayoutConfigJson> { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export class ValidateThemeAndLayers extends Fuse<LayoutConfigJson> { | export class ValidateThemeAndLayers extends Fuse<LayoutConfigJson> { | ||||||
|     constructor(knownImagePaths: Set<string>, path: string, isBuiltin: boolean, sharedTagRenderings: Map<string, any>) { |     constructor(doesImageExist: DoesImageExist, path: string, isBuiltin: boolean, sharedTagRenderings: Map<string, any>) { | ||||||
|         super("Validates a theme and the contained layers", |         super("Validates a theme and the contained layers", | ||||||
|             new ValidateTheme(knownImagePaths, path, isBuiltin, sharedTagRenderings), |             new ValidateTheme(doesImageExist, path, isBuiltin, sharedTagRenderings), | ||||||
|             new On("layers", new Each(new ValidateLayer(undefined, false, knownImagePaths))) |             new On("layers", new Each(new ValidateLayer(undefined, false, doesImageExist))) | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -383,7 +381,7 @@ export class DetectShadowedMappings extends DesugaringStep<QuestionableTagRender | ||||||
|             }) |             }) | ||||||
|             for (let j = 0; j < i; j++) { |             for (let j = 0; j < i; j++) { | ||||||
|                 const doesMatch = parsedConditions[j].matchesProperties(properties) |                 const doesMatch = parsedConditions[j].matchesProperties(properties) | ||||||
|                 if(doesMatch && json.mappings[j].hideInAnswer === true && json.mappings[i].hideInAnswer !== true){ |                 if (doesMatch && json.mappings[j].hideInAnswer === true && json.mappings[i].hideInAnswer !== true) { | ||||||
|                     warnings.push(`At ${context}: Mapping ${i} is shadowed by mapping ${j}. However, mapping ${j} has 'hideInAnswer' set, which will result in a different rendering in question-mode.`) |                     warnings.push(`At ${context}: 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) { |                 } else if (doesMatch) { | ||||||
|                     // The current mapping is shadowed!
 |                     // The current mapping is shadowed!
 | ||||||
|  | @ -414,10 +412,11 @@ export class DetectShadowedMappings extends DesugaringStep<QuestionableTagRender | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export class DetectMappingsWithImages extends DesugaringStep<TagRenderingConfigJson> { | export class DetectMappingsWithImages extends DesugaringStep<TagRenderingConfigJson> { | ||||||
|     private knownImagePaths: Set<string>; |     private readonly _doesImageExist: DoesImageExist; | ||||||
|     constructor(knownImagePaths: Set<string>) { | 
 | ||||||
|  |     constructor(doesImageExist: DoesImageExist) { | ||||||
|         super("Checks that 'then'clauses in mappings don't have images, but use 'icon' instead", [], "DetectMappingsWithImages"); |         super("Checks that 'then'clauses in mappings don't have images, but use 'icon' instead", [], "DetectMappingsWithImages"); | ||||||
|         this.knownImagePaths = knownImagePaths; |         this._doesImageExist = doesImageExist; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -441,9 +440,9 @@ export class DetectMappingsWithImages extends DesugaringStep<TagRenderingConfigJ | ||||||
|      * r.errors.some(msg => msg.indexOf("./assets/layers/bike_parking/staple.svg") >= 0) // => true
 |      * r.errors.some(msg => msg.indexOf("./assets/layers/bike_parking/staple.svg") >= 0) // => true
 | ||||||
|      */ |      */ | ||||||
|     convert(json: TagRenderingConfigJson, context: string): { result: TagRenderingConfigJson; errors?: string[]; warnings?: string[], information?: string[] } { |     convert(json: TagRenderingConfigJson, context: string): { result: TagRenderingConfigJson; errors?: string[]; warnings?: string[], information?: string[] } { | ||||||
|         const errors = [] |         const errors: string[] = [] | ||||||
|         const warnings = [] |         const warnings: string[] = [] | ||||||
|         const information = [] |         const information: string[] = [] | ||||||
|         if (json.mappings === undefined || json.mappings.length === 0) { |         if (json.mappings === undefined || json.mappings.length === 0) { | ||||||
|             return {result: json} |             return {result: json} | ||||||
|         } |         } | ||||||
|  | @ -461,13 +460,8 @@ export class DetectMappingsWithImages extends DesugaringStep<TagRenderingConfigJ | ||||||
|                     information.push(`${ctx}: Ignored image ${images.join(", ")} in 'then'-clause of a mapping as this check has been disabled`) |                     information.push(`${ctx}: Ignored image ${images.join(", ")} in 'then'-clause of a mapping as this check has been disabled`) | ||||||
| 
 | 
 | ||||||
|                     for (const image of images) { |                     for (const image of images) { | ||||||
|                         if (this.knownImagePaths !== undefined && !this.knownImagePaths.has(image)) { |                         this._doesImageExist.convertJoin(image, ctx, errors, warnings, information); | ||||||
| 
 | 
 | ||||||
|                             const closeNames = Utils.sortedByLevenshteinDistance(image, Array.from(this.knownImagePaths), i => i); |  | ||||||
|                              |  | ||||||
|                             const ctx = context === undefined ? "" : ` in a layer defined in the theme ${context}` |  | ||||||
|                             errors.push(`Image with path ${image} not found or not attributed; it is used in ${json.id}${ctx}.\n    Did you mean one of ${closeNames.slice(0, 3).join(", ")}`) |  | ||||||
|                         } |  | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|                 } |                 } | ||||||
|  | @ -486,10 +480,10 @@ export class DetectMappingsWithImages extends DesugaringStep<TagRenderingConfigJ | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export class ValidateTagRenderings extends Fuse<TagRenderingConfigJson> { | export class ValidateTagRenderings extends Fuse<TagRenderingConfigJson> { | ||||||
|     constructor(layerConfig?: LayerConfigJson, knownImagePaths?: Set<string>) { |     constructor(layerConfig?: LayerConfigJson, doesImageExist?: DoesImageExist) { | ||||||
|         super("Various validation on tagRenderingConfigs", |         super("Various validation on tagRenderingConfigs", | ||||||
|             new DetectShadowedMappings(layerConfig), |             new DetectShadowedMappings(layerConfig), | ||||||
|             new DetectMappingsWithImages(knownImagePaths) |             new DetectMappingsWithImages(doesImageExist) | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -501,13 +495,13 @@ export class ValidateLayer extends DesugaringStep<LayerConfigJson> { | ||||||
|      */ |      */ | ||||||
|     private readonly _path?: string; |     private readonly _path?: string; | ||||||
|     private readonly _isBuiltin: boolean; |     private readonly _isBuiltin: boolean; | ||||||
|     private knownImagePaths: Set<string> | undefined; |     private readonly _doesImageExist: DoesImageExist; | ||||||
| 
 | 
 | ||||||
|     constructor(path: string, isBuiltin: boolean, knownImagePaths: Set<string>) { |     constructor(path: string, isBuiltin: boolean, doesImageExist: DoesImageExist) { | ||||||
|         super("Doesn't change anything, but emits warnings and errors", [], "ValidateLayer"); |         super("Doesn't change anything, but emits warnings and errors", [], "ValidateLayer"); | ||||||
|         this._path = path; |         this._path = path; | ||||||
|         this._isBuiltin = isBuiltin; |         this._isBuiltin = isBuiltin; | ||||||
|         this.knownImagePaths = knownImagePaths |         this._doesImageExist = doesImageExist | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     convert(json: LayerConfigJson, context: string): { result: LayerConfigJson; errors: string[]; warnings?: string[], information?: string[] } { |     convert(json: LayerConfigJson, context: string): { result: LayerConfigJson; errors: string[]; warnings?: string[], information?: string[] } { | ||||||
|  | @ -595,7 +589,7 @@ export class ValidateLayer extends DesugaringStep<LayerConfigJson> { | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             if (json.tagRenderings !== undefined) { |             if (json.tagRenderings !== undefined) { | ||||||
|                 const r = new On("tagRenderings", new Each(new ValidateTagRenderings(json, this.knownImagePaths))).convert(json, context) |                 const r = new On("tagRenderings", new Each(new ValidateTagRenderings(json, this._doesImageExist))).convert(json, context) | ||||||
|                 warnings.push(...(r.warnings ?? [])) |                 warnings.push(...(r.warnings ?? [])) | ||||||
|                 errors.push(...(r.errors ?? [])) |                 errors.push(...(r.errors ?? [])) | ||||||
|                 information.push(...(r.information ?? [])) |                 information.push(...(r.information ?? [])) | ||||||
|  |  | ||||||
|  | @ -83,10 +83,10 @@ class LayerOverviewUtils { | ||||||
|         writeFileSync(`./assets/generated/layers/${layer.id}.json`, JSON.stringify(layer, null, "  "), "UTF8"); |         writeFileSync(`./assets/generated/layers/${layer.id}.json`, JSON.stringify(layer, null, "  "), "UTF8"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     getSharedTagRenderings(knownImagePaths: Set<string>): Map<string, TagRenderingConfigJson> { |     getSharedTagRenderings(doesImageExist: DoesImageExist): Map<string, TagRenderingConfigJson> { | ||||||
|         const dict = new Map<string, TagRenderingConfigJson>(); |         const dict = new Map<string, TagRenderingConfigJson>(); | ||||||
|          |          | ||||||
|         const validator = new ValidateTagRenderings(undefined, knownImagePaths); |         const validator = new ValidateTagRenderings(undefined, doesImageExist); | ||||||
|         for (const key in questions["default"]) { |         for (const key in questions["default"]) { | ||||||
|             if (key === "id") { |             if (key === "id") { | ||||||
|                 continue |                 continue | ||||||
|  | @ -153,15 +153,13 @@ class LayerOverviewUtils { | ||||||
| 
 | 
 | ||||||
|     main(_: string[]) { |     main(_: string[]) { | ||||||
| 
 | 
 | ||||||
|         DoesImageExist.doesPathExist = (path) => existsSync(path) |  | ||||||
|          |  | ||||||
|         const licensePaths = new Set<string>() |         const licensePaths = new Set<string>() | ||||||
|         for (const i in licenses) { |         for (const i in licenses) { | ||||||
|             licensePaths.add(licenses[i].path) |             licensePaths.add(licenses[i].path) | ||||||
|         } |         } | ||||||
| 
 |         const doesImageExist = new DoesImageExist(licensePaths, existsSync) | ||||||
|         const sharedLayers = this.buildLayerIndex(licensePaths); |         const sharedLayers = this.buildLayerIndex(doesImageExist); | ||||||
|         const sharedThemes = this.buildThemeIndex(licensePaths, sharedLayers) |         const sharedThemes = this.buildThemeIndex(doesImageExist, sharedLayers) | ||||||
| 
 | 
 | ||||||
|         writeFileSync("./assets/generated/known_layers_and_themes.json", JSON.stringify({ |         writeFileSync("./assets/generated/known_layers_and_themes.json", JSON.stringify({ | ||||||
|             "layers": Array.from(sharedLayers.values()), |             "layers": Array.from(sharedLayers.values()), | ||||||
|  | @ -191,12 +189,12 @@ class LayerOverviewUtils { | ||||||
|         console.log(green("All done!")) |         console.log(green("All done!")) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private buildLayerIndex(knownImagePaths: Set<string>): Map<string, LayerConfigJson> { |     private buildLayerIndex(doesImageExist: DoesImageExist): Map<string, LayerConfigJson> { | ||||||
|         // First, we expand and validate all builtin layers. These are written to assets/generated/layers
 |         // First, we expand and validate all builtin layers. These are written to assets/generated/layers
 | ||||||
|         // At the same time, an index of available layers is built.
 |         // At the same time, an index of available layers is built.
 | ||||||
|         console.log("   ---------- VALIDATING BUILTIN LAYERS ---------") |         console.log("   ---------- VALIDATING BUILTIN LAYERS ---------") | ||||||
| 
 | 
 | ||||||
|         const sharedTagRenderings = this.getSharedTagRenderings(knownImagePaths); |         const sharedTagRenderings = this.getSharedTagRenderings(doesImageExist); | ||||||
|         const layerFiles = ScriptUtils.getLayerFiles(); |         const layerFiles = ScriptUtils.getLayerFiles(); | ||||||
|         const sharedLayers = new Map<string, LayerConfigJson>() |         const sharedLayers = new Map<string, LayerConfigJson>() | ||||||
|         const state: DesugaringContext = { |         const state: DesugaringContext = { | ||||||
|  | @ -212,7 +210,7 @@ class LayerOverviewUtils { | ||||||
|                 fixed.source.osmTags = {"and": [fixed.source.osmTags]} |                 fixed.source.osmTags = {"and": [fixed.source.osmTags]} | ||||||
|             } |             } | ||||||
|              |              | ||||||
|             const validator = new ValidateLayer(sharedLayerJson.path, true, knownImagePaths); |             const validator = new ValidateLayer(sharedLayerJson.path, true, doesImageExist); | ||||||
|             validator.convertStrict(fixed, context) |             validator.convertStrict(fixed, context) | ||||||
| 
 | 
 | ||||||
|             if (sharedLayers.has(fixed.id)) { |             if (sharedLayers.has(fixed.id)) { | ||||||
|  | @ -252,7 +250,7 @@ class LayerOverviewUtils { | ||||||
|         return publicLayerIds |         return publicLayerIds | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private buildThemeIndex(knownImagePaths: Set<string>, sharedLayers: Map<string, LayerConfigJson>): Map<string, LayoutConfigJson> { |     private buildThemeIndex(doesImageExist: DoesImageExist, sharedLayers: Map<string, LayerConfigJson>): Map<string, LayoutConfigJson> { | ||||||
|         console.log("   ---------- VALIDATING BUILTIN THEMES ---------") |         console.log("   ---------- VALIDATING BUILTIN THEMES ---------") | ||||||
|         const themeFiles = ScriptUtils.getThemeFiles(); |         const themeFiles = ScriptUtils.getThemeFiles(); | ||||||
|         const fixed = new Map<string, LayoutConfigJson>(); |         const fixed = new Map<string, LayoutConfigJson>(); | ||||||
|  | @ -261,7 +259,7 @@ class LayerOverviewUtils { | ||||||
| 
 | 
 | ||||||
|         const convertState: DesugaringContext = { |         const convertState: DesugaringContext = { | ||||||
|             sharedLayers, |             sharedLayers, | ||||||
|             tagRenderings: this.getSharedTagRenderings(knownImagePaths), |             tagRenderings: this.getSharedTagRenderings(doesImageExist), | ||||||
|             publicLayers |             publicLayers | ||||||
|         } |         } | ||||||
|         for (const themeInfo of themeFiles) { |         for (const themeInfo of themeFiles) { | ||||||
|  | @ -273,10 +271,7 @@ class LayerOverviewUtils { | ||||||
| 
 | 
 | ||||||
|                 themeFile = new PrepareTheme(convertState).convertStrict(themeFile, themePath) |                 themeFile = new PrepareTheme(convertState).convertStrict(themeFile, themePath) | ||||||
| 
 | 
 | ||||||
|                 if (knownImagePaths === undefined) { |                 new ValidateThemeAndLayers(doesImageExist, themePath, true, convertState.tagRenderings) | ||||||
|                     throw "Could not load known images/licenses" |  | ||||||
|                 } |  | ||||||
|                 new ValidateThemeAndLayers(knownImagePaths, themePath, true, convertState.tagRenderings) |  | ||||||
|                     .convertStrict(themeFile, themePath) |                     .convertStrict(themeFile, themePath) | ||||||
| 
 | 
 | ||||||
|                 this.writeTheme(themeFile) |                 this.writeTheme(themeFile) | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue