import {existsSync, mkdirSync, readdirSync, readFileSync, writeFileSync} from "fs"; import ScriptUtils from "../../scripts/ScriptUtils"; import {Utils} from "../../Utils"; ScriptUtils.fixUtils() 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) { 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 = [] for (let day = startDay; day <= 31; day++) { if (year === currentYear && month === currentMonth && day === today.getDate()) { 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` if (existsSync(path)) { let features = JSON.parse(readFileSync(path, "UTF-8")) features = features?.features ?? features console.log(features) features.push(...features.features ) // 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, got", features.length, "features now") continue } let dayFeatures: any[] = undefined try { dayFeatures = await this.DownloadStatsForDay(year, month, day, path) } catch (e) { console.error(e) console.error("Could not download " + year + "-" + month + "-" + day + "... Trying again") dayFeatures = await this.DownloadStatsForDay(year, month, day, path) } writeFileSync(path, JSON.stringify(dayFeatures)) features.push(...dayFeatures) } writeFileSync(pathM, JSON.stringify({features})) } startDay = 1 } } public async DownloadStatsForDay(year: number, month: number, day: number, path: string): Promise { let page = 1; let allFeatures = [] 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 6e422e2afedb79ef66573982012000281f03dc91', '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(url, headers) page++; allFeatures.push(...result.features) if (result.features === undefined) { console.log("ERROR", result) return } url = result.next } console.log(`Writing ${allFeatures.length} features to `, path, Utils.Times(_ => " ", 80)) 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 { "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 } } } async function main(): Promise { if (!existsSync("graphs")) { mkdirSync("graphs") } const targetDir = "Docs/Tools/stats" 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 allPaths = readdirSync(targetDir) .filter(p => p.startsWith("stats.") && p.endsWith(".json")); let allFeatures: ChangeSetData[] = [].concat(...allPaths .map(path => JSON.parse(readFileSync("Docs/Tools/stats/" + 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.properties.metadata?.theme !== "EMPTY CS") if (process.argv.indexOf("--no-graphs") >= 0) { return } const allFiles = readdirSync("Docs/Tools/stats").filter(p => p.endsWith(".json")) writeFileSync("Docs/Tools/stats/file-overview.json", JSON.stringify(allFiles)) } main().then(_ => console.log("All done!"))