forked from MapComplete/MapComplete
		
	Refactor OsmFeatureSource to increas testability
This commit is contained in:
		
							parent
							
								
									9ebdb81531
								
							
						
					
					
						commit
						06bc136b85
					
				
					 2 changed files with 55 additions and 46 deletions
				
			
		|  | @ -7,7 +7,6 @@ import FilteredLayer from "../../../Models/FilteredLayer"; | ||||||
| import {FeatureSourceForLayer, Tiled} from "../FeatureSource"; | import {FeatureSourceForLayer, Tiled} from "../FeatureSource"; | ||||||
| import {Tiles} from "../../../Models/TileRange"; | import {Tiles} from "../../../Models/TileRange"; | ||||||
| import {BBox} from "../../BBox"; | import {BBox} from "../../BBox"; | ||||||
| import {OsmConnection} from "../../Osm/OsmConnection"; |  | ||||||
| import LayoutConfig from "../../../Models/ThemeConfig/LayoutConfig"; | import LayoutConfig from "../../../Models/ThemeConfig/LayoutConfig"; | ||||||
| import {Or} from "../../Tags/Or"; | import {Or} from "../../Tags/Or"; | ||||||
| import {TagsFilter} from "../../Tags/TagsFilter"; | import {TagsFilter} from "../../Tags/TagsFilter"; | ||||||
|  | @ -27,65 +26,71 @@ export default class OsmFeatureSource { | ||||||
|         handleTile: (tile: FeatureSourceForLayer & Tiled) => void; |         handleTile: (tile: FeatureSourceForLayer & Tiled) => void; | ||||||
|         isActive: UIEventSource<boolean>, |         isActive: UIEventSource<boolean>, | ||||||
|         neededTiles: UIEventSource<number[]>, |         neededTiles: UIEventSource<number[]>, | ||||||
|         state: { |  | ||||||
|             readonly osmConnection: OsmConnection; |  | ||||||
|         }, |  | ||||||
|         markTileVisited?: (tileId: number) => void |         markTileVisited?: (tileId: number) => void | ||||||
|     }; |     }; | ||||||
|     private readonly allowedTags: TagsFilter; |     private readonly allowedTags: TagsFilter; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      *  | ||||||
|  |      * @param options: allowedFeatures is normally calculated from the layoutToUse | ||||||
|  |      */ | ||||||
|     constructor(options: { |     constructor(options: { | ||||||
|         handleTile: (tile: FeatureSourceForLayer & Tiled) => void; |         handleTile: (tile: FeatureSourceForLayer & Tiled) => void; | ||||||
|         isActive: UIEventSource<boolean>, |         isActive: UIEventSource<boolean>, | ||||||
|         neededTiles: UIEventSource<number[]>, |         neededTiles: UIEventSource<number[]>, | ||||||
|         state: { |         state: { | ||||||
|             readonly filteredLayers: UIEventSource<FilteredLayer[]>; |             readonly filteredLayers: UIEventSource<FilteredLayer[]>; | ||||||
|             readonly osmConnection: OsmConnection; |             readonly osmConnection: { | ||||||
|             readonly layoutToUse: LayoutConfig |                 Backend(): string | ||||||
|  |             }; | ||||||
|  |             readonly layoutToUse?: LayoutConfig | ||||||
|         }, |         }, | ||||||
|  |         readonly allowedFeatures?: TagsFilter, | ||||||
|         markTileVisited?: (tileId: number) => void |         markTileVisited?: (tileId: number) => void | ||||||
|     }) { |     }) { | ||||||
|         this.options = options; |         this.options = options; | ||||||
|         this._backend = options.state.osmConnection._oauth_config.url; |         this._backend = options.state.osmConnection.Backend(); | ||||||
|         this.filteredLayers = options.state.filteredLayers.map(layers => layers.filter(layer => layer.layerDef.source.geojsonSource === undefined)) |         this.filteredLayers = options.state.filteredLayers.map(layers => layers.filter(layer => layer.layerDef.source.geojsonSource === undefined)) | ||||||
|         this.handleTile = options.handleTile |         this.handleTile = options.handleTile | ||||||
|         this.isActive = options.isActive |         this.isActive = options.isActive | ||||||
|         const self = this |         const self = this | ||||||
|         options.neededTiles.addCallbackAndRunD(neededTiles => { |         options.neededTiles.addCallbackAndRunD(neededTiles => { | ||||||
|             if (options.isActive?.data === false) { |             self.Update(neededTiles) | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             neededTiles = neededTiles.filter(tile => !self.downloadedTiles.has(tile)) |  | ||||||
| 
 |  | ||||||
|             if (neededTiles.length == 0) { |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             self.isRunning.setData(true) |  | ||||||
|             try { |  | ||||||
| 
 |  | ||||||
|                 for (const neededTile of neededTiles) { |  | ||||||
|                     self.downloadedTiles.add(neededTile) |  | ||||||
|                     self.LoadTile(...Tiles.tile_from_index(neededTile)).then(_ => { |  | ||||||
|                         console.debug("Tile ", Tiles.tile_from_index(neededTile).join("/"), "loaded from OSM") |  | ||||||
|                     }) |  | ||||||
|                 } |  | ||||||
|             } catch (e) { |  | ||||||
|                 console.error(e) |  | ||||||
|             } finally { |  | ||||||
|                 self.isRunning.setData(false) |  | ||||||
|             } |  | ||||||
|         }) |         }) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|         const neededLayers = (options.state.layoutToUse?.layers ?? []) |         const neededLayers = (options.state.layoutToUse?.layers ?? []) | ||||||
|             .filter(layer => !layer.doNotDownload) |             .filter(layer => !layer.doNotDownload) | ||||||
|             .filter(layer => layer.source.geojsonSource === undefined || layer.source.isOsmCacheLayer) |             .filter(layer => layer.source.geojsonSource === undefined || layer.source.isOsmCacheLayer) | ||||||
|         this.allowedTags = new Or(neededLayers.map(l => l.source.osmTags)) |         this.allowedTags = options.allowedFeatures ?? new Or(neededLayers.map(l => l.source.osmTags)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async LoadTile(z, x, y): Promise<void> { |     private async Update(neededTiles: number[]) { | ||||||
|  |         if (this.options.isActive?.data === false) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         neededTiles = neededTiles.filter(tile => !this.downloadedTiles.has(tile)) | ||||||
|  | 
 | ||||||
|  |         if (neededTiles.length == 0) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         this.isRunning.setData(true) | ||||||
|  |         try { | ||||||
|  | 
 | ||||||
|  |             for (const neededTile of neededTiles) { | ||||||
|  |                 this.downloadedTiles.add(neededTile) | ||||||
|  |                 this.LoadTile(...Tiles.tile_from_index(neededTile)) | ||||||
|  |             } | ||||||
|  |         } catch (e) { | ||||||
|  |             console.error(e) | ||||||
|  |         } finally { | ||||||
|  |             this.isRunning.setData(false) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private LoadTile(z, x, y): void { | ||||||
|         if (z > 25) { |         if (z > 25) { | ||||||
|             throw "This is an absurd high zoom level" |             throw "This is an absurd high zoom level" | ||||||
|         } |         } | ||||||
|  | @ -96,11 +101,10 @@ export default class OsmFeatureSource { | ||||||
| 
 | 
 | ||||||
|         const bbox = BBox.fromTile(z, x, y) |         const bbox = BBox.fromTile(z, x, y) | ||||||
|         const url = `${this._backend}/api/0.6/map?bbox=${bbox.minLon},${bbox.minLat},${bbox.maxLon},${bbox.maxLat}` |         const url = `${this._backend}/api/0.6/map?bbox=${bbox.minLon},${bbox.minLat},${bbox.maxLon},${bbox.maxLat}` | ||||||
|         try { |  | ||||||
| 
 | 
 | ||||||
|             const osmJson = await Utils.downloadJson(url) |         Utils.downloadJson(url).then(osmJson => { | ||||||
|             try { |             try { | ||||||
|                 console.debug("Got tile", z, x, y, "from the osm api") |                 console.log("Got tile", z, x, y, "from the osm api") | ||||||
|                 this.rawDataHandlers.forEach(handler => handler(osmJson, Tiles.tile_index(z, x, y))) |                 this.rawDataHandlers.forEach(handler => handler(osmJson, Tiles.tile_index(z, x, y))) | ||||||
|                 const geojson = OsmToGeoJson.default(osmJson, |                 const geojson = OsmToGeoJson.default(osmJson, | ||||||
|                     // @ts-ignore
 |                     // @ts-ignore
 | ||||||
|  | @ -130,17 +134,18 @@ export default class OsmFeatureSource { | ||||||
|             } catch (e) { |             } catch (e) { | ||||||
|                 console.error("Weird error: ", e) |                 console.error("Weird error: ", e) | ||||||
|             } |             } | ||||||
|         } catch (e) { |         }) | ||||||
|  |             .catch(e => { | ||||||
|                 console.error("Could not download tile", z, x, y, "due to", e, "; retrying with smaller bounds") |                 console.error("Could not download tile", z, x, y, "due to", e, "; retrying with smaller bounds") | ||||||
|                 if (e === "rate limited") { |                 if (e === "rate limited") { | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|             await this.LoadTile(z + 1, x * 2, y * 2) |                 this.LoadTile(z + 1, x * 2, y * 2) | ||||||
|             await this.LoadTile(z + 1, 1 + x * 2, y * 2) |                 this.LoadTile(z + 1, 1 + x * 2, y * 2) | ||||||
|             await this.LoadTile(z + 1, x * 2, 1 + y * 2) |                 this.LoadTile(z + 1, x * 2, 1 + y * 2) | ||||||
|             await this.LoadTile(z + 1, 1 + x * 2, 1 + y * 2) |                 this.LoadTile(z + 1, 1 + x * 2, 1 + y * 2) | ||||||
|                 return; |                 return; | ||||||
|         } |             }) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -144,6 +144,10 @@ export class OsmConnection { | ||||||
|         this.loadingStatus.setData("not-attempted") |         this.loadingStatus.setData("not-attempted") | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  |     public Backend(): string { | ||||||
|  |         return this._oauth_config.url; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public AttemptLogin() { |     public AttemptLogin() { | ||||||
|         this.loadingStatus.setData("loading") |         this.loadingStatus.setData("loading") | ||||||
|         if (this.fakeUser) { |         if (this.fakeUser) { | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue