mirror of
				https://github.com/monkeytypegame/monkeytype.git
				synced 2025-10-25 07:17:23 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			310 lines
		
	
	
	
		
			9.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			310 lines
		
	
	
	
		
			9.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import { fontawesomeSubset } from "fontawesome-subset";
 | |
| import { getFontawesomeConfig } from "./scripts/fontawesome";
 | |
| import { generatePreviewFonts } from "./scripts/font-preview";
 | |
| import { VitePWA } from "vite-plugin-pwa";
 | |
| import replace from "vite-plugin-filter-replace";
 | |
| import path from "node:path";
 | |
| import { splitVendorChunkPlugin } from "vite";
 | |
| import childProcess from "child_process";
 | |
| import { checker } from "vite-plugin-checker";
 | |
| import { writeFileSync } from "fs";
 | |
| // eslint-disable-next-line import/no-unresolved
 | |
| import UnpluginInjectPreload from "unplugin-inject-preload/vite";
 | |
| import { readdirSync, readFileSync, statSync } from "node:fs";
 | |
| import { ViteMinifyPlugin } from "vite-plugin-minify";
 | |
| import { sentryVitePlugin } from "@sentry/vite-plugin";
 | |
| 
 | |
| function pad(numbers, maxLength, fillString) {
 | |
|   return numbers.map((number) =>
 | |
|     number.toString().padStart(maxLength, fillString)
 | |
|   );
 | |
| }
 | |
| 
 | |
| const CLIENT_VERSION = (() => {
 | |
|   const date = new Date();
 | |
|   const versionPrefix = pad(
 | |
|     [date.getFullYear(), date.getMonth() + 1, date.getDate()],
 | |
|     2,
 | |
|     "0"
 | |
|   ).join(".");
 | |
|   const versionSuffix = pad([date.getHours(), date.getMinutes()], 2, "0").join(
 | |
|     "."
 | |
|   );
 | |
|   const version = [versionPrefix, versionSuffix].join("_");
 | |
| 
 | |
|   try {
 | |
|     const commitHash = childProcess
 | |
|       .execSync("git rev-parse --short HEAD")
 | |
|       .toString();
 | |
| 
 | |
|     return `${version}_${commitHash}`.replace(/\n/g, "");
 | |
|   } catch (e) {
 | |
|     return `${version}_unknown-hash`;
 | |
|   }
 | |
| })();
 | |
| 
 | |
| /** Enable for font awesome v6 */
 | |
| /*
 | |
| function sassList(values) {
 | |
|   return values.map((it) => `"${it}"`).join(",");
 | |
| }
 | |
| */
 | |
| 
 | |
| /** @type {import("vite").UserConfig} */
 | |
