MapComplete/src/Logic/ImageProviders/Imgur.ts

104 lines
4.1 KiB
TypeScript
Raw Normal View History

2023-09-28 23:50:27 +02:00
import ImageProvider, { ProvidedImage } from "./ImageProvider"
import BaseUIElement from "../../UI/BaseUIElement"
import { Utils } from "../../Utils"
import Constants from "../../Models/Constants"
import { LicenseInfo } from "./LicenseInfo"
import { ImageUploader } from "./ImageUploader"
2023-09-28 23:50:27 +02:00
export class Imgur extends ImageProvider implements ImageUploader {
2021-11-07 16:34:51 +01:00
public static readonly defaultValuePrefix = ["https://i.imgur.com"]
2022-09-08 21:40:48 +02:00
public static readonly singleton = new Imgur()
public readonly defaultKeyPrefixes: string[] = ["image"]
2023-09-28 23:50:27 +02:00
public readonly maxFileSizeInMegabytes = 10
private constructor() {
2022-09-08 21:40:48 +02:00
super()
}
/**
* Uploads an image, returns the URL where to find the image
* @param title
* @param description
* @param blob
*/
public async uploadImage(
2022-09-08 21:40:48 +02:00
title: string,
description: string,
blob: File
2023-09-28 23:50:27 +02:00
): Promise<{ key: string; value: string }> {
2022-09-08 21:40:48 +02:00
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)
2022-09-03 20:53:06 +02:00
const settings: RequestInit = {
2022-09-08 21:40:48 +02:00
method: "POST",
2022-09-03 20:53:06 +02:00
body: formData,
2022-09-08 21:40:48 +02:00
redirect: "follow",
2022-09-03 20:53:06 +02:00
headers: new Headers({
Authorization: `Client-ID ${apiKey}`,
2022-09-08 21:40:48 +02:00
Accept: "application/json",
2022-09-03 20:53:06 +02:00
}),
2022-09-08 21:40:48 +02:00
}
// Response contains stringified JSON
const response = await fetch(apiUrl, settings)
const content = await response.json()
return { key: "image", value: content.data.link }
}
SourceIcon(): BaseUIElement {
2022-09-08 21:40:48 +02:00
return undefined
}
2021-11-07 16:34:51 +01:00
public async ExtractUrls(key: string, value: string): Promise<Promise<ProvidedImage>[]> {
2022-09-08 21:40:48 +02:00
if (Imgur.defaultValuePrefix.some((prefix) => value.startsWith(prefix))) {
return [
Promise.resolve({
url: value,
key: key,
provider: this,
}),
]
2021-11-07 16:34:51 +01:00
}
return []
}
2022-06-13 00:51:53 +02:00
/**
2023-01-13 03:45:02 +01:00
* Download the attribution and license info for the picture at the given URL
2022-09-03 20:53:06 +02:00
*
2022-06-13 00:51:53 +02:00
* 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
*/
2022-09-08 21:40:48 +02:00
public async DownloadAttribution(url: string): Promise<LicenseInfo> {
const hash = url.substr("https://i.imgur.com/".length).split(".jpg")[0]
2022-09-08 21:40:48 +02:00
const apiUrl = "https://api.imgur.com/3/image/" + hash
const response = await Utils.downloadJsonCached(apiUrl, 365 * 24 * 60 * 60, {
Authorization: "Client-ID " + Constants.ImgurApiKey,
})
2022-09-08 21:40:48 +02:00
const descr: string = response.data.description ?? ""
const data: any = {}
for (const tag of descr.split("\n")) {
2022-09-08 21:40:48 +02:00
const kv = tag.split(":")
const k = kv[0]
data[k] = kv[1]?.replace(/\r/g, "")
}
2022-09-08 21:40:48 +02:00
const licenseInfo = new LicenseInfo()
2022-09-08 21:40:48 +02:00
licenseInfo.licenseShortName = data.license
licenseInfo.artist = data.author
return licenseInfo
}
2022-09-08 21:40:48 +02:00
}