import ImageProvider, {ProvidedImage} from "./ImageProvider"; import BaseUIElement from "../../UI/BaseUIElement"; import Svg from "../../Svg"; import Link from "../../UI/Base/Link"; import {Utils} from "../../Utils"; import {LicenseInfo} from "./LicenseInfo"; /** * This module provides endpoints for wikimedia and others */ export class WikimediaImageProvider extends ImageProvider { private readonly commons_key = "wikimedia_commons" public readonly defaultKeyPrefixes = [this.commons_key,"image"] public static readonly singleton = new WikimediaImageProvider(); public static readonly commonsPrefix = "https://commons.wikimedia.org/wiki/" private constructor() { super(); } /** * Recursively walks a wikimedia commons category in order to search for (image) files * Returns (a promise of) a list of URLS * @param categoryName The name of the wikimedia category * @param maxLoad: the maximum amount of images to return * @param continueParameter: if the page indicates that more pages should be loaded, this uses a token to continue. Provided by wikimedia */ private static async GetImagesInCategory(categoryName: string, maxLoad = 10, continueParameter: string = undefined): Promise { if (categoryName === undefined || categoryName === null || categoryName === "") { return []; } if (!categoryName.startsWith("Category:")) { categoryName = "Category:" + categoryName; } let url = "https://commons.wikimedia.org/w/api.php?" + "action=query&list=categorymembers&format=json&" + "&origin=*" + "&cmtitle=" + encodeURIComponent(categoryName); if (continueParameter !== undefined) { url = `${url}&cmcontinue=${continueParameter}`; } console.log("Loading a wikimedia category: ", url) const response = await Utils.downloadJson(url) const members = response.query?.categorymembers ?? []; const imageOverview: string[] = members.map(member => member.title); if (response.continue === undefined) { // We are done crawling through the category - no continuation in sight return imageOverview; } if (maxLoad - imageOverview.length <= 0) { console.log(`Recursive wikimedia category load stopped for ${categoryName}`) return imageOverview; } // We do have a continue token - let's load the next page const recursive = await this.GetImagesInCategory(categoryName, maxLoad - imageOverview.length, response.continue.cmcontinue) imageOverview.push(...recursive) return imageOverview } private static ExtractFileName(url: string) { if (!url.startsWith("http")) { return url; } const path = new URL(url).pathname return path.substring(path.lastIndexOf("/") + 1); } SourceIcon(backlink: string): BaseUIElement { const img = Svg.wikimedia_commons_white_svg() .SetStyle("width:2em;height: 2em"); if (backlink === undefined) { return img } return new Link(Svg.wikimedia_commons_white_img, `https://commons.wikimedia.org/wiki/${backlink}`, true) } private PrepareUrl(value: string): string { if (value.toLowerCase().startsWith("https://commons.wikimedia.org/wiki/")) { return value; } return (`https://commons.wikimedia.org/wiki/Special:FilePath/${encodeURIComponent(value)}?width=500&height=400`) } protected async DownloadAttribution(filename: string): Promise { filename = WikimediaImageProvider.ExtractFileName(filename) if (filename === "") { return undefined; } const url = "https://en.wikipedia.org/w/" + "api.php?action=query&prop=imageinfo&iiprop=extmetadata&" + "titles=" + filename + "&format=json&origin=*"; const data = await Utils.downloadJson(url) const licenseInfo = new LicenseInfo(); const license = (data.query.pages[-1].imageinfo ?? [])[0]?.extmetadata; if (license === undefined) { console.error("This file has no usable metedata or license attached... Please fix the license info file yourself!") return undefined; } licenseInfo.artist = license.Artist?.value; licenseInfo.license = license.License?.value; licenseInfo.copyrighted = license.Copyrighted?.value; licenseInfo.attributionRequired = license.AttributionRequired?.value; licenseInfo.usageTerms = license.UsageTerms?.value; licenseInfo.licenseShortName = license.LicenseShortName?.value; licenseInfo.credit = license.Credit?.value; licenseInfo.description = license.ImageDescription?.value; return licenseInfo; } private async UrlForImage(image: string): Promise { if (!image.startsWith("File:")) { image = "File:" + image } return {url: this.PrepareUrl(image), key: undefined, provider: this} } public async ExtractUrls(key: string, value: string): Promise[]> { if(key !== this.commons_key && !value.startsWith(WikimediaImageProvider.commonsPrefix)){ return [] } if (value.startsWith(WikimediaImageProvider.commonsPrefix)) { value = value.substring(WikimediaImageProvider.commonsPrefix.length) } else if (value.startsWith("https://upload.wikimedia.org")) { const result: ProvidedImage = { key: undefined, url: value, provider: this } return [Promise.resolve(result)] } if (value.startsWith("Category:")) { const urls = await WikimediaImageProvider.GetImagesInCategory(value) return urls.map(image => this.UrlForImage(image)) } if (value.startsWith("File:")) { return [this.UrlForImage(value)] } if (value.startsWith("http")) { // PRobably an error return [] } // We do a last effort and assume this is a file return [this.UrlForImage("File:" + value)] } }