forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			141 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			141 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import ImageProvider, { ProvidedImage } from "./ImageProvider"
 | |
| import BaseUIElement from "../../UI/BaseUIElement"
 | |
| import { Utils } from "../../Utils"
 | |
| import Constants from "../../Models/Constants"
 | |
| import { LicenseInfo } from "./LicenseInfo"
 | |
| 
 | |
| export class Imgur extends ImageProvider {
 | |
|     public static readonly defaultValuePrefix = ["https://i.imgur.com"]
 | |
|     public static readonly singleton = new Imgur()
 | |
|     public readonly defaultKeyPrefixes: string[] = ["image"]
 | |
| 
 | |
|     private constructor() {
 | |
|         super()
 | |
|     }
 | |
| 
 | |
|     static uploadMultiple(
 | |
|         title: string,
 | |
|         description: string,
 | |
|         blobs: FileList,
 | |
|         handleSuccessfullUpload: (imageURL: string) => Promise<void>,
 | |
|         allDone: () => void,
 | |
|         onFail: (reason: string) => void,
 | |
|         offset: number = 0
 | |
|     ) {
 | |
|         if (blobs.length == offset) {
 | |
|             allDone()
 | |
|             return
 | |
|         }
 | |
|         const blob = blobs.item(offset)
 | |
|         const self = this
 | |
|         this.uploadImage(
 | |
|             title,
 | |
|             description,
 | |
|             blob,
 | |
|             async (imageUrl) => {
 | |
|                 await handleSuccessfullUpload(imageUrl)
 | |
|                 self.uploadMultiple(
 | |
|                     title,
 | |
|                     description,
 | |
|                     blobs,
 | |
|                     handleSuccessfullUpload,
 | |
|                     allDone,
 | |
|                     onFail,
 | |
|                     offset + 1
 | |
|                 )
 | |
|             },
 | |
|             onFail
 | |
|         )
 | |
|     }
 | |
| 
 | |
|     static uploadImage(
 | |
|         title: string,
 | |
|         description: string,
 | |
|         blob: File,
 | |
|         handleSuccessfullUpload: (imageURL: string) => Promise<void>,
 | |
|         onFail: (reason: string) => void
 | |
|     ) {
 | |
|         const apiUrl = "https://api.imgur.com/3/image"
 | |
|         const apiKey = Constants.ImgurApiKey
 | |
| 
 | |
|         const formData = new FormData()
 | |
|         formData.append("image", blob)
 | |
|         formData.append("title", title)
 | |
|         formData.append("description", description)
 | |
| 
 | |
|         const settings: RequestInit = {
 | |
|             method: "POST",
 | |
|             body: formData,
 | |
|             redirect: "follow",
 | |
|             headers: new Headers({
 | |
|                 Authorization: `Client-ID ${apiKey}`,
 | |
|                 Accept: "application/json",
 | |
|             }),
 | |
|         }
 | |
| 
 | |
|         // Response contains stringified JSON
 | |
|         // Image URL available at response.data.link
 | |
|         fetch(apiUrl, settings)
 | |
|             .then(async function (response) {
 | |
|                 const content = await response.json()
 | |
|                 await handleSuccessfullUpload(content.data.link)
 | |
|             })
 | |
|             .catch((reason) => {
 | |
|                 console.log("Uploading to IMGUR failed", reason)
 | |
|                 // @ts-ignore
 | |
|                 onFail(reason)
 | |
|             })
 | |
|     }
 | |
| 
 | |
|     SourceIcon(): BaseUIElement {
 | |
|         return undefined
 | |
|     }
 | |
| 
 | |
|     public async ExtractUrls(key: string, value: string): Promise<Promise<ProvidedImage>[]> {
 | |
|         if (Imgur.defaultValuePrefix.some((prefix) => value.startsWith(prefix))) {
 | |
|             return [
 | |
|                 Promise.resolve({
 | |
|                     url: value,
 | |
|                     key: key,
 | |
|                     provider: this,
 | |
|                 }),
 | |
|             ]
 | |
|         }
 | |
|         return []
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Download the attribution and license info for the picture at the given URL
 | |
|      *
 | |
|      * const data = {"data":{"id":"I9t6B7B","title":"Station Knokke","description":"author:Pieter Vander Vennet\r\nlicense:CC-BY 4.0\r\nosmid:node\/9812712386","datetime":1655052078,"type":"image\/jpeg","animated":false,"width":2400,"height":1795,"size":910872,"views":2,"bandwidth":1821744,"vote":null,"favorite":false,"nsfw":false,"section":null,"account_url":null,"account_id":null,"is_ad":false,"in_most_viral":false,"has_sound":false,"tags":[],"ad_type":0,"ad_url":"","edited":"0","in_gallery":false,"link":"https:\/\/i.imgur.com\/I9t6B7B.jpg","ad_config":{"safeFlags":["not_in_gallery","share"],"highRiskFlags":[],"unsafeFlags":["sixth_mod_unsafe"],"wallUnsafeFlags":[],"showsAds":false,"showAdLevel":1}},"success":true,"status":200}
 | |
|      * Utils.injectJsonDownloadForTests("https://api.imgur.com/3/image/E0RuAK3", data)
 | |
|      * const licenseInfo = await Imgur.singleton.DownloadAttribution("https://i.imgur.com/E0RuAK3.jpg")
 | |
|      * const expected = new LicenseInfo()
 | |
|      * expected.licenseShortName = "CC-BY 4.0"
 | |
|      * expected.artist = "Pieter Vander Vennet"
 | |
|      * licenseInfo // => expected
 | |
|      */
 | |
|     public async DownloadAttribution(url: string): Promise<LicenseInfo> {
 | |
|         const hash = url.substr("https://i.imgur.com/".length).split(".jpg")[0]
 | |
| 
 | |
|         const apiUrl = "https://api.imgur.com/3/image/" + hash
 | |
|         const response = await Utils.downloadJsonCached(apiUrl, 365 * 24 * 60 * 60, {
 | |
|             Authorization: "Client-ID " + Constants.ImgurApiKey,
 | |
|         })
 | |
| 
 | |
|         const descr: string = response.data.description ?? ""
 | |
|         const data: any = {}
 | |
|         for (const tag of descr.split("\n")) {
 | |
|             const kv = tag.split(":")
 | |
|             const k = kv[0]
 | |
|             data[k] = kv[1]?.replace(/\r/g, "")
 | |
|         }
 | |
| 
 | |
|         const licenseInfo = new LicenseInfo()
 | |
| 
 | |
|         licenseInfo.licenseShortName = data.license
 | |
|         licenseInfo.artist = data.author
 | |
| 
 | |
|         return licenseInfo
 | |
|     }
 | |
| }
 |