import escapeHtml from "escape-html"; // @ts-ignore import {OsmConnection, UserDetails} from "./OsmConnection"; import {UIEventSource} from "../UIEventSource"; import {ElementStorage} from "../ElementStorage"; import State from "../../State"; import Locale from "../../UI/i18n/Locale"; import LayoutConfig from "../../Customizations/JSON/LayoutConfig"; import Constants from "../../Models/Constants"; export class ChangesetHandler { public readonly currentChangeset: UIEventSource; private readonly _dryRun: boolean; private readonly userDetails: UIEventSource; private readonly auth: any; constructor(layoutName: string, dryRun: boolean, osmConnection: OsmConnection, auth) { this._dryRun = dryRun; this.userDetails = osmConnection.userDetails; this.auth = auth; this.currentChangeset = osmConnection.GetPreference("current-open-changeset-" + layoutName); if (dryRun) { console.log("DRYRUN ENABLED"); } } private static parseUploadChangesetResponse(response: XMLDocument, allElements: ElementStorage) { const nodes = response.getElementsByTagName("node"); // @ts-ignore for (const node of nodes) { const oldId = parseInt(node.attributes.old_id.value); const newId = parseInt(node.attributes.new_id.value); if (oldId !== undefined && newId !== undefined && !isNaN(oldId) && !isNaN(newId)) { if (oldId == newId) { continue; } console.log("Rewriting id: ", oldId, "-->", newId); const element = allElements.getEventSourceById("node/" + oldId); element.data.id = "node/" + newId; allElements.addElementById("node/" + newId, element); element.ping(); } } } public UploadChangeset( layout: LayoutConfig, allElements: ElementStorage, generateChangeXML: (csid: string) => string, continuation: () => void) { if (this.userDetails.data.csCount == 0) { // The user became a contributor! this.userDetails.data.csCount = 1; this.userDetails.ping(); } if (this._dryRun) { const changesetXML = generateChangeXML("123456"); console.log(changesetXML); continuation(); return; } const self = this; if (this.currentChangeset.data === undefined || this.currentChangeset.data === "") { // We have to open a new changeset this.OpenChangeset(layout, (csId) => { this.currentChangeset.setData(csId); const changeset = generateChangeXML(csId); console.log(changeset); self.AddChange(csId, changeset, allElements, () => { }, (e) => { console.error("UPLOADING FAILED!", e) } ) }) } else { // There still exists an open changeset (or at least we hope so) const csId = this.currentChangeset.data; self.AddChange( csId, generateChangeXML(csId), allElements, () => { }, (e) => { console.warn("Could not upload, changeset is probably closed: ", e); // Mark the CS as closed... this.currentChangeset.setData(""); // ... and try again. As the cs is closed, no recursive loop can exist self.UploadChangeset(layout, allElements, generateChangeXML, continuation); } ) } } public CloseChangeset(changesetId: string = undefined, continuation: (() => void) = () => { }) { if (changesetId === undefined) { changesetId = this.currentChangeset.data; } if (changesetId === undefined) { return; } console.log("closing changeset", changesetId); this.currentChangeset.setData(""); this.auth.xhr({ method: 'PUT', path: '/api/0.6/changeset/' + changesetId + '/close', }, function (err, response) { if (response == null) { console.log("err", err); } console.log("Closed changeset ", changesetId) if (continuation !== undefined) { continuation(); } }); } private OpenChangeset( layout: LayoutConfig, continuation: (changesetId: string) => void) { const commentExtra = layout.changesetmessage !== undefined ? " - " + layout.changesetmessage : ""; let path = window.location.pathname; path = path.substr(1, path.lastIndexOf("/")); const metadata = [ ["created_by", `MapComplete ${Constants.vNumber}`], ["comment", `Adding data with #MapComplete for theme #${layout.id}${commentExtra}`], ["theme", layout.id], ["language", Locale.language.data], ["host", window.location.host], ["path", path], ["source", State.state.currentGPSLocation.data !== undefined ? "survey" : undefined], ["imagery", State.state.backgroundLayer.data.id], ["theme-creator", layout.maintainer] ] .filter(kv => (kv[1] ?? "") !== "") .map(kv => ``) .join("\n") this.auth.xhr({ method: 'PUT', path: '/api/0.6/changeset/create', options: {header: {'Content-Type': 'text/xml'}}, content: [``, metadata, ``].join("") }, function (err, response) { if (response === undefined) { console.log("err", err); alert("Could not upload change (opening failed). Please file a bug report") return; } else { continuation(response); } }); } private AddChange(changesetId: string, changesetXML: string, allElements: ElementStorage, continuation: ((changesetId: string, idMapping: any) => void), onFail: ((changesetId: string) => void) = undefined) { this.auth.xhr({ method: 'POST', options: {header: {'Content-Type': 'text/xml'}}, path: '/api/0.6/changeset/' + changesetId + '/upload', content: changesetXML }, function (err, response) { if (response == null) { console.log("err", err); if (onFail) { onFail(changesetId); } return; } const mapping = ChangesetHandler.parseUploadChangesetResponse(response, allElements); console.log("Uploaded changeset ", changesetId); continuation(changesetId, mapping); }); } }