MapComplete/Logic/Osm/ChangesetHandler.ts

408 lines
15 KiB
TypeScript
Raw Normal View History

2022-09-08 21:40:48 +02:00
import escapeHtml from "escape-html"
import UserDetails, { OsmConnection } from "./OsmConnection"
2023-04-13 20:58:49 +02:00
import { Store, UIEventSource } from "../UIEventSource"
2022-09-08 21:40:48 +02:00
import Locale from "../../UI/i18n/Locale"
import Constants from "../../Models/Constants"
import { Changes } from "./Changes"
import { Utils } from "../../Utils"
export interface ChangesetTag {
2022-09-08 21:40:48 +02:00
key: string
value: string | number
aggregate?: boolean
}
export class ChangesetHandler {
private readonly allElements: { addAlias: (id0: String, id1: string) => void }
2022-09-08 21:40:48 +02:00
private osmConnection: OsmConnection
private readonly changes: Changes
2023-04-13 20:58:49 +02:00
private readonly _dryRun: Store<boolean>
2022-09-08 21:40:48 +02:00
private readonly userDetails: UIEventSource<UserDetails>
private readonly backend: string
/**
* Contains previously rewritten IDs
* @private
*/
private readonly _remappings = new Map<string, string>()
2022-09-08 21:40:48 +02:00
constructor(
2023-04-13 20:58:49 +02:00
dryRun: Store<boolean>,
2022-09-08 21:40:48 +02:00
osmConnection: OsmConnection,
2023-04-13 20:58:49 +02:00
allElements: { addAlias: (id0: string, id1: string) => void } | undefined,
changes: Changes
2022-09-08 21:40:48 +02:00
) {
this.osmConnection = osmConnection
this.allElements = allElements
this.changes = changes
this._dryRun = dryRun
this.userDetails = osmConnection.userDetails
this.backend = osmConnection._oauth_config.url
if (dryRun) {
2022-09-08 21:40:48 +02:00
console.log("DRYRUN ENABLED")
}
}
2022-03-15 13:44:34 +01:00
/**
* Creates a new list which contains every key at most once
2022-09-08 21:40:48 +02:00
*
2022-03-16 14:06:10 +01:00
* ChangesetHandler.removeDuplicateMetaTags([{key: "k", value: "v"}, {key: "k0", value: "v0"}, {key: "k", value:"v"}] // => [{key: "k", value: "v"}, {key: "k0", value: "v0"}]
2022-03-15 13:44:34 +01:00
*/
private static removeDuplicateMetaTags(extraMetaTags: ChangesetTag[]): ChangesetTag[] {
2022-09-08 21:40:48 +02:00
const r: ChangesetTag[] = []
2022-03-15 13:44:34 +01:00
const seen = new Set<string>()
for (const extraMetaTag of extraMetaTags) {
2022-09-08 21:40:48 +02:00
if (seen.has(extraMetaTag.key)) {
2022-03-15 13:44:34 +01:00
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
2023-04-07 03:54:11 +02:00
* @public for testing purposes
*/
2023-04-07 03:54:11 +02:00
public static rewriteMetaTags(extraMetaTags: ChangesetTag[], rewriteIds: Map<string, string>) {
2022-09-08 21:40:48 +02:00
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
*
*/
2021-09-26 23:35:26 +02:00
public async UploadChangeset(
generateChangeXML: (csid: number, remappings: Map<string, string>) => string,
2021-12-17 19:28:05 +01:00
extraMetaTags: ChangesetTag[],
2022-09-08 21:40:48 +02:00
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`"
}
2022-09-08 21:40:48 +02:00
extraMetaTags = [...extraMetaTags, ...this.defaultChangesetTags()]
2022-03-15 13:44:34 +01:00
extraMetaTags = ChangesetHandler.removeDuplicateMetaTags(extraMetaTags)
if (this.userDetails.data.csCount == 0) {
2020-09-15 02:29:31 +02:00
// The user became a contributor!
2022-09-08 21:40:48 +02:00
this.userDetails.data.csCount = 1
this.userDetails.ping()
2020-09-15 02:29:31 +02:00
}
if (this._dryRun.data) {
2022-09-08 21:40:48 +02:00
const changesetXML = generateChangeXML(123456, this._remappings)
console.log("Metatags are", extraMetaTags)
2022-09-08 21:40:48 +02:00
console.log(changesetXML)
return
}
2021-12-17 19:28:05 +01:00
if (openChangeset.data === undefined) {
// We have to open a new changeset
2021-09-26 23:35:26 +02:00
try {
const csId = await this.OpenChangeset(extraMetaTags)
2022-09-08 21:40:48 +02:00
openChangeset.setData(csId)
const changeset = generateChangeXML(csId, this._remappings)
2023-06-01 02:52:21 +02:00
console.log(
2022-09-08 21:40:48 +02:00
"Opened a new changeset (openChangeset.data is undefined):",
changeset,
extraMetaTags
2022-09-08 21:40:48 +02:00
)
const changes = await this.UploadChange(csId, changeset)
2022-09-08 21:40:48 +02:00
const hasSpecialMotivationChanges = ChangesetHandler.rewriteMetaTags(
extraMetaTags,
changes
)
if (hasSpecialMotivationChanges) {
// At this point, 'extraMetaTags' will have changed - we need to set the tags again
2022-12-16 01:02:46 +01:00
await this.UpdateTags(csId, extraMetaTags)
}
2021-09-26 23:35:26 +02:00
} catch (e) {
console.error("Could not open/upload changeset due to ", e)
2021-12-17 19:28:05 +01:00
openChangeset.setData(undefined)
2021-09-26 23:35:26 +02:00
}
} else {
// There still exists an open changeset (or at least we hope so)
// Let's check!
2022-09-08 21:40:48 +02:00
const csId = openChangeset.data
2021-09-26 23:35:26 +02:00
try {
const oldChangesetMeta = await this.GetChangesetMeta(csId)
if (!oldChangesetMeta.open) {
// Mark the CS as closed...
2021-12-17 19:28:05 +01:00
console.log("Could not fetch the metadata from the already open changeset")
2022-09-08 21:40:48 +02:00
openChangeset.setData(undefined)
// ... and try again. As the cs is closed, no recursive loop can exist
2021-12-17 19:28:05 +01:00
await this.UploadChangeset(generateChangeXML, extraMetaTags, openChangeset)
2022-09-08 21:40:48 +02:00
return
}
const rewritings = await this.UploadChange(
2021-09-26 23:35:26 +02:00
csId,
2022-09-08 21:40:48 +02:00
generateChangeXML(csId, this._remappings)
)
const rewrittenTags = this.RewriteTagsOf(
extraMetaTags,
rewritings,
oldChangesetMeta
)
await this.UpdateTags(csId, rewrittenTags)
2021-09-26 23:35:26 +02:00
} catch (e) {
2022-09-08 21:40:48 +02:00
console.warn("Could not upload, changeset is probably closed: ", e)
openChangeset.setData(undefined)
2021-09-26 23:35:26 +02:00
}
}
}
/**
* 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
2023-04-07 03:54:11 +02:00
*
* @public for testing purposes
*/
2023-04-07 03:54:11 +02:00
public RewriteTagsOf(
2022-09-08 21:40:48 +02:00
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,
2022-09-08 21:40:48 +02:00
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] {
2022-09-08 21:40:48 +02:00
const oldId = parseInt(node.attributes.old_id.value)
2021-11-07 16:34:51 +01:00
if (node.attributes.new_id === undefined) {
2022-09-08 21:40:48 +02:00
return [type + "/" + oldId, undefined]
2021-11-07 16:34:51 +01:00
}
2022-09-08 21:40:48 +02:00
const newId = parseInt(node.attributes.new_id.value)
// The actual mapping
2021-11-07 16:34:51 +01:00
const result: [string, string] = [type + "/" + oldId, type + "/" + newId]
2022-09-08 21:40:48 +02:00
if (oldId === newId) {
return undefined
2021-11-07 16:34:51 +01:00
}
2022-09-08 21:40:48 +02:00
return result
2021-11-07 16:34:51 +01:00
}
/**
2022-09-08 21:40:48 +02:00
* 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:
2022-09-08 21:40:48 +02:00
*
* - 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> {
2022-09-08 21:40:48 +02:00
const nodes = response.getElementsByTagName("node")
const mappings: [string, string][] = []
2022-03-17 21:51:53 +01:00
for (const node of Array.from(nodes)) {
const mapping = ChangesetHandler.parseIdRewrite(node, "node")
2021-11-07 16:34:51 +01:00
if (mapping !== undefined) {
mappings.push(mapping)
2021-11-07 16:34:51 +01:00
}
}
2022-09-08 21:40:48 +02:00
const ways = response.getElementsByTagName("way")
2022-03-17 21:51:53 +01:00
for (const way of Array.from(ways)) {
const mapping = ChangesetHandler.parseIdRewrite(way, "way")
2021-11-07 16:34:51 +01:00
if (mapping !== undefined) {
mappings.push(mapping)
2021-11-07 16:34:51 +01:00
}
}
for (const mapping of mappings) {
const [oldId, newId] = mapping
2023-04-13 20:58:49 +02:00
this.allElements?.addAlias(oldId, newId)
2022-09-08 21:40:48 +02:00
if (newId !== undefined) {
this._remappings.set(mapping[0], mapping[1])
}
}
return new Map<string, string>(mappings)
2021-11-07 16:34:51 +01:00
}
2023-04-07 03:54:11 +02:00
// 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<{
2022-09-08 21:40:48 +02:00
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
*/
2022-09-08 21:40:48 +02:00
private async UpdateTags(csId: number, tags: ChangesetTag[]) {
2022-03-15 13:44:34 +01:00
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" })
}
2022-09-08 21:40:48 +02:00
private defaultChangesetTags(): ChangesetTag[] {
return [
["created_by", `MapComplete ${Constants.vNumber}`],
["locale", Locale.language.data],
["host", `${window.location.origin}${window.location.pathname}`],
2022-09-08 21:40:48 +02:00
[
"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
}
}