| 
									
										
										
										
											2023-09-01 21:36:39 +02:00
										 |  |  | // @ts-ignore
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | import { osmAuth } from "osm-auth" | 
					
						
							|  |  |  | import { Store, Stores, UIEventSource } from "../UIEventSource" | 
					
						
							|  |  |  | import { OsmPreferences } from "./OsmPreferences" | 
					
						
							|  |  |  | import { Utils } from "../../Utils" | 
					
						
							|  |  |  | import { LocalStorageSource } from "../Web/LocalStorageSource" | 
					
						
							|  |  |  | import { AuthConfig } from "./AuthConfig" | 
					
						
							|  |  |  | import Constants from "../../Models/Constants" | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  | import OSMAuthInstance = OSMAuth.OSMAuthInstance | 
					
						
							| 
									
										
										
										
											2023-09-25 02:55:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-04 04:06:21 +01:00
										 |  |  | export default class UserDetails { | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     public loggedIn = false | 
					
						
							|  |  |  |     public name = "Not logged in" | 
					
						
							|  |  |  |     public uid: number | 
					
						
							|  |  |  |     public csCount = 0 | 
					
						
							|  |  |  |     public img?: string | 
					
						
							|  |  |  |     public unreadMessages = 0 | 
					
						
							|  |  |  |     public totalMessages: number = 0 | 
					
						
							|  |  |  |     public home: { lon: number; lat: number } | 
					
						
							|  |  |  |     public backend: string | 
					
						
							|  |  |  |     public account_created: string | 
					
						
							|  |  |  |     public tracesCount: number = 0 | 
					
						
							|  |  |  |     public description: string | 
					
						
							| 
									
										
										
										
											2023-11-19 01:05:15 +01:00
										 |  |  |     public languages: string[] | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     constructor(backend: string) { | 
					
						
							|  |  |  |         this.backend = backend | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-01 21:36:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-06 03:30:18 +01:00
										 |  |  | export type OsmServiceState = "online" | "readonly" | "offline" | "unknown" | "unreachable" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-24 00:35:19 +02:00
										 |  |  | export class OsmConnection { | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |     public auth: OSMAuthInstance | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     public userDetails: UIEventSource<UserDetails> | 
					
						
							|  |  |  |     public isLoggedIn: Store<boolean> | 
					
						
							|  |  |  |     public gpxServiceIsOnline: UIEventSource<OsmServiceState> = new UIEventSource<OsmServiceState>( | 
					
						
							|  |  |  |         "unknown" | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     public apiIsOnline: UIEventSource<OsmServiceState> = new UIEventSource<OsmServiceState>( | 
					
						
							|  |  |  |         "unknown" | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public loadingStatus = new UIEventSource<"not-attempted" | "loading" | "error" | "logged-in">( | 
					
						
							|  |  |  |         "not-attempted" | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     public preferencesHandler: OsmPreferences | 
					
						
							|  |  |  |     public readonly _oauth_config: AuthConfig | 
					
						
							|  |  |  |     private readonly _dryRun: Store<boolean> | 
					
						
							|  |  |  |     private readonly fakeUser: boolean | 
					
						
							|  |  |  |     private _onLoggedIn: ((userDetails: UserDetails) => void)[] = [] | 
					
						
							|  |  |  |     private readonly _iframeMode: Boolean | boolean | 
					
						
							|  |  |  |     private readonly _singlePage: boolean | 
					
						
							|  |  |  |     private isChecking = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constructor(options?: { | 
					
						
							|  |  |  |         dryRun?: Store<boolean> | 
					
						
							|  |  |  |         fakeUser?: false | boolean | 
					
						
							|  |  |  |         oauth_token?: UIEventSource<string> | 
					
						
							|  |  |  |         // Used to keep multiple changesets open and to write to the correct changeset
 | 
					
						
							|  |  |  |         singlePage?: boolean | 
					
						
							|  |  |  |         attemptLogin?: true | boolean | 
					
						
							|  |  |  |     }) { | 
					
						
							|  |  |  |         options ??= {} | 
					
						
							|  |  |  |         this.fakeUser = options?.fakeUser ?? false | 
					
						
							|  |  |  |         this._singlePage = options?.singlePage ?? true | 
					
						
							|  |  |  |         this._oauth_config = Constants.osmAuthConfig | 
					
						
							|  |  |  |         console.debug("Using backend", this._oauth_config.url) | 
					
						
							|  |  |  |         this._iframeMode = Utils.runningFromConsole ? false : window !== window.top | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Check if there are settings available in environment variables, and if so, use those
 | 
					
						
							|  |  |  |         if ( | 
					
						
							|  |  |  |             import.meta.env.VITE_OSM_OAUTH_CLIENT_ID !== undefined && | 
					
						
							|  |  |  |             import.meta.env.VITE_OSM_OAUTH_SECRET !== undefined | 
					
						
							|  |  |  |         ) { | 
					
						
							|  |  |  |             console.debug("Using environment variables for oauth config") | 
					
						
							| 
									
										
										
										
											2023-10-03 20:09:19 +02:00
										 |  |  |             this._oauth_config.oauth_client_id = import.meta.env.VITE_OSM_OAUTH_CLIENT_ID | 
					
						
							|  |  |  |             this._oauth_config.oauth_secret = import.meta.env.VITE_OSM_OAUTH_SECRET | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.userDetails = new UIEventSource<UserDetails>( | 
					
						
							|  |  |  |             new UserDetails(this._oauth_config.url), | 
					
						
							|  |  |  |             "userDetails" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         if (options.fakeUser) { | 
					
						
							|  |  |  |             const ud = this.userDetails.data | 
					
						
							|  |  |  |             ud.csCount = 5678 | 
					
						
							| 
									
										
										
										
											2023-10-30 13:45:44 +01:00
										 |  |  |             ud.uid = 42 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |             ud.loggedIn = true | 
					
						
							|  |  |  |             ud.unreadMessages = 0 | 
					
						
							|  |  |  |             ud.name = "Fake user" | 
					
						
							|  |  |  |             ud.totalMessages = 42 | 
					
						
							| 
									
										
										
										
											2023-11-19 01:05:15 +01:00
										 |  |  |             ud.languages = ["en"] | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         const self = this | 
					
						
							|  |  |  |         this.UpdateCapabilities() | 
					
						
							|  |  |  |         this.isLoggedIn = this.userDetails.map( | 
					
						
							|  |  |  |             (user) => | 
					
						
							|  |  |  |                 user.loggedIn && | 
					
						
							|  |  |  |                 (self.apiIsOnline.data === "unknown" || self.apiIsOnline.data === "online"), | 
					
						
							|  |  |  |             [this.apiIsOnline] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         this.isLoggedIn.addCallback((isLoggedIn) => { | 
					
						
							|  |  |  |             if (self.userDetails.data.loggedIn == false && isLoggedIn == true) { | 
					
						
							|  |  |  |                 // We have an inconsistency: the userdetails say we _didn't_ log in, but this actor says we do
 | 
					
						
							|  |  |  |                 // This means someone attempted to toggle this; so we attempt to login!
 | 
					
						
							|  |  |  |                 self.AttemptLogin() | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this._dryRun = options.dryRun ?? new UIEventSource<boolean>(false) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.updateAuthObject() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.preferencesHandler = new OsmPreferences( | 
					
						
							|  |  |  |             this.auth, | 
					
						
							|  |  |  |             <any /*This is needed to make the tests work*/>this | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (options.oauth_token?.data !== undefined) { | 
					
						
							|  |  |  |             console.log(options.oauth_token.data) | 
					
						
							|  |  |  |             const self = this | 
					
						
							| 
									
										
										
										
											2023-11-02 04:35:32 +01:00
										 |  |  |             this.auth.bootstrapToken(options.oauth_token.data, (err, result) => { | 
					
						
							|  |  |  |                 console.log("Bootstrap token called back", err, result) | 
					
						
							|  |  |  |                 self.AttemptLogin() | 
					
						
							|  |  |  |             }) | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             options.oauth_token.setData(undefined) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (this.auth.authenticated() && options.attemptLogin !== false) { | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |             this.AttemptLogin() | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             console.log("Not authenticated") | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     public GetPreference( | 
					
						
							|  |  |  |         key: string, | 
					
						
							|  |  |  |         defaultValue: string = undefined, | 
					
						
							|  |  |  |         options?: { | 
					
						
							|  |  |  |             documentation?: string | 
					
						
							|  |  |  |             prefix?: string | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ): UIEventSource<string> { | 
					
						
							|  |  |  |         return this.preferencesHandler.GetPreference(key, defaultValue, options) | 
					
						
							| 
									
										
										
										
											2020-08-26 15:36:04 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public GetLongPreference(key: string, prefix: string = "mapcomplete-"): UIEventSource<string> { | 
					
						
							|  |  |  |         return this.preferencesHandler.GetLongPreference(key, prefix) | 
					
						
							| 
									
										
										
										
											2020-06-24 00:35:19 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public OnLoggedIn(action: (userDetails: UserDetails) => void) { | 
					
						
							|  |  |  |         this._onLoggedIn.push(action) | 
					
						
							| 
									
										
										
										
											2023-03-26 05:58:28 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public LogOut() { | 
					
						
							|  |  |  |         this.auth.logout() | 
					
						
							|  |  |  |         this.userDetails.data.loggedIn = false | 
					
						
							|  |  |  |         this.userDetails.data.csCount = 0 | 
					
						
							|  |  |  |         this.userDetails.data.name = "" | 
					
						
							|  |  |  |         this.userDetails.ping() | 
					
						
							|  |  |  |         console.log("Logged out") | 
					
						
							|  |  |  |         this.loadingStatus.setData("not-attempted") | 
					
						
							| 
									
										
										
										
											2023-10-19 16:34:42 +02:00
										 |  |  |         this.preferencesHandler.preferences.setData(undefined) | 
					
						
							| 
									
										
										
										
											2023-03-26 05:58:28 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * The backend host, without path or trailing '/' | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * new OsmConnection().Backend() // => "https://www.openstreetmap.org"
 | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public Backend(): string { | 
					
						
							|  |  |  |         return this._oauth_config.url | 
					
						
							| 
									
										
										
										
											2023-03-26 05:58:28 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public AttemptLogin() { | 
					
						
							|  |  |  |         this.UpdateCapabilities() | 
					
						
							| 
									
										
										
										
											2023-10-17 01:36:22 +02:00
										 |  |  |         if (this.loadingStatus.data !== "logged-in") { | 
					
						
							|  |  |  |             // Stay 'logged-in' if we are already logged in; this simply means we are checking for messages
 | 
					
						
							|  |  |  |             this.loadingStatus.setData("loading") | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         if (this.fakeUser) { | 
					
						
							|  |  |  |             this.loadingStatus.setData("logged-in") | 
					
						
							|  |  |  |             console.log("AttemptLogin called, but ignored as fakeUser is set") | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2023-03-26 05:58:28 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         const self = this | 
					
						
							|  |  |  |         console.log("Trying to log in...") | 
					
						
							|  |  |  |         this.updateAuthObject() | 
					
						
							|  |  |  |         LocalStorageSource.Get("location_before_login").setData( | 
					
						
							|  |  |  |             Utils.runningFromConsole ? undefined : window.location.href | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         this.auth.xhr( | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 method: "GET", | 
					
						
							|  |  |  |                 path: "/api/0.6/user/details", | 
					
						
							|  |  |  |             }, | 
					
						
							| 
									
										
										
										
											2023-11-19 01:05:15 +01:00
										 |  |  |             function (err, details: XMLDocument) { | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |                 if (err != null) { | 
					
						
							|  |  |  |                     console.log(err) | 
					
						
							|  |  |  |                     self.loadingStatus.setData("error") | 
					
						
							|  |  |  |                     if (err.status == 401) { | 
					
						
							|  |  |  |                         console.log("Clearing tokens...") | 
					
						
							|  |  |  |                         // Not authorized - our token probably got revoked
 | 
					
						
							|  |  |  |                         self.auth.logout() | 
					
						
							|  |  |  |                         self.LogOut() | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (details == null) { | 
					
						
							|  |  |  |                     self.loadingStatus.setData("error") | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 self.CheckForMessagesContinuously() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // details is an XML DOM of user details
 | 
					
						
							|  |  |  |                 let userInfo = details.getElementsByTagName("user")[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 let data = self.userDetails.data | 
					
						
							|  |  |  |                 data.loggedIn = true | 
					
						
							|  |  |  |                 console.log("Login completed, userinfo is ", userInfo) | 
					
						
							|  |  |  |                 data.name = userInfo.getAttribute("display_name") | 
					
						
							|  |  |  |                 data.account_created = userInfo.getAttribute("account_created") | 
					
						
							|  |  |  |                 data.uid = Number(userInfo.getAttribute("id")) | 
					
						
							| 
									
										
										
										
											2023-11-19 01:05:15 +01:00
										 |  |  |                 data.languages = Array.from( | 
					
						
							|  |  |  |                     userInfo.getElementsByTagName("languages")[0].getElementsByTagName("lang") | 
					
						
							|  |  |  |                 ).map((l) => l.textContent) | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |                 data.csCount = Number.parseInt( | 
					
						
							| 
									
										
										
										
											2023-11-19 01:05:15 +01:00
										 |  |  |                     userInfo.getElementsByTagName("changesets")[0].getAttribute("count") ?? "0" | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |                 ) | 
					
						
							|  |  |  |                 data.tracesCount = Number.parseInt( | 
					
						
							| 
									
										
										
										
											2023-11-19 01:05:15 +01:00
										 |  |  |                     userInfo.getElementsByTagName("traces")[0].getAttribute("count") ?? "0" | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 data.img = undefined | 
					
						
							|  |  |  |                 const imgEl = userInfo.getElementsByTagName("img") | 
					
						
							|  |  |  |                 if (imgEl !== undefined && imgEl[0] !== undefined) { | 
					
						
							|  |  |  |                     data.img = imgEl[0].getAttribute("href") | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 const description = userInfo.getElementsByTagName("description") | 
					
						
							|  |  |  |                 if (description !== undefined && description[0] !== undefined) { | 
					
						
							|  |  |  |                     data.description = description[0]?.innerHTML | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 const homeEl = userInfo.getElementsByTagName("home") | 
					
						
							|  |  |  |                 if (homeEl !== undefined && homeEl[0] !== undefined) { | 
					
						
							|  |  |  |                     const lat = parseFloat(homeEl[0].getAttribute("lat")) | 
					
						
							|  |  |  |                     const lon = parseFloat(homeEl[0].getAttribute("lon")) | 
					
						
							|  |  |  |                     data.home = { lat: lat, lon: lon } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 self.loadingStatus.setData("logged-in") | 
					
						
							|  |  |  |                 const messages = userInfo | 
					
						
							|  |  |  |                     .getElementsByTagName("messages")[0] | 
					
						
							|  |  |  |                     .getElementsByTagName("received")[0] | 
					
						
							|  |  |  |                 data.unreadMessages = parseInt(messages.getAttribute("unread")) | 
					
						
							|  |  |  |                 data.totalMessages = parseInt(messages.getAttribute("count")) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 self.userDetails.ping() | 
					
						
							|  |  |  |                 for (const action of self._onLoggedIn) { | 
					
						
							|  |  |  |                     action(self.userDetails.data) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 self._onLoggedIn = [] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Interact with the API. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |      * @param path the path to query, without host and without '/api/0.6'. Example 'notes/1234/close' | 
					
						
							|  |  |  |      * @param method | 
					
						
							|  |  |  |      * @param header | 
					
						
							|  |  |  |      * @param content | 
					
						
							|  |  |  |      * @param allowAnonymous if set, will use the anonymous-connection if the main connection is not authenticated | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public async interact( | 
					
						
							|  |  |  |         path: string, | 
					
						
							|  |  |  |         method: "GET" | "POST" | "PUT" | "DELETE", | 
					
						
							|  |  |  |         header?: Record<string, string | number>, | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |         content?: string, | 
					
						
							|  |  |  |         allowAnonymous: boolean = false | 
					
						
							|  |  |  |     ): Promise<string> { | 
					
						
							|  |  |  |         let connection: OSMAuthInstance = this.auth | 
					
						
							| 
									
										
										
										
											2023-11-02 04:35:32 +01:00
										 |  |  |         if (allowAnonymous && !this.auth.authenticated()) { | 
					
						
							|  |  |  |             const possibleResult = await Utils.downloadAdvanced( | 
					
						
							|  |  |  |                 `${this.Backend()}/api/0.6/${path}`, | 
					
						
							|  |  |  |                 header, | 
					
						
							|  |  |  |                 method, | 
					
						
							|  |  |  |                 content | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             if (possibleResult["content"]) { | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |                 return possibleResult["content"] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             console.error(possibleResult) | 
					
						
							| 
									
										
										
										
											2023-11-02 04:35:32 +01:00
										 |  |  |             throw "Could not interact with OSM:" + possibleResult["error"] | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         return new Promise((ok, error) => { | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |             connection.xhr( | 
					
						
							| 
									
										
										
										
											2023-11-02 04:35:32 +01:00
										 |  |  |                 <any>{ | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |                     method, | 
					
						
							|  |  |  |                     options: { | 
					
						
							|  |  |  |                         header, | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     content, | 
					
						
							|  |  |  |                     path: `/api/0.6/${path}`, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 function (err, response) { | 
					
						
							|  |  |  |                     if (err !== null) { | 
					
						
							|  |  |  |                         error(err) | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         ok(response) | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public async post( | 
					
						
							|  |  |  |         path: string, | 
					
						
							|  |  |  |         content?: string, | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |         header?: Record<string, string | number>, | 
					
						
							|  |  |  |         allowAnonymous: boolean = false | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     ): Promise<any> { | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |         return await this.interact(path, "POST", header, content, allowAnonymous) | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-26 05:58:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     public async put( | 
					
						
							|  |  |  |         path: string, | 
					
						
							|  |  |  |         content?: string, | 
					
						
							|  |  |  |         header?: Record<string, string | number> | 
					
						
							|  |  |  |     ): Promise<any> { | 
					
						
							|  |  |  |         return await this.interact(path, "PUT", header, content) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-02 04:35:32 +01:00
										 |  |  |     public async get( | 
					
						
							|  |  |  |         path: string, | 
					
						
							|  |  |  |         header?: Record<string, string | number>, | 
					
						
							|  |  |  |         allowAnonymous: boolean = false | 
					
						
							|  |  |  |     ): Promise<string> { | 
					
						
							|  |  |  |         return await this.interact(path, "GET", header, undefined, allowAnonymous) | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public closeNote(id: number | string, text?: string): Promise<void> { | 
					
						
							|  |  |  |         let textSuffix = "" | 
					
						
							|  |  |  |         if ((text ?? "") !== "") { | 
					
						
							|  |  |  |             textSuffix = "?text=" + encodeURIComponent(text) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (this._dryRun.data) { | 
					
						
							|  |  |  |             console.warn("Dryrun enabled - not actually closing note ", id, " with text ", text) | 
					
						
							|  |  |  |             return new Promise((ok) => { | 
					
						
							|  |  |  |                 ok() | 
					
						
							|  |  |  |             }) | 
					
						
							| 
									
										
										
										
											2022-01-08 04:22:50 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         return this.post(`notes/${id}/close${textSuffix}`) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-08 04:22:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     public reopenNote(id: number | string, text?: string): Promise<void> { | 
					
						
							|  |  |  |         if (this._dryRun.data) { | 
					
						
							|  |  |  |             console.warn("Dryrun enabled - not actually reopening note ", id, " with text ", text) | 
					
						
							|  |  |  |             return new Promise((ok) => { | 
					
						
							|  |  |  |                 ok() | 
					
						
							|  |  |  |             }) | 
					
						
							| 
									
										
										
										
											2022-01-14 01:41:19 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         let textSuffix = "" | 
					
						
							|  |  |  |         if ((text ?? "") !== "") { | 
					
						
							|  |  |  |             textSuffix = "?text=" + encodeURIComponent(text) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return this.post(`notes/${id}/reopen${textSuffix}`) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public async openNote(lat: number, lon: number, text: string): Promise<{ id: number }> { | 
					
						
							|  |  |  |         if (this._dryRun.data) { | 
					
						
							|  |  |  |             console.warn("Dryrun enabled - not actually opening note with text ", text) | 
					
						
							|  |  |  |             return new Promise<{ id: number }>((ok) => { | 
					
						
							|  |  |  |                 window.setTimeout( | 
					
						
							|  |  |  |                     () => ok({ id: Math.floor(Math.random() * 1000) }), | 
					
						
							|  |  |  |                     Math.random() * 5000 | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Lat and lon must be strings for the API to accept it
 | 
					
						
							|  |  |  |         const content = `lat=${lat}&lon=${lon}&text=${encodeURIComponent(text)}` | 
					
						
							| 
									
										
										
										
											2023-11-02 04:35:32 +01:00
										 |  |  |         const response = await this.post( | 
					
						
							|  |  |  |             "notes.json", | 
					
						
							|  |  |  |             content, | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "Content-Type": "application/x-www-form-urlencoded; charset=UTF-8", | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             true | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         const parsed = JSON.parse(response) | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |         console.log("Got result:", parsed) | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         const id = parsed.properties | 
					
						
							|  |  |  |         console.log("OPENED NOTE", id) | 
					
						
							|  |  |  |         return id | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-11 02:20:57 +01:00
										 |  |  |     public static GpxTrackVisibility = ["private", "public", "trackable", "identifiable"] as const | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     public async uploadGpxTrack( | 
					
						
							|  |  |  |         gpx: string, | 
					
						
							|  |  |  |         options: { | 
					
						
							|  |  |  |             description: string | 
					
						
							| 
									
										
										
										
											2024-01-11 02:20:57 +01:00
										 |  |  |             visibility: (typeof OsmConnection.GpxTrackVisibility)[number] | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |             filename?: string | 
					
						
							|  |  |  |             /** | 
					
						
							|  |  |  |              * Some words to give some properties; | 
					
						
							|  |  |  |              * | 
					
						
							|  |  |  |              * Note: these are called 'tags' on the wiki, but I opted to name them 'labels' instead as they aren't "key=value" tags, but just words. | 
					
						
							|  |  |  |              */ | 
					
						
							|  |  |  |             labels: string[] | 
					
						
							| 
									
										
										
										
											2022-11-02 14:44:06 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     ): Promise<{ id: number }> { | 
					
						
							|  |  |  |         if (this._dryRun.data) { | 
					
						
							|  |  |  |             console.warn("Dryrun enabled - not actually uploading GPX ", gpx) | 
					
						
							|  |  |  |             return new Promise<{ id: number }>((ok, error) => { | 
					
						
							|  |  |  |                 window.setTimeout( | 
					
						
							|  |  |  |                     () => ok({ id: Math.floor(Math.random() * 1000) }), | 
					
						
							|  |  |  |                     Math.random() * 5000 | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             }) | 
					
						
							| 
									
										
										
										
											2022-02-16 00:56:48 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         const contents = { | 
					
						
							|  |  |  |             file: gpx, | 
					
						
							| 
									
										
										
										
											2024-01-11 02:20:57 +01:00
										 |  |  |             description: options.description, | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |             tags: options.labels?.join(",") ?? "", | 
					
						
							|  |  |  |             visibility: options.visibility, | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-02-16 00:56:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-11 02:20:57 +01:00
										 |  |  |         if (!contents.description) { | 
					
						
							|  |  |  |             throw "The description of a GPS-trace cannot be the empty string, undefined or null" | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         const extras = { | 
					
						
							|  |  |  |             file: | 
					
						
							|  |  |  |                 '; filename="' + | 
					
						
							|  |  |  |                 (options.filename ?? "gpx_track_mapcomplete_" + new Date().toISOString()) + | 
					
						
							|  |  |  |                 '"\r\nContent-Type: application/gpx+xml', | 
					
						
							| 
									
										
										
										
											2022-02-16 00:56:48 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const boundary = "987654" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let body = "" | 
					
						
							|  |  |  |         for (const key in contents) { | 
					
						
							|  |  |  |             body += "--" + boundary + "\r\n" | 
					
						
							|  |  |  |             body += 'Content-Disposition: form-data; name="' + key + '"' | 
					
						
							|  |  |  |             if (extras[key] !== undefined) { | 
					
						
							|  |  |  |                 body += extras[key] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             body += "\r\n\r\n" | 
					
						
							|  |  |  |             body += contents[key] + "\r\n" | 
					
						
							| 
									
										
										
										
											2022-02-16 00:56:48 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         body += "--" + boundary + "--\r\n" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const response = await this.post("gpx/create", body, { | 
					
						
							|  |  |  |             "Content-Type": "multipart/form-data; boundary=" + boundary, | 
					
						
							|  |  |  |             "Content-Length": body.length, | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |         const parsed = JSON.parse(response) | 
					
						
							|  |  |  |         console.log("Uploaded GPX track", parsed) | 
					
						
							|  |  |  |         return { id: parsed } | 
					
						
							| 
									
										
										
										
											2020-07-30 16:34:06 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-01-06 03:30:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     public addCommentToNote(id: number | string, text: string): Promise<void> { | 
					
						
							|  |  |  |         if (this._dryRun.data) { | 
					
						
							|  |  |  |             console.warn("Dryrun enabled - not actually adding comment ", text, "to  note ", id) | 
					
						
							|  |  |  |             return new Promise((ok) => { | 
					
						
							|  |  |  |                 ok() | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ((text ?? "") === "") { | 
					
						
							|  |  |  |             throw "Invalid text!" | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return new Promise((ok, error) => { | 
					
						
							|  |  |  |             this.auth.xhr( | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     method: "POST", | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     path: `/api/0.6/notes/${id}/comment?text=${encodeURIComponent(text)}`, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 function (err, _) { | 
					
						
							|  |  |  |                     if (err !== null) { | 
					
						
							|  |  |  |                         error(err) | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         ok() | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2023-09-25 02:55:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * To be called by land.html | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public finishLogin(callback: (previousURL: string) => void) { | 
					
						
							|  |  |  |         this.auth.authenticate(function () { | 
					
						
							|  |  |  |             // Fully authed at this point
 | 
					
						
							|  |  |  |             console.log("Authentication successful!") | 
					
						
							|  |  |  |             const previousLocation = LocalStorageSource.Get("location_before_login") | 
					
						
							|  |  |  |             callback(previousLocation.data) | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2023-01-06 03:30:18 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     private updateAuthObject() { | 
					
						
							|  |  |  |         let pwaStandAloneMode = false | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |             if (Utils.runningFromConsole) { | 
					
						
							|  |  |  |                 pwaStandAloneMode = true | 
					
						
							|  |  |  |             } else if ( | 
					
						
							|  |  |  |                 window.matchMedia("(display-mode: standalone)").matches || | 
					
						
							|  |  |  |                 window.matchMedia("(display-mode: fullscreen)").matches | 
					
						
							|  |  |  |             ) { | 
					
						
							|  |  |  |                 pwaStandAloneMode = true | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } catch (e) { | 
					
						
							|  |  |  |             console.warn( | 
					
						
							|  |  |  |                 "Detecting standalone mode failed", | 
					
						
							|  |  |  |                 e, | 
					
						
							|  |  |  |                 ". Assuming in browser and not worrying furhter" | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2023-01-06 03:30:18 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         const standalone = this._iframeMode || pwaStandAloneMode || !this._singlePage | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // In standalone mode, we DON'T use single page login, as 'redirecting' opens a new window anyway...
 | 
					
						
							|  |  |  |         // Same for an iframe...
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.auth = new osmAuth({ | 
					
						
							|  |  |  |             client_id: this._oauth_config.oauth_client_id, | 
					
						
							|  |  |  |             url: this._oauth_config.url, | 
					
						
							| 
									
										
										
										
											2023-10-24 00:35:42 +02:00
										 |  |  |             scope: "read_prefs write_prefs write_api write_gpx write_notes openid", | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |             redirect_uri: Utils.runningFromConsole | 
					
						
							|  |  |  |                 ? "https://mapcomplete.org/land.html" | 
					
						
							|  |  |  |                 : window.location.protocol + "//" + window.location.host + "/land.html", | 
					
						
							|  |  |  |             singlepage: !standalone, | 
					
						
							|  |  |  |             auto: true, | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2023-09-25 02:55:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     private CheckForMessagesContinuously() { | 
					
						
							|  |  |  |         const self = this | 
					
						
							|  |  |  |         if (this.isChecking) { | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         this.isChecking = true | 
					
						
							|  |  |  |         Stores.Chronic(5 * 60 * 1000).addCallback((_) => { | 
					
						
							|  |  |  |             if (self.isLoggedIn.data) { | 
					
						
							|  |  |  |                 self.AttemptLogin() | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2023-09-25 02:55:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     private UpdateCapabilities(): void { | 
					
						
							|  |  |  |         const self = this | 
					
						
							| 
									
										
										
										
											2023-12-16 01:29:42 +01:00
										 |  |  |         if (this.fakeUser) { | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |         this.FetchCapabilities().then(({ api, gpx }) => { | 
					
						
							|  |  |  |             self.apiIsOnline.setData(api) | 
					
						
							|  |  |  |             self.gpxServiceIsOnline.setData(gpx) | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2023-09-25 02:55:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-02 04:35:32 +01:00
										 |  |  |     private readonly _userInfoCache: Record<number, any> = {} | 
					
						
							|  |  |  |     public async getInformationAboutUser(id: number): Promise<{ | 
					
						
							|  |  |  |         id: number | 
					
						
							|  |  |  |         display_name: string | 
					
						
							|  |  |  |         account_created: string | 
					
						
							|  |  |  |         description: string | 
					
						
							|  |  |  |         contributor_terms: { agreed: boolean } | 
					
						
							|  |  |  |         roles: [] | 
					
						
							|  |  |  |         changesets: { count: number } | 
					
						
							|  |  |  |         traces: { count: number } | 
					
						
							|  |  |  |         blocks: { received: { count: number; active: number } } | 
					
						
							|  |  |  |     }> { | 
					
						
							|  |  |  |         if (id === undefined) { | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (this._userInfoCache[id]) { | 
					
						
							|  |  |  |             return this._userInfoCache[id] | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const info = await this.get("user/" + id + ".json", { accepts: "application/json" }, true) | 
					
						
							|  |  |  |         const parsed = JSON.parse(info)["user"] | 
					
						
							|  |  |  |         this._userInfoCache[id] = parsed | 
					
						
							|  |  |  |         return parsed | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-27 22:21:35 +02:00
										 |  |  |     private async FetchCapabilities(): Promise<{ api: OsmServiceState; gpx: OsmServiceState }> { | 
					
						
							|  |  |  |         if (Utils.runningFromConsole) { | 
					
						
							|  |  |  |             return { api: "online", gpx: "online" } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const result = await Utils.downloadAdvanced(this.Backend() + "/api/0.6/capabilities") | 
					
						
							|  |  |  |         if (result["content"] === undefined) { | 
					
						
							|  |  |  |             console.log("Something went wrong:", result) | 
					
						
							|  |  |  |             return { api: "unreachable", gpx: "unreachable" } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const xmlRaw = result["content"] | 
					
						
							|  |  |  |         const parsed = new DOMParser().parseFromString(xmlRaw, "text/xml") | 
					
						
							|  |  |  |         const statusEl = parsed.getElementsByTagName("status")[0] | 
					
						
							|  |  |  |         const api = <OsmServiceState>statusEl.getAttribute("api") | 
					
						
							|  |  |  |         const gpx = <OsmServiceState>statusEl.getAttribute("gpx") | 
					
						
							|  |  |  |         return { api, gpx } | 
					
						
							| 
									
										
										
										
											2023-01-06 03:30:18 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  | } |