120 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			120 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
// @ts-check
 | 
						|
 | 
						|
const fetch = require("node-fetch").default
 | 
						|
 | 
						|
const passthrough = require("../passthrough")
 | 
						|
const {sync, db, select} = passthrough
 | 
						|
/** @type {import("./mreq")} */
 | 
						|
const mreq = sync.require("./mreq")
 | 
						|
 | 
						|
const DISCORD_IMAGES_BASE = "https://cdn.discordapp.com"
 | 
						|
const IMAGE_SIZE = 1024
 | 
						|
 | 
						|
/** @type {Map<string, Promise<string>>} */
 | 
						|
const inflight = new Map()
 | 
						|
 | 
						|
/**
 | 
						|
 * @param {string} url
 | 
						|
 */
 | 
						|
function _removeExpiryParams(url) {
 | 
						|
	return url.replace(/\?(?:(?:ex|is|sg|hm)=[a-f0-9]+&?)*$/, "")
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * @param {string} path or full URL if it's not a Discord CDN file
 | 
						|
 */
 | 
						|
async function uploadDiscordFileToMxc(path) {
 | 
						|
	let url
 | 
						|
	if (path.startsWith("http")) {
 | 
						|
		url = path
 | 
						|
	} else {
 | 
						|
		url = DISCORD_IMAGES_BASE + path
 | 
						|
	}
 | 
						|
 | 
						|
	// Discord attachment content is always the same no matter what their ?ex parameter is.
 | 
						|
	const urlNoExpiry = _removeExpiryParams(url)
 | 
						|
 | 
						|
	// Are we uploading this file RIGHT NOW? Return the same inflight promise with the same resolution
 | 
						|
	const existingInflight = inflight.get(urlNoExpiry)
 | 
						|
	if (existingInflight) {
 | 
						|
		return existingInflight
 | 
						|
	}
 | 
						|
 | 
						|
	// Has this file already been uploaded in the past? Grab the existing copy from the database.
 | 
						|
	const existingFromDb = select("file", "mxc_url", {discord_url: urlNoExpiry}).pluck().get()
 | 
						|
	if (typeof existingFromDb === "string") {
 | 
						|
		return existingFromDb
 | 
						|
	}
 | 
						|
 | 
						|
	// Download from Discord
 | 
						|
	const promise = fetch(url, {}).then(/** @param {import("node-fetch").Response} res */ async res => {
 | 
						|
		// Upload to Matrix
 | 
						|
		const root = await module.exports._actuallyUploadDiscordFileToMxc(urlNoExpiry, res)
 | 
						|
 | 
						|
		// Store relationship in database
 | 
						|
		db.prepare("INSERT INTO file (discord_url, mxc_url) VALUES (?, ?)").run(urlNoExpiry, root.content_uri)
 | 
						|
		inflight.delete(urlNoExpiry)
 | 
						|
 | 
						|
		return root.content_uri
 | 
						|
	})
 | 
						|
	inflight.set(urlNoExpiry, promise)
 | 
						|
 | 
						|
	return promise
 | 
						|
}
 | 
						|
 | 
						|
async function _actuallyUploadDiscordFileToMxc(url, res) {
 | 
						|
	const body = res.body
 | 
						|
	/** @type {import("../types").R.FileUploaded} */
 | 
						|
	const root = await mreq.mreq("POST", "/media/v3/upload", body, {
 | 
						|
		headers: {
 | 
						|
			"Content-Type": res.headers.get("content-type")
 | 
						|
		}
 | 
						|
	})
 | 
						|
	return root
 | 
						|
}
 | 
						|
 | 
						|
function guildIcon(guild) {
 | 
						|
	return `/icons/${guild.id}/${guild.icon}.png?size=${IMAGE_SIZE}`
 | 
						|
}
 | 
						|
 | 
						|
function userAvatar(user) {
 | 
						|
	return `/avatars/${user.id}/${user.avatar}.png?size=${IMAGE_SIZE}`
 | 
						|
}
 | 
						|
 | 
						|
function memberAvatar(guildID, user, member) {
 | 
						|
	if (!member.avatar) return userAvatar(user)
 | 
						|
	return `/guilds/${guildID}/users/${user.id}/avatars/${member.avatar}.png?size=${IMAGE_SIZE}`
 | 
						|
}
 | 
						|
 | 
						|
function emoji(emojiID, animated) {
 | 
						|
	const base = `/emojis/${emojiID}`
 | 
						|
	if (animated) return base + ".gif"
 | 
						|
	else return base + ".png"
 | 
						|
}
 | 
						|
 | 
						|
const stickerFormat = new Map([
 | 
						|
	[1, {label: "PNG", ext: "png", mime: "image/png"}],
 | 
						|
	[2, {label: "APNG", ext: "png", mime: "image/apng"}],
 | 
						|
	[3, {label: "LOTTIE", ext: "json", mime: "lottie"}],
 | 
						|
	[4, {label: "GIF", ext: "gif", mime: "image/gif"}]
 | 
						|
])
 | 
						|
 | 
						|
/** @param {{id: string, format_type: number}} sticker */
 | 
						|
function sticker(sticker) {
 | 
						|
	const format = stickerFormat.get(sticker.format_type)
 | 
						|
	if (!format) throw new Error(`No such format ${sticker.format_type} for sticker ${JSON.stringify(sticker)}`)
 | 
						|
	const ext = format.ext
 | 
						|
	return `/stickers/${sticker.id}.${ext}`
 | 
						|
}
 | 
						|
 | 
						|
module.exports.DISCORD_IMAGES_BASE = DISCORD_IMAGES_BASE
 | 
						|
module.exports.guildIcon = guildIcon
 | 
						|
module.exports.userAvatar = userAvatar
 | 
						|
module.exports.memberAvatar = memberAvatar
 | 
						|
module.exports.emoji = emoji
 | 
						|
module.exports.stickerFormat = stickerFormat
 | 
						|
module.exports.sticker = sticker
 | 
						|
module.exports.uploadDiscordFileToMxc = uploadDiscordFileToMxc
 | 
						|
module.exports._actuallyUploadDiscordFileToMxc = _actuallyUploadDiscordFileToMxc
 | 
						|
module.exports._removeExpiryParams = _removeExpiryParams
 |