MapComplete/Logic/FeatureSource/Sources/OsmFeatureSource.ts

213 lines
7.8 KiB
TypeScript
Raw Normal View History

2022-09-08 21:40:48 +02:00
import { Utils } from "../../../Utils"
2023-01-16 15:05:22 +01:00
import OsmToGeoJson from "osmtogeojson"
import { ImmutableStore, Store, UIEventSource } from "../../UIEventSource"
2022-09-08 21:40:48 +02:00
import { Tiles } from "../../../Models/TileRange"
import { BBox } from "../../BBox"
import { TagsFilter } from "../../Tags/TagsFilter"
import { Feature } from "geojson"
import FeatureSourceMerger from "../Sources/FeatureSourceMerger"
import OsmObjectDownloader from "../../Osm/OsmObjectDownloader"
2023-06-01 02:52:21 +02:00
import FullNodeDatabaseSource from "../TiledFeatureSource/FullNodeDatabaseSource";
/**
* If a tile is needed (requested via the UIEventSource in the constructor), will download the appropriate tile and pass it via 'handleTile'
*/
export default class OsmFeatureSource extends FeatureSourceMerger {
private readonly _bounds: Store<BBox>
private readonly isActive: Store<boolean>
2022-09-08 21:40:48 +02:00
private readonly _backend: string
private readonly allowedTags: TagsFilter
2023-06-01 02:52:21 +02:00
private options: {
bounds: Store<BBox>
readonly allowedFeatures: TagsFilter
backend?: "https://openstreetmap.org/" | string
/**
* If given: this featureSwitch will not update if the store contains 'false'
*/
isActive?: Store<boolean>,
patchRelations?: true | boolean,
fullNodeDatabase?: FullNodeDatabaseSource
};
public readonly isRunning: UIEventSource<boolean> = new UIEventSource<boolean>(false)
private readonly _downloadedTiles: Set<number> = new Set<number>()
private readonly _downloadedData: Feature[][] = []
private readonly _patchRelations: boolean;
/**
* Downloads data directly from the OSM-api within the given bounds.
* All features which match the TagsFilter 'allowedFeatures' are kept and converted into geojson
*/
constructor(options: {
bounds: Store<BBox>
readonly allowedFeatures: TagsFilter
backend?: "https://openstreetmap.org/" | string
/**
* If given: this featureSwitch will not update if the store contains 'false'
*/
isActive?: Store<boolean>,
2023-06-01 02:52:21 +02:00
patchRelations?: true | boolean,
fullNodeDatabase?: FullNodeDatabaseSource
}) {
super()
2023-06-01 02:52:21 +02:00
this.options = options;
this._bounds = options.bounds
this.allowedTags = options.allowedFeatures
this.isActive = options.isActive ?? new ImmutableStore(true)
this._backend = options.backend ?? "https://www.openstreetmap.org"
this._bounds.addCallbackAndRunD((bbox) => this.loadData(bbox))
this._patchRelations = options?.patchRelations ?? true
}
private async loadData(bbox: BBox) {
if (this.isActive?.data === false) {
console.log("OsmFeatureSource: not triggering: inactive")
2022-09-08 21:40:48 +02:00
return
}
const z = 15
const neededTiles = Tiles.tileRangeFrom(bbox, z)
if (neededTiles.total == 0) {
2022-09-08 21:40:48 +02:00
return
}
this.isRunning.setData(true)
try {
const tileNumbers = Tiles.MapRange(neededTiles, (x, y) => {
return Tiles.tile_index(z, x, y)
})
await Promise.all(tileNumbers.map((i) => this.LoadTile(...Tiles.tile_from_index(i))))
} catch (e) {
console.error(e)
} finally {
this.isRunning.setData(false)
}
}
private registerFeatures(features: Feature[]): void {
this._downloadedData.push(features)
super.addData(this._downloadedData)
}
/**
* The requested tile might only contain part of the relation.
*
* This method will download the full relation and return it as geojson if it was incomplete.
* If the feature is already complete (or is not a relation), the feature will be returned as is
*/
2022-09-08 21:40:48 +02:00
private async patchIncompleteRelations(
feature: { properties: { id: string } },
originalJson: { elements: { type: "node" | "way" | "relation"; id: number }[] }
): Promise<any> {
if (!feature.properties.id.startsWith("relation") || !this._patchRelations) {
return feature
}
2022-09-08 21:40:48 +02:00
const relationSpec = originalJson.elements.find(
(f) => "relation/" + f.id === feature.properties.id
)
const members: { type: string; ref: number }[] = relationSpec["members"]
for (const member of members) {
2022-09-08 21:40:48 +02:00
const isFound = originalJson.elements.some(
(f) => f.id === member.ref && f.type === member.type
)
if (isFound) {
continue
}
2022-09-08 21:40:48 +02:00
// This member is missing. We redownload the entire relation instead
2022-09-08 21:40:48 +02:00
console.debug("Fetching incomplete relation " + feature.properties.id)
const dfeature = await new OsmObjectDownloader(this._backend).DownloadObjectAsync(
feature.properties.id
)
if (dfeature === "deleted") {
console.warn(
"This relation has been deleted in the meantime: ",
feature.properties.id
)
return
}
return dfeature.asGeoJson()
}
2022-09-08 21:40:48 +02:00
return feature
}
2023-06-01 02:52:21 +02:00
private async LoadTile(z: number, x: number, y: number): Promise<void> {
console.log("OsmFeatureSource: loading ", z, x, y, "from", this._backend)
if (z >= 22) {
throw "This is an absurd high zoom level"
}
2022-01-26 21:40:38 +01:00
2023-04-26 18:04:42 +02:00
if (z < 15) {
throw `Zoom ${z} is too much for OSM to handle! Use a higher zoom level!`
}
const index = Tiles.tile_index(z, x, y)
if (this._downloadedTiles.has(index)) {
return
}
this._downloadedTiles.add(index)
const bbox = BBox.fromTile(z, x, y)
const url = `${this._backend}/api/0.6/map?bbox=${bbox.minLon},${bbox.minLat},${bbox.maxLon},${bbox.maxLat}`
2022-09-08 21:40:48 +02:00
let error = undefined
try {
2023-03-28 05:13:48 +02:00
const osmJson = await Utils.downloadJsonCached(url, 2000)
try {
2023-06-01 02:52:21 +02:00
this.options?.fullNodeDatabase?.handleOsmJson(osmJson, z, x, y)
let features = <Feature<any, { id: string }>[]>OsmToGeoJson(
2022-09-08 21:40:48 +02:00
osmJson,
// @ts-ignore
{
2022-09-08 21:40:48 +02:00
flatProperties: true,
}
).features
// The geojson contains _all_ features at the given location
// We only keep what is needed
features = features.filter((feature) =>
2022-09-08 21:40:48 +02:00
this.allowedTags.matchesProperties(feature.properties)
)
for (let i = 0; i < features.length; i++) {
features[i] = await this.patchIncompleteRelations(features[i], osmJson)
}
features = Utils.NoNull(features)
features.forEach((f) => {
2021-12-17 19:28:05 +01:00
f.properties["_backend"] = this._backend
})
this.registerFeatures(features)
2022-09-08 21:40:48 +02:00
} catch (e) {
console.error(
"PANIC: got the tile from the OSM-api, but something crashed handling this tile"
)
error = e
}
} catch (e) {
2022-09-08 21:40:48 +02:00
console.error(
"Could not download tile",
z,
x,
y,
"due to",
e,
2023-06-01 02:52:21 +02:00
e === "rate limited" ? "; stopping now" : "; retrying with smaller bounds"
2022-09-08 21:40:48 +02:00
)
if (e === "rate limited") {
2022-09-08 21:40:48 +02:00
return
}
await Promise.all([
this.LoadTile(z + 1, x * 2, y * 2),
this.LoadTile(z + 1, 1 + x * 2, y * 2),
this.LoadTile(z + 1, x * 2, 1 + y * 2),
this.LoadTile(z + 1, 1 + x * 2, 1 + y * 2),
])
}
2022-09-08 21:40:48 +02:00
if (error !== undefined) {
throw error
}
}
2022-09-08 21:40:48 +02:00
}