| export default {
 | |
|   plugins: [
 | |
|     {
 | |
|       name: "vite-plugin-fontawesome-subset",
 | |
|       apply: "build",
 | |
|       buildStart() {
 | |
|         const fontawesomeClasses = getFontawesomeConfig();
 | |
|         fontawesomeSubset(fontawesomeClasses, "src/webfonts-generated", {
 | |
|           targetFormats: ["woff2"],
 | |
|         });
 | |
|       },
 | |
|     },
 | |
|     {
 | |
|       name: "generate-version-json",
 | |
|       apply: "build",
 | |
| 
 | |
|       closeBundle() {
 | |
|         const version = CLIENT_VERSION;
 | |
|         const versionJson = JSON.stringify({ version });
 | |
|         const versionPath = path.resolve(__dirname, "dist/version.json");
 | |
|         writeFileSync(versionPath, versionJson);
 | |
|       },
 | |
|     },
 | |
|     {
 | |
|       name: "vite-plugin-webfonts-preview",
 | |
|       apply: "build",
 | |
|       buildStart() {
 | |
|         generatePreviewFonts();
 | |
|       },
 | |
|     },
 | |
|     checker({
 | |
|       typescript: {
 | |
|         tsconfigPath: path.resolve(__dirname, "./tsconfig.json"),
 | |
|       },
 | |
|     }),
 | |
|     splitVendorChunkPlugin(),
 | |
|     ViteMinifyPlugin({}),
 | |
|     VitePWA({
 | |
|       // injectRegister: "networkfirst",
 | |
|       injectRegister: null,
 | |
|       registerType: "autoUpdate",
 | |
|       manifest: {
 | |
|         short_name: "Monkeytype",
 | |
|         name: "Monkeytype",
 | |
|         start_url: "/",
 | |
|         icons: [
 | |
|           {
 | |
|             src: "/images/icons/maskable_icon_x512.png",
 | |
|             sizes: "512x512",
 | |
|             type: "image/png",
 | |
|             purpose: "maskable",
 | |
|           },
 | |
|           {
 | |
|             src: "/images/icons/general_icon_x512.png",
 | |
|             sizes: "512x512",
 | |
|             type: "image/png",
 | |
|             purpose: "any",
 | |
|           },
 | |
|         ],
 | |
|         background_color: "#323437",
 | |
|         display: "standalone",
 | |
|         theme_color: "#323437",
 | |
|       },
 | |
|       manifestFilename: "manifest.json",
 | |
|       workbox: {
 | |
|         clientsClaim: true,
 | |
|         cleanupOutdatedCaches: true,
 | |
|         globIgnores: ["**/.*"],
 | |
|         globPatterns: [],
 | |
|         navigateFallback: "",
 | |
|         runtimeCaching: [
 | |
|           {
 | |
|             urlPattern: (options) => {
 | |
|               const isApi = options.url.hostname === "api.monkeytype.com";
 | |
|               return options.sameOrigin && !isApi;
 | |
|             },
 | |
|             handler: "NetworkFirst",
 | |
|             options: {},
 | |
|           },
 | |
|           {
 | |
|             urlPattern: (options) => {
 | |
|               //disable caching for version.json
 | |
|               return options.url.pathname === "/version.json";
 | |
|             },
 | |
|             handler: "NetworkOnly",
 | |
|             options: {},
 | |
|           },
 | |
|         ],
 | |
|       },
 | |
|     }),
 | |
|     process.env.SENTRY
 | |
|       ? sentryVitePlugin({
 | |
|           authToken: process.env.SENTRY_AUTH_TOKEN,
 | |
|           org: "monkeytype",
 | |
|           project: "frontend",
 | |
|           release: {
 | |
|             name: CLIENT_VERSION,
 | |
|           },
 | |
|           applicationKey: "monkeytype-frontend",
 | |
|         })
 | |
|       : null,
 | |
|     replace([
 | |
|       {
 | |
|         filter: /firebase\.ts$/,
 | |
|         replace: {
 | |
|           from: /\.\/constants\/firebase-config/gi,
 | |
|           to: "./constants/firebase-config-live",
 | |
|         },
 | |
|       },
 | |
|     ]),
 | |
|     UnpluginInjectPreload({
 | |
|       files: [
 | |
|         {
 | |
|           outputMatch: /css\/vendor.*\.css$/,
 | |
|           attributes: {
 | |
|             as: "style",
 | |
|             type: "text/css",
 | |
|             rel: "preload",
 | |
|             crossorigin: true,
 | |
|           },
 | |
|         },
 | |
|         {
 | |
|           outputMatch: /.*\.woff2$/,
 | |
|           attributes: {
 | |
|             as: "font",
 | |
|             type: "font/woff2",
 | |
|             rel: "preload",
 | |
|             crossorigin: true,
 | |
|           },
 | |
|         },
 | |
|       ],
 | |
|       injectTo: "head-prepend",
 | |
|     }),
 | |
|     {
 | |
|       name: "minify-json",
 | |
|       apply: "build",
 | |
|       generateBundle() {
 | |
|         let totalOriginalSize = 0;
 | |
|         let totalMinifiedSize = 0;
 | |
| 
 | |
|         const minifyJsonFiles = (dir) => {
 | |
|           readdirSync(dir).forEach((file) => {
 | |
|             const sourcePath = path.join(dir, file);
 | |
|             const stat = statSync(sourcePath);
 | |
| 
 | |
|             if (stat.isDirectory()) {
 | |
|               minifyJsonFiles(sourcePath);
 | |
|             } else if (path.extname(file) === ".json") {
 | |
|               const originalContent = readFileSync(sourcePath, "utf8");
 | |
|               const originalSize = Buffer.byteLength(originalContent, "utf8");
 | |
|               const minifiedContent = JSON.stringify(
 | |
|                 JSON.parse(originalContent)
 | |
|               );
 | |
|               const minifiedSize = Buffer.byteLength(minifiedContent, "utf8");
 | |
| 
 | |
|               totalOriginalSize += originalSize;
 | |
|               totalMinifiedSize += minifiedSize;
 | |
| 
 | |
|               writeFileSync(sourcePath, minifiedContent);
 | |
| 
 | |
|               // const savings =
 | |
|               //   ((originalSize - minifiedSize) / originalSize) * 100;
 | |
|               // console.log(
 | |
|               //   `\x1b[0m \x1b[36m${sourcePath}\x1b[0m | ` +
 | |
|               //     `\x1b[90mOriginal: ${originalSize} bytes\x1b[0m | ` +
 | |
|               //     `\x1b[90mMinified: ${minifiedSize} bytes\x1b[0m | ` +
 | |
|               //     `\x1b[32mSavings: ${savings.toFixed(2)}%\x1b[0m`
 | |
|               // );
 | |
|             }
 | |
|           });
 | |
|         };
 | |
| 
 | |
|         // console.log("\n\x1b[1mMinifying JSON files...\x1b[0m\n");
 | |
| 
 | |
|         minifyJsonFiles("./dist");
 | |
| 
 | |
|         const totalSavings =
 | |
|           ((totalOriginalSize - totalMinifiedSize) / totalOriginalSize) * 100;
 | |
| 
 | |
|         console.log(
 | |
|           `\n\n\x1b[1mJSON Minification Summary:\x1b[0m\n` +
 | |
|             `  \x1b[90mTotal original size: ${(
 | |
|               totalOriginalSize /
 | |
|               1024 /
 | |
|               1024
 | |
|             ).toFixed(2)} mB\x1b[0m\n` +
 | |
|             `  \x1b[90mTotal minified size: ${(
 | |
|               totalMinifiedSize /
 | |
|               1024 /
 | |
|               1024
 | |
|             ).toFixed(2)} mB\x1b[0m\n` +
 | |
|             `  \x1b[32mTotal savings: ${totalSavings.toFixed(2)}%\x1b[0m\n`
 | |
|         );
 | |
|       },
 | |
|     },
 | |
|   ],
 | |
|   build: {
 | |
|     sourcemap: process.env.SENTRY,
 | |
|     emptyOutDir: true,
 | |
|     outDir: "../dist",
 | |
|     assetsInlineLimit: 0, //dont inline small files as data
 | |
|     rollupOptions: {
 | |
|       input: {
 | |
|         monkeytype: path.resolve(__dirname, "src/index.html"),
 | |
|         email: path.resolve(__dirname, "src/email-handler.html"),
 | |
|         privacy: path.resolve(__dirname, "src/privacy-policy.html"),
 | |
|         security: path.resolve(__dirname, "src/security-policy.html"),
 | |
|         terms: path.resolve(__dirname, "src/terms-of-service.html"),
 | |
|         404: path.resolve(__dirname, "src/404.html"),
 | |
|       },
 | |
|       output: {
 | |
|         assetFileNames: (assetInfo) => {
 | |
|           let extType = assetInfo.name.split(".").at(1);
 | |
|           if (/png|jpe?g|svg|gif|tiff|bmp|ico/i.test(extType)) {
 | |
|             extType = "images";
 | |
|           }
 | |
|           if (/\.(woff|woff2|eot|ttf|otf)$/.test(assetInfo.name)) {
 | |
|             return `webfonts/[name]-[hash].${extType}`;
 | |
|           }
 | |
|           return `${extType}/[name].[hash][extname]`;
 | |
|         },
 | |
|         chunkFileNames: "js/[name].[hash].js",
 | |
|         entryFileNames: "js/[name].[hash].js",
 | |
|       },
 | |
|     },
 | |
|   },
 | |
|   define: {
 | |
|     BACKEND_URL: JSON.stringify(
 | |
|       process.env.BACKEND_URL || "https://api.monkeytype.com"
 | |
|     ),
 | |
|     IS_DEVELOPMENT: JSON.stringify(false),
 | |
|     CLIENT_VERSION: JSON.stringify(CLIENT_VERSION),
 | |
|     RECAPTCHA_SITE_KEY: JSON.stringify(process.env.RECAPTCHA_SITE_KEY),
 | |
|     QUICK_LOGIN_EMAIL: undefined,
 | |
|     QUICK_LOGIN_PASSWORD: undefined,
 | |
|   },
 | |
|   /** Enable for font awesome v6 */
 | |
|   /*preprocessorOptions: {
 | |
|     scss: {
 | |
|       additionalData(source, fp) {
 | |
|         if (fp.endsWith("index.scss")) {
 | |
|           const fontawesomeClasses = getFontawesomeConfig();
 | |
|           return `
 | |
|           //inject variables into sass context
 | |
|           $fontawesomeBrands: ${sassList(
 | |
|             fontawesomeClasses.brands
 | |
|           )};             
 | |
|           $fontawesomeSolid: ${sassList(fontawesomeClasses.solid)};
 | |
| 
 | |
|           ${source}`;
 | |
|         } else {
 | |
|           return source;
 | |
|         }
 | |
|       },
 | |
|     },
 | |
|   },*/
 | |
| };
 |