| 
									
										
										
										
											2020-11-18 12:50:28 +01:00
										 |  |  | import escapeHtml from "escape-html"; | 
					
						
							| 
									
										
										
										
											2021-12-13 02:05:34 +01:00
										 |  |  | import UserDetails, {OsmConnection} from "./OsmConnection"; | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  | import {UIEventSource} from "../UIEventSource"; | 
					
						
							| 
									
										
										
										
											2020-08-30 01:13:18 +02:00
										 |  |  | import {ElementStorage} from "../ElementStorage"; | 
					
						
							| 
									
										
										
										
											2020-09-21 23:57:50 +02:00
										 |  |  | import Locale from "../../UI/i18n/Locale"; | 
					
						
							| 
									
										
										
										
											2021-01-02 19:09:49 +01:00
										 |  |  | import Constants from "../../Models/Constants"; | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  | import {Changes} from "./Changes"; | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | import {Utils} from "../../Utils"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export interface ChangesetTag { | 
					
						
							|  |  |  |     key: string, | 
					
						
							|  |  |  |     value: string | number, | 
					
						
							|  |  |  |     aggregate?: boolean | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export class ChangesetHandler { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |     private readonly allElements: ElementStorage; | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |     private osmConnection: OsmConnection; | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |     private readonly changes: Changes; | 
					
						
							| 
									
										
										
										
											2022-01-21 01:57:16 +01:00
										 |  |  |     private readonly _dryRun: UIEventSource<boolean>; | 
					
						
							| 
									
										
										
										
											2020-09-15 02:29:31 +02:00
										 |  |  |     private readonly userDetails: UIEventSource<UserDetails>; | 
					
						
							|  |  |  |     private readonly auth: any; | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |     private readonly backend: string; | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Contains previously rewritten IDs | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private readonly _remappings = new Map<string, string>() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-15 23:44:16 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Use 'osmConnection.CreateChangesetHandler' instead | 
					
						
							|  |  |  |      * @param dryRun | 
					
						
							|  |  |  |      * @param osmConnection | 
					
						
							|  |  |  |      * @param allElements | 
					
						
							|  |  |  |      * @param changes | 
					
						
							|  |  |  |      * @param auth | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-02-14 02:50:21 +01:00
										 |  |  |     constructor(dryRun: UIEventSource<boolean>, | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |                 osmConnection: OsmConnection, | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |                 allElements: ElementStorage, | 
					
						
							|  |  |  |                 changes: Changes, | 
					
						
							|  |  |  |                 auth) { | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |         this.osmConnection = osmConnection; | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |         this.allElements = allElements; | 
					
						
							|  |  |  |         this.changes = changes; | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |         this._dryRun = dryRun; | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |         this.userDetails = osmConnection.userDetails; | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |         this.backend = osmConnection._oauth_config.url | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |         this.auth = auth; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (dryRun) { | 
					
						
							|  |  |  |             console.log("DRYRUN ENABLED"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-12-17 19:28:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2022-03-15 13:44:34 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Creates a new list which contains every key at most once | 
					
						
							| 
									
										
										
										
											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
										 |  |  |      */ | 
					
						
							|  |  |  |     public 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 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |      * Inplace rewrite of extraMetaTags | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |      * 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 | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |     static rewriteMetaTags(extraMetaTags: ChangesetTag[], rewriteIds: Map<string, string>) { | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01: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 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-30 18:48:23 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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( | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |         generateChangeXML: (csid: number, remappings: Map<string, string>) => string, | 
					
						
							| 
									
										
										
										
											2021-12-17 19:28:05 +01:00
										 |  |  |         extraMetaTags: ChangesetTag[], | 
					
						
							|  |  |  |         openChangeset: UIEventSource<number>): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         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-03-14 03:33:03 +01:00
										 |  |  |          | 
					
						
							|  |  |  |         extraMetaTags = [...extraMetaTags, ...this.defaultChangesetTags()] | 
					
						
							| 
									
										
										
										
											2022-03-15 13:44:34 +01:00
										 |  |  |         extraMetaTags = ChangesetHandler.removeDuplicateMetaTags(extraMetaTags) | 
					
						
							| 
									
										
										
										
											2021-06-19 18:28:30 +02:00
										 |  |  |         if (this.userDetails.data.csCount == 0) { | 
					
						
							| 
									
										
										
										
											2020-09-15 02:29:31 +02:00
										 |  |  |             // The user became a contributor!
 | 
					
						
							|  |  |  |             this.userDetails.data.csCount = 1; | 
					
						
							|  |  |  |             this.userDetails.ping(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-21 01:57:16 +01:00
										 |  |  |         if (this._dryRun.data) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |             const changesetXML = generateChangeXML(123456, this._remappings); | 
					
						
							| 
									
										
										
										
											2021-11-09 01:49:07 +01:00
										 |  |  |             console.log("Metatags are", extraMetaTags) | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |             console.log(changesetXML); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-17 19:28:05 +01:00
										 |  |  |         if (openChangeset.data === undefined) { | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |             // We have to open a new changeset
 | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |             try { | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |                 const csId = await this.OpenChangeset(extraMetaTags) | 
					
						
							| 
									
										
										
										
											2021-12-17 19:28:05 +01:00
										 |  |  |                 openChangeset.setData(csId); | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |                 const changeset = generateChangeXML(csId, this._remappings); | 
					
						
							| 
									
										
										
										
											2021-12-17 19:28:05 +01:00
										 |  |  |                 console.trace("Opened a new changeset (openChangeset.data is undefined):", changeset); | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |                 const changes = await this.UploadChange(csId, changeset) | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |                 const hasSpecialMotivationChanges = ChangesetHandler.rewriteMetaTags(extraMetaTags, changes) | 
					
						
							|  |  |  |                 if(hasSpecialMotivationChanges){ | 
					
						
							|  |  |  |                     // At this point, 'extraMetaTags' will have changed - we need to set the tags again
 | 
					
						
							|  |  |  |                     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
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             // There still exists an open changeset (or at least we hope so)
 | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |             // Let's check!
 | 
					
						
							| 
									
										
										
										
											2021-12-17 19:28:05 +01:00
										 |  |  |             const csId = openChangeset.data; | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |             try { | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |                 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") | 
					
						
							|  |  |  |                     openChangeset.setData(undefined); | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |                     // ... 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) | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |                     return; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |                 const rewritings = await this.UploadChange( | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |                     csId, | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |                     generateChangeXML(csId, this._remappings)) | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |                 const rewrittenTags = this.RewriteTagsOf(extraMetaTags, rewritings, oldChangesetMeta) | 
					
						
							|  |  |  |                 await this.UpdateTags(csId, rewrittenTags) | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |             } catch (e) { | 
					
						
							|  |  |  |                 console.warn("Could not upload, changeset is probably closed: ", e); | 
					
						
							| 
									
										
										
										
											2021-12-17 19:28:05 +01:00
										 |  |  |                 openChangeset.setData(undefined); | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01: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 | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |      * @param extraMetaTags: new changeset tags to add/fuse with this changeset | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |      * @param rewriteIds: the mapping of ids | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |      * @param oldChangesetMeta: the metadata-object of the already existing changeset | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |     public RewriteTagsOf(extraMetaTags: ChangesetTag[], | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |                                 rewriteIds: Map<string, string>, | 
					
						
							|  |  |  |                                 oldChangesetMeta: { | 
					
						
							|  |  |  |                                     open: boolean, | 
					
						
							|  |  |  |                                     id: number | 
					
						
							|  |  |  |                                     uid: number, // User ID
 | 
					
						
							|  |  |  |                                     changes_count: number, | 
					
						
							|  |  |  |                                     tags: any | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |                                 }) : ChangesetTag[] { | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // 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) | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |         return extraMetaTags | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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] { | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         const oldId = parseInt(node.attributes.old_id.value); | 
					
						
							|  |  |  |         if (node.attributes.new_id === undefined) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |             return [type+"/"+oldId, undefined]; | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const newId = parseInt(node.attributes.new_id.value); | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |         // The actual mapping
 | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         const result: [string, string] = [type + "/" + oldId, type + "/" + newId] | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |         if(oldId === newId){ | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +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: | 
					
						
							|  |  |  |      *  | 
					
						
							|  |  |  |      * - create a mapping `{'node/-1' --> "node/9650458521", 'way/-2' --> "way/9650458521"}
 | 
					
						
							|  |  |  |      * - Call this.changes.registerIdRewrites | 
					
						
							|  |  |  |      * - Call handleIdRewrites as needed | 
					
						
							|  |  |  |      * @param response | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |     private parseUploadChangesetResponse(response: XMLDocument): Map<string, string> { | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         const nodes = response.getElementsByTagName("node"); | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |         const mappings : [string, string][]= [] | 
					
						
							| 
									
										
										
										
											2022-03-17 21:51:53 +01:00
										 |  |  |          | 
					
						
							|  |  |  |         for (const node of Array.from(nodes)) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |             const mapping = ChangesetHandler.parseIdRewrite(node, "node") | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |             if (mapping !== undefined) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |                 mappings.push(mapping) | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const ways = response.getElementsByTagName("way"); | 
					
						
							| 
									
										
										
										
											2022-03-17 21:51:53 +01:00
										 |  |  |         for (const way of Array.from(ways)) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |             const mapping = ChangesetHandler.parseIdRewrite(way, "way") | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |             if (mapping !== undefined) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |                 mappings.push(mapping) | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |         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) | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-30 18:48:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |     private async CloseChangeset(changesetId: number = undefined): Promise<void> { | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |         const self = this | 
					
						
							|  |  |  |         return new Promise<void>(function (resolve, reject) { | 
					
						
							|  |  |  |             if (changesetId === undefined) { | 
					
						
							|  |  |  |                 return; | 
					
						
							| 
									
										
										
										
											2021-06-19 18:28:30 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |             self.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) | 
					
						
							|  |  |  |                 resolve() | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2021-06-19 18:28:30 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |     async GetChangesetMeta(csId: number): Promise<{ | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +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] | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Puts the specified tags onto the changesets as they are. | 
					
						
							|  |  |  |      * This method will erase previously set tags | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |     private async UpdateTags( | 
					
						
							|  |  |  |         csId: number, | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |         tags: ChangesetTag[]) { | 
					
						
							| 
									
										
										
										
											2022-03-15 13:44:34 +01:00
										 |  |  |         tags = ChangesetHandler.removeDuplicateMetaTags(tags) | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |         const self = this; | 
					
						
							|  |  |  |         return new Promise<string>(function (resolve, reject) { | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |             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)}"/>`) | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             self.auth.xhr({ | 
					
						
							|  |  |  |                 method: 'PUT', | 
					
						
							|  |  |  |                 path: '/api/0.6/changeset/' + csId, | 
					
						
							|  |  |  |                 options: {header: {'Content-Type': 'text/xml'}}, | 
					
						
							|  |  |  |                 content: [`<osm><changeset>`, | 
					
						
							|  |  |  |                     metadata, | 
					
						
							|  |  |  |                     `</changeset></osm>`].join("") | 
					
						
							|  |  |  |             }, function (err, response) { | 
					
						
							|  |  |  |                 if (response === undefined) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |                     console.error("Updating the tags of changeset "+csId+" failed:", err); | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |                     reject(err) | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     resolve(response); | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |             }); | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |      | 
					
						
							|  |  |  |     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 | 
					
						
							|  |  |  |         })) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Opens a changeset with the specified tags | 
					
						
							|  |  |  |      * @param changesetTags | 
					
						
							|  |  |  |      * @constructor | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |     private OpenChangeset( | 
					
						
							|  |  |  |         changesetTags: ChangesetTag[] | 
					
						
							|  |  |  |     ): Promise<number> { | 
					
						
							|  |  |  |         const self = this; | 
					
						
							|  |  |  |         return new Promise<number>(function (resolve, reject) { | 
					
						
							| 
									
										
										
										
											2020-09-30 22:22:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |             const metadata = changesetTags.map(cstag => [cstag.key, cstag.value]) | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |                 .filter(kv => (kv[1] ?? "") !== "") | 
					
						
							|  |  |  |                 .map(kv => `<tag k="${kv[0]}" v="${escapeHtml(kv[1])}"/>`) | 
					
						
							|  |  |  |                 .join("\n") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             self.auth.xhr({ | 
					
						
							|  |  |  |                 method: 'PUT', | 
					
						
							|  |  |  |                 path: '/api/0.6/changeset/create', | 
					
						
							|  |  |  |                 options: {header: {'Content-Type': 'text/xml'}}, | 
					
						
							|  |  |  |                 content: [`<osm><changeset>`, | 
					
						
							|  |  |  |                     metadata, | 
					
						
							|  |  |  |                     `</changeset></osm>`].join("") | 
					
						
							|  |  |  |             }, function (err, response) { | 
					
						
							|  |  |  |                 if (response === undefined) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |                     console.error("Opening a changeset failed:", err); | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |                     reject(err) | 
					
						
							|  |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2021-10-04 03:12:42 +02:00
										 |  |  |                     resolve(Number(response)); | 
					
						
							| 
									
										
										
										
											2021-07-13 00:40:27 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |             }); | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-30 18:48:23 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Upload a changesetXML | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-03-14 03:33:03 +01:00
										 |  |  |     private UploadChange(changesetId: number, | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |                       changesetXML: string): Promise<Map<string, string>> { | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |         const self = this; | 
					
						
							|  |  |  |         return new Promise(function (resolve, reject) { | 
					
						
							|  |  |  |             self.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) { | 
					
						
							| 
									
										
										
										
											2022-04-08 04:18:53 +02:00
										 |  |  |                     console.error("Uploading an actual change failed", err); | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |                     reject(err); | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |                 const changes = self.parseUploadChangesetResponse(response); | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |                 console.log("Uploaded changeset ", changesetId); | 
					
						
							| 
									
										
										
										
											2022-02-01 00:09:28 +01:00
										 |  |  |                 resolve(changes); | 
					
						
							| 
									
										
										
										
											2021-09-26 23:35:26 +02:00
										 |  |  |             }); | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-06 22:01:01 +01:00
										 |  |  | } |