export interface MicropubChannel { readonly uid: string, readonly name: string } export interface MF2 { type: string[], properties: { [key:string]: (string | MF2 | {[key:string]: string})[] | undefined } } export interface MicropubConfig { readonly channels?: MicropubChannel[], readonly "media-endpoint"?: string } export interface MicropubErrorMessage { readonly error: string, readonly error_description: string | undefined } export class MicropubError extends Error { readonly status: number | null readonly response: MicropubErrorMessage | null constructor(status: number | null, response: MicropubErrorMessage | null, cause: Error | null = null) { // Needs to pass both `message` and `options` to install the "cause" property. if (status == null) { super("Micropub endpoint didn't respond properly", { cause }); } else if (response == null) { super(`Micropub endpoint returned HTTP ${status}`, { cause }); } else { super( `Micropub endpoint returned ${response.error}: ${response.error_description ?? "(no description was provided)"}`, { cause } ) } this.status = status; this.response = response; } } export class Micropub { readonly token: string readonly micropub_endpoint: URL private config_response: MicropubConfig | null constructor({ endpoint, token }: { endpoint: URL, token: string }) { this.micropub_endpoint = endpoint; this.token = token; this.config_response = null; } async config(): Promise<MicropubConfig> { if (this.config_response != null) { return this.config_response } let url = this.micropub_endpoint; let params = new URLSearchParams(); for (const [key, val] of url.searchParams) { params.append(key, val) } params.set("q", "config") url.search = "?" + params.toString(); const response = await fetch(url, { headers: { "Authorization": `Bearer ${this.token}` } }); if (response.ok) { const config = await response.json() as MicropubConfig; this.config_response = config return config } else { throw new MicropubError(response.status, await response.json() as MicropubErrorMessage); } } async submit(mf2: MF2): Promise<URL> { const response = await fetch(this.micropub_endpoint, { method: "POST", headers: { "Authorization": `Bearer ${this.token}`, "Content-Type": "application/json" }, body: JSON.stringify(mf2) }) if (response.status != 201 && response.status != 202) { let err = await response.json() as MicropubErrorMessage; throw new MicropubError(response.status, err) } else { return new URL(response.headers.get("Location") as string) } } async upload(file: File): Promise<URL> { const config = await this.config(); const media = config["media-endpoint"]; if (media == null) { throw new Error("Micropub endpoint doesn't support file uploads") } const form = new FormData(); form.set("file", file); const response = await fetch(media, { method: "POST", headers: { "Authorization": `Bearer ${this.token}`, }, body: form }) if (response.ok) { return new URL(response.headers.get("Location") as string) } else { throw new MicropubError(response.status, await response.json()); } } }