forked from MapComplete/MapComplete
		
	
		
			
	
	
		
			157 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
		
		
			
		
	
	
			157 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| 
								 | 
							
								import Script from "./Script"
							 | 
						||
| 
								 | 
							
								import { Overpass } from "../src/Logic/Osm/Overpass"
							 | 
						||
| 
								 | 
							
								import { RegexTag } from "../src/Logic/Tags/RegexTag"
							 | 
						||
| 
								 | 
							
								import Constants from "../src/Models/Constants"
							 | 
						||
| 
								 | 
							
								import { BBox } from "../src/Logic/BBox"
							 | 
						||
| 
								 | 
							
								import { existsSync, readFileSync, writeFileSync } from "fs"
							 | 
						||
| 
								 | 
							
								import { PanoramaxUploader } from "../src/Logic/ImageProviders/Panoramax"
							 | 
						||
| 
								 | 
							
								import { Feature } from "geojson"
							 | 
						||
| 
								 | 
							
								import { LicenseInfo } from "../src/Logic/ImageProviders/LicenseInfo"
							 | 
						||
| 
								 | 
							
								import { GeoOperations } from "../src/Logic/GeoOperations"
							 | 
						||
| 
								 | 
							
								import { Tag } from "../src/Logic/Tags/Tag"
							 | 
						||
| 
								 | 
							
								import { Utils } from "../src/Utils"
							 | 
						||
| 
								 | 
							
								import ChangeTagAction from "../src/Logic/Osm/Actions/ChangeTagAction"
							 | 
						||
| 
								 | 
							
								import { And } from "../src/Logic/Tags/And"
							 | 
						||
| 
								 | 
							
								import { Changes } from "../src/Logic/Osm/Changes"
							 | 
						||
| 
								 | 
							
								import { ChangeDescription } from "../src/Logic/Osm/Actions/ChangeDescription"
							 | 
						||
| 
								 | 
							
								import OsmObjectDownloader from "../src/Logic/Osm/OsmObjectDownloader"
							 | 
						||
| 
								 | 
							
								import { OsmObject } from "../src/Logic/Osm/OsmObject"
							 | 
						||
| 
								 | 
							
								import { createReadStream } from "node:fs"
							 | 
						||
| 
								 | 
							
								import { File } from 'buffer';
							 | 
						||
| 
								 | 
							
								import { open } from 'node:fs/promises';
							 | 
						||
