forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			168 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			168 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| 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<string[]> {
 | |
|         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}`;
 | |
|         }
 | |
|         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.debug(`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<LicenseInfo> {
 | |
|         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<ProvidedImage> {
 | |
|         if (!image.startsWith("File:")) {
 | |
|             image = "File:" + image
 | |
|         }
 | |
|         return {url: this.PrepareUrl(image), key: undefined, provider: this}
 | |
|     }
 | |
| 
 | |
|     public async ExtractUrls(key: string, value: string): Promise<Promise<ProvidedImage>[]> {
 | |
|         if(key !== undefined && 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)]
 | |
|     }
 | |
| 
 | |
| 
 | |
| }
 | |
| 
 |