import {LitElement, html} from './lit-all.min.js';
import * as tfrpc from '/static/tfrpc.js';

class TfSneakerAppElement extends LitElement {
	static get properties() {
		return {
			feeds: {type: Object},
			progress: {type: Object},
			result: {type: String},
		};
	}

	constructor() {
		super();
		this.feeds = [];
		this.progress = undefined;
		this.result = undefined;
	}

	async search() {
		let q = this.renderRoot.getElementById('search').value;
		let result = await tfrpc.rpc.query(
			`
			SELECT messages.author AS id, json_extract(messages.content, '$.name') AS name
			FROM messages_fts(?)
			JOIN messages ON messages.rowid = messages_fts.rowid
			WHERE
				json_extract(messages.content, '$.type') = 'about' AND
				json_extract(messages.content, '$.about') = messages.author AND
				json_extract(messages.content, '$.name') IS NOT NULL
			GROUP BY messages.author
			HAVING MAX(messages.sequence)
			ORDER BY COUNT(*) DESC
			`,
			[`"${q.replaceAll('"', '""')}"`]
		);
		this.feeds = Object.fromEntries(result.map((x) => [x.id, x.name]));
	}

	format_message(message) {
		const k_flag_sequence_before_author = 1;
		let out = {
			previous: message.previous ?? null,
		};
		if (message.flags & k_flag_sequence_before_author) {
			out.sequence = message.sequence;
			out.author = message.author;
		} else {
			out.author = message.author;
			out.sequence = message.sequence;
		}
		out.timestamp = message.timestamp;
		out.hash = message.hash;
		out.content = JSON.parse(message.content);
		out.signature = message.signature;
		return {key: message.id, value: out};
	}

	sanitize(value) {
		return value.replaceAll('/', '_').replaceAll('+', '-');
	}

	guess_ext(data) {
		function startsWith(prefix) {
			if (data.length < prefix.length) {
				return false;
			}
			for (let i = 0; i < prefix.length; i++) {
				if (prefix[i] !== null && data[i] !== prefix[i]) {
					return false;
				}
			}
			return true;
		}

		if (
			startsWith(data, [0xff, 0xd8, 0xff, 0xdb]) ||
			startsWith(
				data,
				[0xff, 0xd8, 0xff, 0xe0, 0x00, 0x10, 0x4a, 0x46, 0x49, 0x46, 0x00, 0x01]
			) ||
			startsWith(data, [0xff, 0xd8, 0xff, 0xee]) ||
			startsWith(data, [
				0xff,
				0xd8,
				0xff,
				0xe1,
				null,
				null,
				0x45,
				0x78,
				0x69,
				0x66,
				0x00,
				0x00,
			])
		) {
			return '.jpg';
		} else if (
			startsWith(data, [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])
		) {
			return '.png';
		} else if (
			startsWith(data, [0x47, 0x49, 0x46, 0x38, 0x37, 0x61]) ||
			startsWith(data, [0x47, 0x49, 0x46, 0x38, 0x39, 0x61])
		) {
			return '.gif';
		} else if (
			startsWith(data, [
				0x52,
				0x49,
				0x46,
				0x46,
				null,
				null,
				null,
				null,
				0x57,
				0x45,
				0x42,
				0x50,
			])
		) {
			return '.webp';
		} else if (startsWith(data, [0x3c, 0x73, 0x76, 0x67])) {
			return '.svg';
		} else if (
			startsWith(data, [
				null,
				null,
				null,
				null,
				0x66,
				0x74,
				0x79,
				0x70,
				0x6d,
				0x70,
				0x34,
				0x32,
			])
		) {
			return '.mp3';
		} else if (
			startsWith(data, [
				null,
				null,
				null,
				null,
				0x66,
				0x74,
				0x79,
				0x70,
				0x69,
				0x73,
				0x6f,
				0x6d,
			]) ||
			startsWith(data, [
				null,
				null,
				null,
				null,
				0x66,
				0x74,
				0x79,
				0x70,
				0x6d,
				0x70,
				0x34,
				0x32,
			])
		) {
			return '.mp4';
		} else {
			return '.bin';
		}
	}

