forked from MapComplete/MapComplete
Scripts: add velopark export script
This commit is contained in:
parent
dbff4c9b83
commit
e731c677b9
2 changed files with 80 additions and 19 deletions
55
scripts/velopark/veloParkToGeojson.ts
Normal file
55
scripts/velopark/veloParkToGeojson.ts
Normal file
|
@ -0,0 +1,55 @@
|
||||||
|
import Script from "../Script"
|
||||||
|
import { Utils } from "../../src/Utils"
|
||||||
|
import VeloparkLoader, { VeloparkData } from "../../src/Logic/Web/VeloparkLoader"
|
||||||
|
import fs from "fs"
|
||||||
|
import OverpassFeatureSource from "../../src/Logic/FeatureSource/Sources/OverpassFeatureSource"
|
||||||
|
import { Overpass } from "../../src/Logic/Osm/Overpass"
|
||||||
|
import { RegexTag } from "../../src/Logic/Tags/RegexTag"
|
||||||
|
import Constants from "../../src/Models/Constants"
|
||||||
|
import { ImmutableStore } from "../../src/Logic/UIEventSource"
|
||||||
|
import { BBox } from "../../src/Logic/BBox"
|
||||||
|
class VeloParkToGeojson extends Script {
|
||||||
|
constructor() {
|
||||||
|
super("Downloads the latest Velopark data and converts it to a geojson, which will be saved at the current directory")
|
||||||
|
}
|
||||||
|
|
||||||
|
async main(args: string[]): Promise<void> {
|
||||||
|
console.log("Downloading velopark data")
|
||||||
|
// Download data for NIS-code 1000. 1000 means: all of belgium
|
||||||
|
const url = "https://www.velopark.be/api/parkings/1000"
|
||||||
|
const data = <VeloparkData[]>await Utils.downloadJson(url)
|
||||||
|
|
||||||
|
const bboxBelgium = new BBox([[2.51357303225, 49.5294835476],[ 6.15665815596, 51.4750237087]])
|
||||||
|
const alreadyLinkedQuery = new Overpass(new RegexTag("ref:velopark", /.+/),
|
||||||
|
[],
|
||||||
|
Constants.defaultOverpassUrls[0],
|
||||||
|
new ImmutableStore(60*5),
|
||||||
|
false
|
||||||
|
)
|
||||||
|
const alreadyLinkedFeatures = await alreadyLinkedQuery.queryGeoJson(bboxBelgium)
|
||||||
|
const seenIds = new Set<string>(alreadyLinkedFeatures[0].features.map(f => f.properties["ref:velopark"]))
|
||||||
|
const features = data.map(f => VeloparkLoader.convert(f))
|
||||||
|
.filter(f => !seenIds.has(f.properties["ref:velopark"]))
|
||||||
|
|
||||||
|
const allProperties = new Set<string>()
|
||||||
|
for (const feature of features) {
|
||||||
|
Object.keys(feature.properties).forEach(k => allProperties.add(k))
|
||||||
|
}
|
||||||
|
allProperties.delete("ref:velopark")
|
||||||
|
for (const feature of features) {
|
||||||
|
allProperties.forEach(k => {
|
||||||
|
delete feature.properties[k]
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.writeFileSync("velopark_id_only_export_" + new Date().toISOString() + ".geojson", JSON.stringify({
|
||||||
|
"type": "FeatureCollection",
|
||||||
|
features,
|
||||||
|
}, null, " "))
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
new VeloParkToGeojson().run()
|
|
@ -1,4 +1,4 @@
|
||||||
import { Feature, Point } from "geojson"
|
import { Feature, Geometry, Point } from "geojson"
|
||||||
import { OH } from "../../UI/OpeningHours/OpeningHours"
|
import { OH } from "../../UI/OpeningHours/OpeningHours"
|
||||||
import EmailValidator from "../../UI/InputElement/Validators/EmailValidator"
|
import EmailValidator from "../../UI/InputElement/Validators/EmailValidator"
|
||||||
import PhoneValidator from "../../UI/InputElement/Validators/PhoneValidator"
|
import PhoneValidator from "../../UI/InputElement/Validators/PhoneValidator"
|
||||||
|
@ -18,12 +18,13 @@ export default class VeloparkLoader {
|
||||||
|
|
||||||
private static readonly coder = new CountryCoder(
|
private static readonly coder = new CountryCoder(
|
||||||
Constants.countryCoderEndpoint,
|
Constants.countryCoderEndpoint,
|
||||||
Utils.downloadJson
|
Utils.downloadJson,
|
||||||
)
|
)
|
||||||
|
|
||||||
public static convert(veloparkData: VeloparkData): Feature<Point> {
|
public static convert(veloparkData: VeloparkData): Feature {
|
||||||
|
|
||||||
const properties: {
|
const properties: {
|
||||||
|
"ref:velopark":string,
|
||||||
"operator:email"?: string,
|
"operator:email"?: string,
|
||||||
"operator:phone"?: string,
|
"operator:phone"?: string,
|
||||||
fee?: string,
|
fee?: string,
|
||||||
|
@ -31,7 +32,9 @@ export default class VeloparkLoader {
|
||||||
access?: string
|
access?: string
|
||||||
maxstay?: string
|
maxstay?: string
|
||||||
operator?: string
|
operator?: string
|
||||||
} = {}
|
} = {
|
||||||
|
"ref:velopark": veloparkData["id"] ?? veloparkData["@id"]
|
||||||
|
}
|
||||||
|
|
||||||
properties.operator = veloparkData.operatedBy?.companyName
|
properties.operator = veloparkData.operatedBy?.companyName
|
||||||
|
|
||||||
|
@ -53,15 +56,18 @@ export default class VeloparkLoader {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
let coordinates: [number, number] = undefined
|
let geometry = veloparkData.geometry
|
||||||
for (const g of veloparkData["@graph"]) {
|
for (const g of veloparkData["@graph"]) {
|
||||||
coordinates = [g.geo[0].longitude, g.geo[0].latitude]
|
if (g.geo[0]) {
|
||||||
|
geometry = { type: "Point", coordinates: [g.geo[0].longitude, g.geo[0].latitude] }
|
||||||
|
}
|
||||||
if (g.maximumParkingDuration?.endsWith("D") && g.maximumParkingDuration?.startsWith("P")) {
|
if (g.maximumParkingDuration?.endsWith("D") && g.maximumParkingDuration?.startsWith("P")) {
|
||||||
const duration = g.maximumParkingDuration.substring(1, g.maximumParkingDuration.length - 1)
|
const duration = g.maximumParkingDuration.substring(1, g.maximumParkingDuration.length - 1)
|
||||||
properties.maxstay = duration + " days"
|
properties.maxstay = duration + " days"
|
||||||
}
|
}
|
||||||
properties.access = g.publicAccess ? "yes" : "no"
|
properties.access = g.publicAccess ? "yes" : "no"
|
||||||
const prefix = "http://schema.org/"
|
const prefix = "http://schema.org/"
|
||||||
|
if (g.openingHoursSpecification) {
|
||||||
const oh = OH.simplify(g.openingHoursSpecification.map(spec => {
|
const oh = OH.simplify(g.openingHoursSpecification.map(spec => {
|
||||||
const dayOfWeek = spec.dayOfWeek.substring(prefix.length, prefix.length + 2).toLowerCase()
|
const dayOfWeek = spec.dayOfWeek.substring(prefix.length, prefix.length + 2).toLowerCase()
|
||||||
const startHour = spec.opens
|
const startHour = spec.opens
|
||||||
|
@ -70,19 +76,19 @@ export default class VeloparkLoader {
|
||||||
return OH.ToString(merged)
|
return OH.ToString(merged)
|
||||||
}).join("; "))
|
}).join("; "))
|
||||||
properties.opening_hours = oh
|
properties.opening_hours = oh
|
||||||
|
}
|
||||||
if (g.priceSpecification[0]) {
|
if (g.priceSpecification?.[0]) {
|
||||||
properties.fee = g.priceSpecification[0].freeOfCharge ? "no" : "yes"
|
properties.fee = g.priceSpecification[0].freeOfCharge ? "no" : "yes"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return { type: "Feature", properties, geometry }
|
||||||
return { type: "Feature", properties, geometry: { type: "Point", coordinates } }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
interface VeloparkData {
|
export interface VeloparkData {
|
||||||
|
geometry?: Geometry
|
||||||
"@context": any,
|
"@context": any,
|
||||||
"@id": string // "https://data.velopark.be/data/NMBS_541",
|
"@id": string // "https://data.velopark.be/data/NMBS_541",
|
||||||
"@type": "BicycleParkingStation",
|
"@type": "BicycleParkingStation",
|
||||||
|
|
Loading…
Add table
Reference in a new issue