| 
								 | 
							
								import { UploadableTag } from "../src/Logic/Tags/TagTypes"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								export class ImgurToPanoramax extends Script {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private readonly panoramax = new PanoramaxUploader(Constants.panoramax.url, Constants.panoramax.token)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private _imageDirectory: string
							 | 
						||
| 
								 | 
							
								    private _licenseDirectory: string
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private readonly sequenceIds = {
							 | 
						||
| 
								 | 
							
								        test: "7f34cf53-27ff-46c9-ac22-78511fa8457a",
							 | 
						||
| 
								 | 
							
								        cc0: "f0d6f78a-ff95-4db1-8494-6eb44a17bb37",
							 | 
						||
| 
								 | 
							
								        ccby: "288a8052-b475-422c-811a-4f6f1a00015e",
							 | 
						||
| 
								 | 
							
								        ccbysa: "f3d02893-b4c1-4cd6-8b27-e27ab57eb59a",
							 | 
						||
| 
								 | 
							
								    } as const
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    constructor() {
							 | 
						||
| 
								 | 
							
								        super(
							 | 
						||
| 
								 | 
							
								            "Queries OSM for 'imgur'-images, uploads them to Panoramax and creates a changeset to update OSM",
							 | 
						||
| 
								 | 
							
								        )
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    async uploadImage(key: string, feat: Feature, sequences: ({
							 | 
						||
| 
								 | 
							
								        id: string;
							 | 
						||
| 
								 | 
							
								        "stats:items": { count: number }
							 | 
						||
| 
								 | 
							
								    })[]): Promise<UploadableTag | undefined> {
							 | 
						||
| 
								 | 
							
								        const v = feat.properties[key]
							 | 
						||
| 
								 | 
							
								        if (!v) {
							 | 
						||
| 
								 | 
							
								            return undefined
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        const imageHash = v.split("/").at(-1).split(".").at(0)
							 | 
						||
| 
								 | 
							
								        let path: string = undefined
							 | 
						||
| 
								 | 
							
								        if (existsSync(this._imageDirectory + "/" + imageHash + ".jpg")) {
							 | 
						||
| 
								 | 
							
								            path = this._imageDirectory + "/" + imageHash + ".jpg"
							 | 
						||
| 
								 | 
							
								        } else if (existsSync(this._imageDirectory + "/" + imageHash + ".jpeg")) {
							 | 
						||
| 
								 | 
							
								            path = this._imageDirectory + "/" + imageHash + ".jpeg"
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        if (!path) {
							 | 
						||
| 
								 | 
							
								            return undefined
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        const licensePath = this._licenseDirectory + "/" + v.replaceAll(/[^a-zA-Z0-9]/g, "_") + ".json"
							 | 
						||
| 
								 | 
							
								        if (!existsSync(licensePath)) {
							 | 
						||
| 
								 | 
							
								            return undefined
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        const licenseText: LicenseInfo = JSON.parse(readFileSync(licensePath, "utf8"))
							 | 
						||
| 
								 | 
							
								        if (!licenseText.licenseShortName) {
							 | 
						||
| 
								 | 
							
								            console.log("No license found for", path, licenseText)
							 | 
						||
| 
								 | 
							
								            return undefined
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        const license = licenseText.licenseShortName.toLowerCase().split(" ")[0].replace(/-/g, "")
							 | 
						||
| 
								 | 
							
								        const sequence = this.sequenceIds[license]
							 | 
						||
| 
								 | 
							
								        const author = licenseText.artist
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const handle = await open(path);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const stat = await handle.stat();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        class MyFile extends File {
							 | 
						||
| 
								 | 
							
								            // we should set correct size
							 | 
						||
| 
								 | 
							
								            // otherwise we will encounter UND_ERR_REQ_CONTENT_LENGTH_MISMATCH
							 | 
						||
| 
								 | 
							
								            size = stat.size;
							 | 
						||
| 
								 | 
							
								            stream = undefined
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const file = new MyFile([], path)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        file.stream = function() {
							 | 
						||
| 
								 | 
							
								            return handle.readableWebStream();
							 | 
						||
| 
								 | 
							
								        };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        console.log("Uploading", imageHash, sequence)
							 | 
						||
| 
								 | 
							
								        const result = await this.panoramax.uploadImage(<any> file, GeoOperations.centerpointCoordinates(feat), author, true, sequence)
							 | 
						||
| 
								 | 
							
								        await handle.close()
							 | 
						||
| 
								 | 
							
								        return new And([new Tag(key.replace("image", result.key), result.value),
							 | 
						||
| 
								 | 
							
								            new Tag(key,"")])
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    async main(args: string[]): Promise<void> {
							 | 
						||
| 
								 | 
							
								        this._imageDirectory = args[0] ?? "/home/pietervdvn/data/imgur-image-backup"
							 | 
						||
| 
								 | 
							
								        this._licenseDirectory = args[1] ?? "/home/pietervdvn/git/MapComplete-data/ImageLicenseInfo"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const bounds = new BBox([[3.6984301050112833, 51.06715570450848], [3.7434328399847914, 51.039379568816145]])
							 | 
						||
| 
								 | 
							
								        const maxcount = 100
							 | 
						||
| 
								 | 
							
								        const filter = new RegexTag("image", /^https:\/\/i.imgur.com\/.*/)
							 | 
						||
| 
								 | 
							
								        const overpass = new Overpass(filter, [], Constants.defaultOverpassUrls[0])
							 | 
						||
| 
								 | 
							
								        const features = (await overpass.queryGeoJson(bounds))[0].features
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        let converted = 0
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const pano = this.panoramax.panoramax
							 | 
						||
| 
								 | 
							
								        const sequences = await pano.mySequences()
							 | 
						||
| 
								 | 
							
								        const changes: ChangeDescription[] = []
							 | 
						||
| 
								 | 
							
								        do {
							 | 
						||
| 
								 | 
							
								            const f = features.shift()
							 | 
						||
| 
								 | 
							
								            if (!f) {
							 | 
						||
| 
								 | 
							
								                break
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            const changedTags: (UploadableTag | undefined)[] = []
							 | 
						||
| 
								 | 
							
								            for (const k of ["image", "image:menu", "image:streetsign"]) {
							 | 
						||
| 
								 | 
							
								                changedTags.push(await this.uploadImage(k, f, sequences))
							 | 
						||
| 
								 | 
							
								                for (let i = 0; i < 20; i++) {
							 | 
						||
| 
								 | 
							
								                    changedTags.push(
							 | 
						||
| 
								 | 
							
								                        await this.uploadImage(k + ":" + i, f, sequences),
							 | 
						||
| 
								 | 
							
								                    )
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            const action = new ChangeTagAction(f.properties.id, new And(Utils.NoNull(changedTags)),
							 | 
						||
| 
								 | 
							
								                f.properties, {
							 | 
						||
| 
								 | 
							
								                    theme: "image-mover",
							 | 
						||
| 
								 | 
							
								                    changeType: "link-image",
							 | 
						||
| 
								 | 
							
								                },
							 | 
						||
| 
								 | 
							
								            )
							 | 
						||
| 
								 | 
							
								            changes.push(...await action.CreateChangeDescriptions())
							 | 
						||
| 
								 | 
							
								            converted++
							 | 
						||
| 
								 | 
							
								        } while (converted < maxcount)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        const modif: string[] = Utils.Dedup(changes.map(ch => ch.type + "/" + ch.id))
							 | 
						||
| 
								 | 
							
								        const modifiedObjectsFresh =
							 | 
						||
| 
								 | 
							
								          <OsmObject[]>  (await Promise.all(modif.map(id => new OsmObjectDownloader().DownloadObjectAsync(id))))
							 | 
						||
| 
								 | 
							
								                .filter(m => m !== "deleted")
							 | 
						||
| 
								 | 
							
								        const modifiedObjects = Changes.createChangesetObjectsStatic(
							 | 
						||
| 
								 | 
							
								            changes,
							 | 
						||
| 
								 | 
							
								            modifiedObjectsFresh,false, [])
							 | 
						||
| 
								 | 
							
								        const cs = Changes.buildChangesetXML("0", modifiedObjects)
							 | 
						||
| 
								 | 
							
								        writeFileSync("imgur_to_panoramax.osc", cs, "utf8")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								new ImgurToPanoramax().run()
							 |