forked from MapComplete/MapComplete
319 lines
11 KiB
TypeScript
319 lines
11 KiB
TypeScript
import { existsSync, readdirSync, readFileSync, unlinkSync, writeFileSync } from "fs"
|
|
import ScriptUtils from "./ScriptUtils"
|
|
import { Utils } from "../src/Utils"
|
|
import Script from "./Script"
|
|
import { GeoOperations } from "../src/Logic/GeoOperations"
|
|
import { Feature, Polygon } from "geojson"
|
|
import { Tiles } from "../src/Models/TileRange"
|
|
import { BBox } from "../src/Logic/BBox"
|
|
|
|
class StatsDownloader {
|
|
private readonly urlTemplate =
|
|
"https://osmcha.org/api/v1/changesets/?date__gte={start_date}&date__lte={end_date}&page={page}&comment=%23mapcomplete&page_size=100"
|
|
|
|
private readonly _targetDirectory: string
|
|
|
|
constructor(targetDirectory = ".") {
|
|
this._targetDirectory = targetDirectory
|
|
}
|
|
|
|
public async DownloadStats(startYear = 2020, startMonth = 5, startDay = 1): Promise<void> {
|
|
const today = new Date()
|
|
const currentYear = today.getFullYear()
|
|
const currentMonth = today.getMonth() + 1
|
|
for (let year = startYear; year <= currentYear; year++) {
|
|
for (let month = 1; month <= 12; month++) {
|
|
if (year === startYear && month < startMonth) {
|
|
continue
|
|
}
|
|
|
|
if (year === currentYear && month > currentMonth) {
|
|
break
|
|
}
|
|
|
|
const pathM = `${this._targetDirectory}/stats.${year}-${month}.json`
|
|
if (existsSync(pathM)) {
|
|
continue
|
|
}
|
|
|
|
const features = []
|
|
let monthIsFinished = true
|
|
const writtenFiles = []
|
|
for (let day = startDay; day <= 31; day++) {
|
|
if (year === currentYear && month === currentMonth && day === today.getDate()) {
|
|
monthIsFinished = false
|
|
break
|
|
}
|
|
{
|
|
const date = new Date(year, month - 1, day)
|
|
if (date.getMonth() != month - 1) {
|
|
// We did roll over
|
|
continue
|
|
}
|
|
}
|
|
const path = `${this._targetDirectory}/stats.${year}-${month}-${
|
|
(day < 10 ? "0" : "") + day
|
|
}.day.json`
|
|
writtenFiles.push(path)
|
|
if (existsSync(path)) {
|
|
let loadedFeatures = JSON.parse(readFileSync(path, { encoding: "utf-8" }))
|
|
loadedFeatures = loadedFeatures?.features ?? loadedFeatures
|
|
features.push(...loadedFeatures) // day-stats are generally a list already, but in some ad-hoc cases might be a geojson-collection too
|
|
console.log(
|
|
"Loaded ",
|
|
path,
|
|
"from disk, which has",
|
|
features.length,
|
|
"features now"
|
|
)
|
|
continue
|
|
}
|
|
let dayFeatures: any[] = undefined
|
|
try {
|
|
dayFeatures = await this.DownloadStatsForDay(year, month, day)
|
|
} catch (e) {
|
|
console.error(e)
|
|
console.error(
|
|
"Could not download " +
|
|
year +
|
|
"-" +
|
|
month +
|
|
"-" +
|
|
day +
|
|
"... Trying again"
|
|
)
|
|
dayFeatures = await this.DownloadStatsForDay(year, month, day)
|
|
}
|
|
writeFileSync(path, JSON.stringify(dayFeatures))
|
|
features.push(...dayFeatures)
|
|
}
|
|
if (monthIsFinished) {
|
|
writeFileSync(pathM, JSON.stringify({ features }))
|
|
for (const writtenFile of writtenFiles) {
|
|
unlinkSync(writtenFile)
|
|
}
|
|
}
|
|
}
|
|
startDay = 1
|
|
}
|
|
}
|
|
|
|
public async DownloadStatsForDay(
|
|
year: number,
|
|
month: number,
|
|
day: number
|
|
): Promise<ChangeSetData[]> {
|
|
let page = 1
|
|
let allFeatures: ChangeSetData[] = []
|
|
let endDay = new Date(year, month - 1 /* Zero-indexed: 0 = january*/, day + 1)
|
|
let endDate = `${endDay.getFullYear()}-${Utils.TwoDigits(
|
|
endDay.getMonth() + 1
|
|
)}-${Utils.TwoDigits(endDay.getDate())}`
|
|
let url = this.urlTemplate
|
|
.replace(
|
|
"{start_date}",
|
|
year + "-" + Utils.TwoDigits(month) + "-" + Utils.TwoDigits(day)
|
|
)
|
|
.replace("{end_date}", endDate)
|
|
.replace("{page}", "" + page)
|
|
|
|
let headers = {
|
|
"User-Agent":
|
|
"Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:86.0) Gecko/20100101 Firefox/86.0",
|
|
"Accept-Language": "en-US,en;q=0.5",
|
|
Referer:
|
|
"https://osmcha.org/?filters=%7B%22date__gte%22%3A%5B%7B%22label%22%3A%222020-07-05%22%2C%22value%22%3A%222020-07-05%22%7D%5D%2C%22editor%22%3A%5B%7B%22label%22%3A%22mapcomplete%22%2C%22value%22%3A%22mapcomplete%22%7D%5D%7D",
|
|
"Content-Type": "application/json",
|
|
Authorization: "Token 9cc11ad2868778272eadbb1a423ebb507184bc04",
|
|
DNT: "1",
|
|
Connection: "keep-alive",
|
|
TE: "Trailers",
|
|
Pragma: "no-cache",
|
|
"Cache-Control": "no-cache",
|
|
}
|
|
|
|
while (url) {
|
|
ScriptUtils.erasableLog(
|
|
`Downloading stats for ${year}-${month}-${day}, page ${page} ${url}`
|
|
)
|
|
const result = await Utils.downloadJson<{ features: []; next: string }>(url, headers)
|
|
page++
|
|
allFeatures.push(...result.features)
|
|
if (result.features === undefined) {
|
|
console.log("ERROR", result)
|
|
return
|
|
}
|
|
url = result.next
|
|
}
|
|
allFeatures = Utils.NoNull(allFeatures)
|
|
allFeatures.forEach((f) => {
|
|
f.properties = { ...f.properties, ...f.properties.metadata }
|
|
delete f.properties.metadata
|
|
f.properties["id"] = f.id
|
|
})
|
|
return allFeatures
|
|
}
|
|
}
|
|
|
|
interface ChangeSetData extends Feature<Polygon> {
|
|
id: number
|
|
type: "Feature"
|
|
geometry: {
|
|
type: "Polygon"
|
|
coordinates: [number, number][][]
|
|
}
|
|
properties: {
|
|
check_user: null
|
|
reasons: []
|
|
tags: []
|
|
features: []
|
|
user: string
|
|
uid: string
|
|
editor: string
|
|
comment: string
|
|
comments_count: number
|
|
source: string
|
|
imagery_used: string
|
|
date: string
|
|
reviewed_features: []
|
|
create: number
|
|
modify: number
|
|
delete: number
|
|
area: number
|
|
is_suspect: boolean
|
|
harmful: any
|
|
checked: boolean
|
|
check_date: any
|
|
metadata: {
|
|
host: string
|
|
theme: string
|
|
imagery: string
|
|
language: string
|
|
}
|
|
}
|
|
}
|
|
|
|
class GenerateSeries extends Script {
|
|
constructor() {
|
|
super("Downloads metadata about changesets made by MapComplete from OsmCha")
|
|
}
|
|
|
|
async main(args: string[]): Promise<void> {
|
|
const targetDir = args[0] ?? "../../git/MapComplete-data"
|
|
|
|
await this.downloadStatistics(targetDir + "/changeset-metadata")
|
|
this.generateCenterPoints(
|
|
targetDir + "/changeset-metadata",
|
|
targetDir + "/mapcomplete-changes/",
|
|
{
|
|
zoomlevel: 8,
|
|
}
|
|
)
|
|
}
|
|
|
|
private async downloadStatistics(targetDir: string) {
|
|
let year = 2020
|
|
let month = 5
|
|
let day = 1
|
|
if (!isNaN(Number(process.argv[2]))) {
|
|
year = Number(process.argv[2])
|
|
}
|
|
if (!isNaN(Number(process.argv[3]))) {
|
|
month = Number(process.argv[3])
|
|
}
|
|
|
|
if (!isNaN(Number(process.argv[4]))) {
|
|
day = Number(process.argv[4])
|
|
}
|
|
|
|
do {
|
|
try {
|
|
await new StatsDownloader(targetDir).DownloadStats(year, month, day)
|
|
break
|
|
} catch (e) {
|
|
console.log(e)
|
|
}
|
|
} while (true)
|
|
|
|
const allFiles = readdirSync(targetDir).filter((p) => p.endsWith(".json"))
|
|
writeFileSync(targetDir + "/file-overview.json", JSON.stringify(allFiles))
|
|
}
|
|
|
|
private generateCenterPoints(
|
|
sourceDir: string,
|
|
targetDir: string,
|
|
options: {
|
|
zoomlevel: number
|
|
}
|
|
) {
|
|
const allPaths = readdirSync(sourceDir).filter(
|
|
(p) => p.startsWith("stats.") && p.endsWith(".json")
|
|
)
|
|
let allFeatures: ChangeSetData[] = allPaths.flatMap(
|
|
(path) => JSON.parse(readFileSync(sourceDir + "/" + path, "utf-8")).features
|
|
)
|
|
allFeatures = allFeatures.filter(
|
|
(f) =>
|
|
f?.properties !== undefined &&
|
|
(f.properties.editor === null ||
|
|
f.properties.editor.toLowerCase().startsWith("mapcomplete"))
|
|
)
|
|
|
|
allFeatures = allFeatures.filter(
|
|
(f) => f.geometry !== null && f.properties.metadata?.theme !== "EMPTY CS"
|
|
)
|
|
allFeatures = allFeatures.filter(
|
|
(f) =>
|
|
f?.properties !== undefined &&
|
|
(f.properties.editor === null ||
|
|
f.properties.editor.toLowerCase().startsWith("mapcomplete"))
|
|
)
|
|
|
|
allFeatures = allFeatures.filter(
|
|
(f) => f.properties.metadata?.theme !== "EMPTY CS" && f.geometry.coordinates.length > 0
|
|
)
|
|
const centerpointsAll = allFeatures.map((f) => {
|
|
const centerpoint = GeoOperations.centerpoint(f)
|
|
const c = centerpoint.geometry.coordinates
|
|
// OsmCha doesn't adhere to the Geojson standard and uses `lat` `lon` as coordinates instead of `lon`, `lat`
|
|
centerpoint.geometry.coordinates = [c[1], c[0]]
|
|
return centerpoint
|
|
})
|
|
const centerpoints = centerpointsAll.filter((p) => {
|
|
const bbox = BBox.get(p)
|
|
if (bbox.minLat === -90 && bbox.maxLat === -90) {
|
|
// Due to some bug somewhere, those invalid bboxes might appear if the latitude is < 90
|
|
// This crashes the 'spreadIntoBBoxes
|
|
// As workaround, we simply ignore them for now
|
|
return false
|
|
}
|
|
return true
|
|
})
|
|
console.log("Found", centerpoints.length, " changesets in total")
|
|
|
|
const perBbox = GeoOperations.spreadIntoBboxes(centerpoints, options.zoomlevel)
|
|
|
|
for (const [tileNumber, features] of perBbox) {
|
|
const [z, x, y] = Tiles.tile_from_index(tileNumber)
|
|
const path = `${targetDir}/tile_${z}_${x}_${y}.geojson`
|
|
features.forEach((f) => {
|
|
delete f.bbox
|
|
})
|
|
writeFileSync(
|
|
path,
|
|
JSON.stringify(
|
|
{
|
|
type: "FeatureCollection",
|
|
features: features,
|
|
},
|
|
null,
|
|
" "
|
|
)
|
|
)
|
|
|
|
ScriptUtils.erasableLog("Written ", path, "which has ", features.length, "features")
|
|
}
|
|
}
|
|
}
|
|
|
|
new GenerateSeries().run()
|