radiquum.github.io--photos/generate/build.tsx

134 lines
3.4 KiB
TypeScript

import { initializeApp } from "firebase-admin/app";
import { getFirestore } from "firebase-admin/firestore";
import { cert } from "firebase-admin/app";
import { Log } from "./utils";
import { renderToString } from "react-dom/server";
import fs from "fs";
import exec from "child_process";
import Base from "./templates/Base";
import Header from "./templates/Header";
const log = new Log();
if (!process.env.FIREBASE_SERVICE_ACCOUNT) {
throw new Error("FIREBASE_SERVICE_ACCOUNT is not set");
}
if (!process.env.FIREBASE_COLLECTION) {
throw new Error("FIREBASE_COLLECTION is not set");
}
const ENVIRONMENT = process.env.ENVIRONMENT || "prod";
const serviceAccount = require(process.env.FIREBASE_SERVICE_ACCOUNT as string);
const app = initializeApp({
credential: cert(serviceAccount),
});
const db = getFirestore(app);
interface Url {
name: string;
value: string;
}
interface Image {
id: string;
alt: string;
tags: string[];
urls: Url[];
mimetype: string;
width: number;
height: number;
date: number;
}
type Years = Record<string, Image[]>;
let tags: string[] = [];
let items: Years = {};
function addTag(tag: string) {
if (tags.includes(tag)) {
return;
}
tags.push(tag);
}
if (
!fs.existsSync("data") ||
!fs.existsSync("data/tags.json") ||
!fs.existsSync("data/items.json")
) {
log.warn("data/tags.json or data/items.json does not exist");
await db
.collection(process.env.FIREBASE_COLLECTION as string)
.get()
.then((snapshot) => {
snapshot.forEach((doc) => {
const data = doc.data();
data.tags.forEach((tag: string) => {
addTag(tag);
});
const year: string = new Date(data.date).getFullYear().toString();
if (!(year in items)) {
items[year] = [];
}
items[year].push({
id: doc.id,
alt: data.alt,
tags: data.tags,
urls: data.urls,
mimetype: data.mimetype,
width: data.width,
height: data.height,
date: data.date,
});
});
});
if (!fs.existsSync("data")) fs.mkdirSync("data");
fs.writeFileSync("data/tags.json", JSON.stringify(tags));
fs.writeFileSync("data/items.json", JSON.stringify(items));
} else {
log.warn("using cached data");
tags = JSON.parse(fs.readFileSync("data/tags.json", "utf-8"));
items = JSON.parse(fs.readFileSync("data/items.json", "utf-8"));
}
const html = renderToString(
<Base isDev={ENVIRONMENT == "dev"}>
<Header />
</Base>
);
if (!fs.existsSync("out")) fs.mkdirSync("out");
if (!fs.existsSync("out/static")) fs.mkdirSync("out/static");
if (!fs.existsSync("out/static/js")) fs.mkdirSync("out/static/js");
if (!fs.existsSync("out/static/css")) fs.mkdirSync("out/static/css");
fs.cpSync("static", "out/static", { recursive: true });
if (ENVIRONMENT == "dev") {
fs.cpSync("static_dev/hotreload.js", "out/static/js/hotreload.js");
} else {
log.info("Minifying resources...");
exec.exec(
"bun run tailwindcss -i static_dev/input.css -o out/static/css/tailwind.css --build --minify",
(error, stdout, stderr) => {
if (error) {
log.error(error.message);
return;
}
if (stderr) {
log.error(stderr);
return;
}
log.info(stdout);
}
);
}
fs.writeFileSync("out/index.html", `<!DOCTYPE html />${html}`);
log.info("Build finished!");
// console.log(await db.collection("images").count().get().then((snapshot) => {
// return snapshot.data().count;
// }));