	async export(id) {
		let all_messages = '';
		let sequence = -1;
		let messages_done = 0;
		let messages_max = (
			await tfrpc.rpc.query(
				'SELECT MAX(sequence) AS total FROM messages WHERE author = ?',
				[id]
			)
		)[0].total;
		while (true) {
			let messages = await tfrpc.rpc.query(
				`
				SELECT author, id, sequence, timestamp, hash, json(content) AS content, signature, flags
				FROM messages
				WHERE author = ? AND SEQUENCE > ?
				ORDER BY sequence LIMIT 100
				`,
				[id, sequence]
			);
			if (messages?.length) {
				all_messages +=
					messages
						.map((x) => JSON.stringify(this.format_message(x)))
						.join('\n') + '\n';
				sequence = messages[messages.length - 1].sequence;
				messages_done += messages.length;
				this.progress = {
					name: 'messages',
					value: messages_done,
					max: messages_max,
				};
			} else {
				break;
			}
		}

		let zip = new JSZip();
		zip.file(`message/classic/${this.sanitize(id)}.ndjson`, all_messages);

		let blobs = await tfrpc.rpc.query(
			`SELECT messages_refs.ref AS id
			FROM messages
			JOIN messages_refs ON messages.id = messages_refs.message
			WHERE messages.author = ? AND messages_refs.ref LIKE '&%.sha256'`,
			[id]
		);
		let blobs_done = 0;
		for (let row of blobs) {
			this.progress = {name: 'blobs', value: blobs_done, max: blobs.length};
			let blob;
			try {
				blob = await tfrpc.rpc.get_blob(row.id);
			} catch (e) {
				console.log(`Failed to get ${row.id}: ${e.message}`);
			}
			if (blob) {
				zip.file(
					`blob/classic/${this.sanitize(row.id)}${this.guess_ext(blob)}`,
					new Uint8Array(blob)
				);
			}
			blobs_done++;
		}

		this.progress = {name: 'saving'};
		let blob = await zip.generateAsync({type: 'blob'});
		saveAs(blob, `${this.sanitize(id)}.zip`);
		this.progress = null;
	}

	keypress(event) {
		if (event.key == 'Enter') {
			this.search();
		}
	}

	async import(event) {
		let file = event.target.files[0];
		if (!file) {
			return;
		}

		this.progress = {name: 'loading'};
		let zip = new JSZip();
		file = await zip.loadAsync(file);
		let messages = [];
		let blobs = [];
		file.forEach(function (path, entry) {
			if (!entry.dir) {
				if (path.startsWith('message/classic/')) {
					messages.push(entry);
				} else {
					blobs.push(entry);
				}
			}
		});
		let success = {messages: 0, blobs: 0};
		let progress = 0;
		let total_messages = 0;
		for (let entry of messages) {
			let lines = (await entry.async('string')).split('\n');
			total_messages += lines.length;
			for (let line of lines) {
				if (!line.length) {
					continue;
				}
				let message = JSON.parse(line);
				this.progress = {
					name: 'messages',
					value: progress++,
					max: total_messages,
				};
				if (await tfrpc.rpc.store_message(message.value)) {
					success.messages++;
				}
			}
		}
		progress = 0;
		for (let blob of blobs) {
			this.progress = {name: 'blobs', value: progress++, max: blobs.length};
			if (await tfrpc.rpc.store_blob(await blob.async('arraybuffer'))) {
				success.blobs++;
			}
		}
		this.progress = undefined;
		this.result = `imported ${success.messages} messages and ${success.blobs} blobs`;
	}

	render() {
		let progress;
		if (this.progress) {
			if (this.progress.max) {
				progress = html`<div>
					<label for="progress">${this.progress.name}</label
					><progress
						value=${this.progress.value}
						max=${this.progress.max}
					></progress>
				</div>`;
			} else {
				progress = html`<div><span>${this.progress.name}</span></div>`;
			}
		}
		return html`<h1>SSB 👟net</h1>
			<code>${this.result}</code>
			${progress}

			<h2>Import</h2>
			<input type="file" id="import" @change=${this.import}></input>

			<h2>Export</h2>
			<input type="text" id="search" @keypress=${this.keypress}></input>
			<input type="button" value="Search Users" @click=${this.search}></input>
			<ul>
				${Object.entries(this.feeds).map(
					([id, name]) => html`
						<li>
							${this.progress
								? undefined
								: html`<input type="button" value="Export" @click=${() => this.export(id)}></input>`}
							${name}
							<code style="color: #ccc">${id}</code>
						</li>
					`
				)}
			</ul>
		`;
	}
}
customElements.define('tf-sneaker-app', TfSneakerAppElement);