forked from MapComplete/MapComplete
406 lines
15 KiB
TypeScript
406 lines
15 KiB
TypeScript
import escapeHtml from "escape-html"
|
|
import UserDetails, { OsmConnection } from "./OsmConnection"
|
|
import { Store, UIEventSource } from "../UIEventSource"
|
|
import Locale from "../../UI/i18n/Locale"
|
|
import Constants from "../../Models/Constants"
|
|
import { Changes } from "./Changes"
|
|
import { Utils } from "../../Utils"
|
|
|
|
export interface ChangesetTag {
|
|
key: string
|
|
value: string | number
|
|
aggregate?: boolean
|
|
}
|
|
|
|
export class ChangesetHandler {
|
|
private readonly allElements: { addAlias: (id0: String, id1: string) => void }
|
|
private osmConnection: OsmConnection
|
|
private readonly changes: Changes
|
|
private readonly _dryRun: Store<boolean>
|
|
private readonly userDetails: UIEventSource<UserDetails>
|
|
private readonly backend: string
|
|
|
|
/**
|
|
* Contains previously rewritten IDs
|
|
* @private
|
|
*/
|
|
private readonly _remappings = new Map<string, string>()
|
|
|
|
constructor(
|
|
dryRun: Store<boolean>,
|
|
osmConnection: OsmConnection,
|
|
allElements: { addAlias: (id0: string, id1: string) => void } | undefined,
|
|
changes: Changes
|
|
) {
|
|
this.osmConnection = osmConnection
|
|
this.allElements = allElements
|
|
this.changes = changes
|
|
this._dryRun = dryRun
|
|
this.userDetails = osmConnection.userDetails
|
|
this.backend = osmConnection._oauth_config.url
|
|
|
|
if (dryRun) {
|
|
console.log("DRYRUN ENABLED")
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates a new list which contains every key at most once
|
|
*
|
|
* ChangesetHandler.removeDuplicateMetaTags([{key: "k", value: "v"}, {key: "k0", value: "v0"}, {key: "k", value:"v"}] // => [{key: "k", value: "v"}, {key: "k0", value: "v0"}]
|
|
*/
|
|
private static removeDuplicateMetaTags(extraMetaTags: ChangesetTag[]): ChangesetTag[] {
|
|
const r: ChangesetTag[] = []
|
|
const seen = new Set<string>()
|
|
for (const extraMetaTag of extraMetaTags) {
|
|
if (seen.has(extraMetaTag.key)) {
|
|
continue
|
|
}
|
|
r.push(extraMetaTag)
|
|
seen.add(extraMetaTag.key)
|
|
}
|
|
return r
|
|
}
|
|
|
|
/**
|
|
* Inplace rewrite of extraMetaTags
|
|
* If the metatags contain a special motivation of the format "<change-type>:node/-<number>", this method will rewrite this negative number to the actual ID
|
|
* The key is changed _in place_; true will be returned if a change has been applied
|
|
* @param extraMetaTags
|
|
* @param rewriteIds
|
|
* @public for testing purposes
|
|
*/
|
|
public static rewriteMetaTags(extraMetaTags: ChangesetTag[], rewriteIds: Map<string, string>) {
|
|
let hasChange = false
|
|
for (const tag of extraMetaTags) {
|
|
const match = tag.key.match(/^([a-zA-Z0-9_]+):(node\/-[0-9])$/)
|
|
if (match == null) {
|
|
continue
|
|
}
|
|
// This is a special motivation which has a negative ID -> we check for rewrites
|
|
const [_, reason, id] = match
|
|
if (rewriteIds.has(id)) {
|
|
tag.key = reason + ":" + rewriteIds.get(id)
|
|
hasChange = true
|
|
}
|
|
}
|
|
return hasChange
|
|
}
|
|
|
|
/**
|
|
* The full logic to upload a change to one or more elements.
|
|
*
|
|
* This method will attempt to reuse an existing, open changeset for this theme (or open one if none available).
|
|
* Then, it will upload a changes-xml within this changeset (and leave the changeset open)
|
|
* When upload is successfull, eventual id-rewriting will be handled (aka: don't worry about that)
|
|
*
|
|
* If 'dryrun' is specified, the changeset XML will be printed to console instead of being uploaded
|
|
*
|
|
*/
|
|
public async UploadChangeset(
|
|
generateChangeXML: (csid: number, remappings: Map<string, string>) => string,
|
|
extraMetaTags: ChangesetTag[],
|
|
openChangeset: UIEventSource<number>
|
|
): Promise<void> {
|
|
if (
|
|
!extraMetaTags.some((tag) => tag.key === "comment") ||
|
|
!extraMetaTags.some((tag) => tag.key === "theme")
|
|
) {
|
|
throw "The meta tags should at least contain a `comment` and a `theme`"
|
|
}
|
|
|
|
extraMetaTags = [...extraMetaTags, ...this.defaultChangesetTags()]
|
|
extraMetaTags = ChangesetHandler.removeDuplicateMetaTags(extraMetaTags)
|
|
if (this.userDetails.data.csCount == 0) {
|
|
// The user became a contributor!
|
|
this.userDetails.data.csCount = 1
|
|
this.userDetails.ping()
|
|
}
|
|
if (this._dryRun.data) {
|
|
const changesetXML = generateChangeXML(123456, this._remappings)
|
|
console.log("Metatags are", extraMetaTags)
|
|
console.log(changesetXML)
|
|
return
|
|
}
|
|
|
|
if (openChangeset.data === undefined) {
|
|
// We have to open a new changeset
|
|
try {
|
|
const csId = await this.OpenChangeset(extraMetaTags)
|
|
openChangeset.setData(csId)
|
|
const changeset = generateChangeXML(csId, this._remappings)
|
|
console.log(
|
|
"Opened a new changeset (openChangeset.data is undefined):",
|
|
changeset, extraMetaTags
|
|
)
|
|
const changes = await this.UploadChange(csId, changeset)
|
|
const hasSpecialMotivationChanges = ChangesetHandler.rewriteMetaTags(
|
|
extraMetaTags,
|
|
changes
|
|
)
|
|
if (hasSpecialMotivationChanges) {
|
|
// At this point, 'extraMetaTags' will have changed - we need to set the tags again
|
|
await this.UpdateTags(csId, extraMetaTags)
|
|
}
|
|
} catch (e) {
|
|
console.error("Could not open/upload changeset due to ", e)
|
|
openChangeset.setData(undefined)
|
|
}
|
|
} else {
|
|
// There still exists an open changeset (or at least we hope so)
|
|
// Let's check!
|
|
const csId = openChangeset.data
|
|
try {
|
|
const oldChangesetMeta = await this.GetChangesetMeta(csId)
|
|
if (!oldChangesetMeta.open) {
|
|
// Mark the CS as closed...
|
|
console.log("Could not fetch the metadata from the already open changeset")
|
|
openChangeset.setData(undefined)
|
|
// ... and try again. As the cs is closed, no recursive loop can exist
|
|
await this.UploadChangeset(generateChangeXML, extraMetaTags, openChangeset)
|
|
return
|
|
}
|
|
|
|
const rewritings = await this.UploadChange(
|
|
csId,
|
|
generateChangeXML(csId, this._remappings)
|
|
)
|
|
|
|
const rewrittenTags = this.RewriteTagsOf(
|
|
extraMetaTags,
|
|
rewritings,
|
|
oldChangesetMeta
|
|
)
|
|
await this.UpdateTags(csId, rewrittenTags)
|
|
} catch (e) {
|
|
console.warn("Could not upload, changeset is probably closed: ", e)
|
|
openChangeset.setData(undefined)
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Given an existing changeset with metadata and extraMetaTags to add, will fuse them to a new set of metatags
|
|
* Does not yet send data
|
|
* @param extraMetaTags: new changeset tags to add/fuse with this changeset
|
|
* @param rewriteIds: the mapping of ids
|
|
* @param oldChangesetMeta: the metadata-object of the already existing changeset
|
|
*
|
|
* @public for testing purposes
|
|
*/
|
|
public RewriteTagsOf(
|
|
extraMetaTags: ChangesetTag[],
|
|
rewriteIds: Map<string, string>,
|
|
oldChangesetMeta: {
|
|
open: boolean
|
|
id: number
|
|
uid: number // User ID
|
|
changes_count: number
|
|
tags: any
|
|
}
|
|
): ChangesetTag[] {
|
|
// Note: extraMetaTags is where all the tags are collected into
|
|
|
|
// same as 'extraMetaTag', but indexed
|
|
// Note that updates to 'extraTagsById.get(<key>).value = XYZ' is shared with extraMetatags
|
|
const extraTagsById = new Map<string, ChangesetTag>()
|
|
for (const extraMetaTag of extraMetaTags) {
|
|
extraTagsById.set(extraMetaTag.key, extraMetaTag)
|
|
}
|
|
|
|
const oldCsTags = oldChangesetMeta.tags
|
|
for (const key in oldCsTags) {
|
|
const newMetaTag = extraTagsById.get(key)
|
|
const existingValue = oldCsTags[key]
|
|
|
|
if (newMetaTag !== undefined && newMetaTag.value === existingValue) {
|
|
continue
|
|
}
|
|
if (newMetaTag === undefined) {
|
|
extraMetaTags.push({
|
|
key: key,
|
|
value: oldCsTags[key],
|
|
})
|
|
continue
|
|
}
|
|
|
|
if (newMetaTag.aggregate) {
|
|
let n = Number(newMetaTag.value)
|
|
if (isNaN(n)) {
|
|
n = 0
|
|
}
|
|
let o = Number(oldCsTags[key])
|
|
if (isNaN(o)) {
|
|
o = 0
|
|
}
|
|
// We _update_ the tag itself, as it'll be updated in 'extraMetaTags' straight away
|
|
newMetaTag.value = "" + (n + o)
|
|
} else {
|
|
// The old value is overwritten, thus we drop this old key
|
|
}
|
|
}
|
|
|
|
ChangesetHandler.rewriteMetaTags(extraMetaTags, rewriteIds)
|
|
return extraMetaTags
|
|
}
|
|
|
|
/**
|
|
* Updates the id in the AllElements store, returns the new ID
|
|
* @param node: the XML-element, e.g. <node old_id="-1" new_id="9650458521" new_version="1"/>
|
|
* @param type
|
|
* @private
|
|
*/
|
|
private static parseIdRewrite(node: any, type: string): [string, string] {
|
|
const oldId = parseInt(node.attributes.old_id.value)
|
|
if (node.attributes.new_id === undefined) {
|
|
return [type + "/" + oldId, undefined]
|
|
}
|
|
|
|
const newId = parseInt(node.attributes.new_id.value)
|
|
// The actual mapping
|
|
const result: [string, string] = [type + "/" + oldId, type + "/" + newId]
|
|
if (oldId === newId) {
|
|
return undefined
|
|
}
|
|
return result
|
|
}
|
|
|
|
/**
|
|
* Given a diff-result XML of the form
|
|
* <diffResult version="0.6">
|
|
* <node old_id="-1" new_id="9650458521" new_version="1"/>
|
|
* <way old_id="-2" new_id="1050127772" new_version="1"/>
|
|
* </diffResult>,
|
|
* will:
|
|
*
|
|
* - create a mapping `{'node/-1' --> "node/9650458521", 'way/-2' --> "way/9650458521"}
|
|
* - Call this.changes.registerIdRewrites
|
|
* - Call handleIdRewrites as needed
|
|
* @param response
|
|
* @private
|
|
*/
|
|
private parseUploadChangesetResponse(response: XMLDocument): Map<string, string> {
|
|
const nodes = response.getElementsByTagName("node")
|
|
const mappings: [string, string][] = []
|
|
|
|
for (const node of Array.from(nodes)) {
|
|
const mapping = ChangesetHandler.parseIdRewrite(node, "node")
|
|
if (mapping !== undefined) {
|
|
mappings.push(mapping)
|
|
}
|
|
}
|
|
|
|
const ways = response.getElementsByTagName("way")
|
|
for (const way of Array.from(ways)) {
|
|
const mapping = ChangesetHandler.parseIdRewrite(way, "way")
|
|
if (mapping !== undefined) {
|
|
mappings.push(mapping)
|
|
}
|
|
}
|
|
for (const mapping of mappings) {
|
|
const [oldId, newId] = mapping
|
|
this.allElements?.addAlias(oldId, newId)
|
|
if (newId !== undefined) {
|
|
this._remappings.set(mapping[0], mapping[1])
|
|
}
|
|
}
|
|
return new Map<string, string>(mappings)
|
|
}
|
|
|
|
// noinspection JSUnusedLocalSymbols
|
|
private async CloseChangeset(changesetId: number = undefined): Promise<void> {
|
|
if (changesetId === undefined) {
|
|
return
|
|
}
|
|
await this.osmConnection.put("changeset/" + changesetId + "/close")
|
|
console.log("Closed changeset ", changesetId)
|
|
}
|
|
|
|
private async GetChangesetMeta(csId: number): Promise<{
|
|
id: number
|
|
open: boolean
|
|
uid: number
|
|
changes_count: number
|
|
tags: any
|
|
}> {
|
|
const url = `${this.backend}/api/0.6/changeset/${csId}`
|
|
const csData = await Utils.downloadJson(url)
|
|
return csData.elements[0]
|
|
}
|
|
|
|
/**
|
|
* Puts the specified tags onto the changesets as they are.
|
|
* This method will erase previously set tags
|
|
*/
|
|
private async UpdateTags(csId: number, tags: ChangesetTag[]) {
|
|
tags = ChangesetHandler.removeDuplicateMetaTags(tags)
|
|
|
|
tags = Utils.NoNull(tags).filter(
|
|
(tag) =>
|
|
tag.key !== undefined &&
|
|
tag.value !== undefined &&
|
|
tag.key !== "" &&
|
|
tag.value !== ""
|
|
)
|
|
const metadata = tags.map((kv) => `<tag k="${kv.key}" v="${escapeHtml(kv.value)}"/>`)
|
|
const content = [`<osm><changeset>`, metadata, `</changeset></osm>`].join("")
|
|
return this.osmConnection.put("changeset/" + csId, content, { "Content-Type": "text/xml" })
|
|
}
|
|
|
|
private defaultChangesetTags(): ChangesetTag[] {
|
|
return [
|
|
["created_by", `MapComplete ${Constants.vNumber}`],
|
|
["locale", Locale.language.data],
|
|
["host", `${window.location.origin}${window.location.pathname}`],
|
|
[
|
|
"source",
|
|
this.changes.state["currentUserLocation"]?.features?.data?.length > 0
|
|
? "survey"
|
|
: undefined,
|
|
],
|
|
["imagery", this.changes.state["backgroundLayer"]?.data?.id],
|
|
].map(([key, value]) => ({
|
|
key,
|
|
value,
|
|
aggretage: false,
|
|
}))
|
|
}
|
|
|
|
/**
|
|
* Opens a changeset with the specified tags
|
|
* @param changesetTags
|
|
* @constructor
|
|
* @private
|
|
*/
|
|
private async OpenChangeset(changesetTags: ChangesetTag[]): Promise<number> {
|
|
const metadata = changesetTags
|
|
.map((cstag) => [cstag.key, cstag.value])
|
|
.filter((kv) => (kv[1] ?? "") !== "")
|
|
.map((kv) => `<tag k="${kv[0]}" v="${escapeHtml(kv[1])}"/>`)
|
|
.join("\n")
|
|
|
|
const csId = await this.osmConnection.put(
|
|
"changeset/create",
|
|
[`<osm><changeset>`, metadata, `</changeset></osm>`].join(""),
|
|
{ "Content-Type": "text/xml" }
|
|
)
|
|
return Number(csId)
|
|
}
|
|
|
|
/**
|
|
* Upload a changesetXML
|
|
*/
|
|
private async UploadChange(
|
|
changesetId: number,
|
|
changesetXML: string
|
|
): Promise<Map<string, string>> {
|
|
const response = await this.osmConnection.post(
|
|
"changeset/" + changesetId + "/upload",
|
|
changesetXML,
|
|
{ "Content-Type": "text/xml" }
|
|
)
|
|
const changes = this.parseUploadChangesetResponse(response)
|
|
console.log("Uploaded changeset ", changesetId)
|
|
return changes
|
|
}
|
|
}
|