 ba368552f2
			
		
	
	ba368552f2
	
	
	
		
			
			* web: fix esbuild issue with style sheets
Getting ESBuild, Lit, and Storybook to all agree on how to read and parse stylesheets is a serious
pain. This fix better identifies the value types (instances) being passed from various sources in
the repo to the three *different* kinds of style processors we're using (the native one, the
polyfill one, and whatever the heck Storybook does internally).
Falling back to using older CSS instantiating techniques one era at a time seems to do the trick.
It's ugly, but in the face of the aggressive styling we use to avoid Flashes of Unstyled Content
(FLoUC), it's the logic with which we're left.
In standard mode, the following warning appears on the console when running a Flow:
```
Autofocus processing was blocked because a document already has a focused element.
```
In compatibility mode, the following **error** appears on the console when running a Flow:
```
crawler-inject.js:1106 Uncaught TypeError: Failed to execute 'observe' on 'MutationObserver': parameter 1 is not of type 'Node'.
    at initDomMutationObservers (crawler-inject.js:1106:18)
    at crawler-inject.js:1114:24
    at Array.forEach (<anonymous>)
    at initDomMutationObservers (crawler-inject.js:1114:10)
    at crawler-inject.js:1549:1
initDomMutationObservers @ crawler-inject.js:1106
(anonymous) @ crawler-inject.js:1114
initDomMutationObservers @ crawler-inject.js:1114
(anonymous) @ crawler-inject.js:1549
```
Despite this error, nothing seems to be broken and flows work as anticipated.
* web: always build sourcemaps
		
	
		
			
				
	
	
		
			152 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import * as chokidar from "chokidar";
 | |
| import esbuild from "esbuild";
 | |
| import fs from "fs";
 | |
| import { globSync } from "glob";
 | |
| import path from "path";
 | |
| import { cwd } from "process";
 | |
| import process from "process";
 | |
| import { fileURLToPath } from "url";
 | |
| 
 | |
| const __dirname = fileURLToPath(new URL(".", import.meta.url));
 | |
| 
 | |
| // eslint-disable-next-line no-undef
 | |
| const isProdBuild = process.env.NODE_ENV === "production";
 | |
| 
 | |
| // eslint-disable-next-line no-undef
 | |
| const apiBasePath = process.env.AK_API_BASE_PATH || "";
 | |
| 
 | |
| const definitions = {
 | |
|     "process.env.NODE_ENV": JSON.stringify(isProdBuild ? "production" : "development"),
 | |
|     "process.env.CWD": JSON.stringify(cwd()),
 | |
|     "process.env.AK_API_BASE_PATH": JSON.stringify(apiBasePath),
 | |
| };
 | |
| 
 | |
| // All is magic is just to make sure the assets are copied into the right places. This is a very stripped down version
 | |
| // of what the rollup-copy-plugin does, without any of the features we don't use, and using globSync instead of globby
 | |
| // since we already had globSync lying around thanks to Typescript. If there's a third argument in an array entry, it's
 | |
| // used to replace the internal path before concatenating it all together as the destination target.
 | |
| 
 | |
| const otherFiles = [
 | |
|     ["node_modules/@patternfly/patternfly/patternfly.min.css", "."],
 | |
|     ["node_modules/@patternfly/patternfly/assets/**", ".", "node_modules/@patternfly/patternfly/"],
 | |
|     ["src/custom.css", "."],
 | |
|     ["src/common/styles/**", "."],
 | |
|     ["src/assets/images/**", "./assets/images"],
 | |
|     ["./icons/*", "./assets/icons"],
 | |
| ];
 | |
| 
 | |
| const isFile = (filePath) => fs.statSync(filePath).isFile();
 | |
| function nameCopyTarget(src, dest, strip) {
 | |
|     const target = path.join(dest, strip ? src.replace(strip, "") : path.parse(src).base);
 | |
|     return [src, target];
 | |
| }
 | |
| 
 | |
| for (const [source, rawdest, strip] of otherFiles) {
 | |
|     const matchedPaths = globSync(source);
 | |
|     const dest = path.join("dist", rawdest);
 | |
|     const copyTargets = matchedPaths.map((path) => nameCopyTarget(path, dest, strip));
 | |
|     for (const [src, dest] of copyTargets) {
 | |
|         if (isFile(src)) {
 | |
|             fs.mkdirSync(path.dirname(dest), { recursive: true });
 | |
|             fs.copyFileSync(src, dest);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // This starts the definitions used for esbuild: Our targets, our arguments, the function for running a build, and three
 | |
| // options for building: watching, building, and building the proxy.
 | |
| // Ordered by largest to smallest interface to build even faster
 | |
| const interfaces = [
 | |
|     ["admin/AdminInterface/AdminInterface.ts", "admin"],
 | |
|     ["user/UserInterface.ts", "user"],
 | |
|     ["flow/FlowInterface.ts", "flow"],
 | |
|     ["standalone/api-browser/index.ts", "standalone/api-browser"],
 | |
|     ["enterprise/rac/index.ts", "enterprise/rac"],
 | |
|     ["standalone/loading/index.ts", "standalone/loading"],
 | |
|     ["polyfill/poly.ts", "."],
 | |
| ];
 | |
| 
 | |
| const baseArgs = {
 | |
|     bundle: true,
 | |
|     write: true,
 | |
|     sourcemap: true,
 | |
|     minify: isProdBuild,
 | |
|     splitting: true,
 | |
|     treeShaking: true,
 | |
|     external: ["*.woff", "*.woff2"],
 | |
|     tsconfig: "./tsconfig.json",
 | |
|     loader: { ".css": "text", ".md": "text" },
 | |
|     define: definitions,
 | |
|     format: "esm",
 | |
| };
 | |
| 
 | |
| async function buildOneSource(source, dest) {
 | |
|     const DIST = path.join(__dirname, "./dist", dest);
 | |
|     console.log(`[${new Date(Date.now()).toISOString()}] Starting build for target ${source}`);
 | |
| 
 | |
|     try {
 | |
|         const start = Date.now();
 | |
|         await esbuild.build({
 | |
|             ...baseArgs,
 | |
|             entryPoints: [`./src/${source}`],
 | |
|             outdir: DIST,
 | |
|         });
 | |
|         const end = Date.now();
 | |
|         console.log(
 | |
|             `[${new Date(end).toISOString()}] Finished build for target ${source} in ${
 | |
|                 Date.now() - start
 | |
|             }ms`,
 | |
|         );
 | |
|     } catch (exc) {
 | |
|         console.error(`[${new Date(Date.now()).toISOString()}] Failed to build ${source}: ${exc}`);
 | |
|     }
 | |
| }
 | |
| 
 | |
| async function buildAuthentik(interfaces) {
 | |
|     await Promise.allSettled(interfaces.map(([source, dest]) => buildOneSource(source, dest)));
 | |
| }
 | |
| 
 | |
| let timeoutId = null;
 | |
| function debouncedBuild() {
 | |
|     if (timeoutId !== null) {
 | |
|         clearTimeout(timeoutId);
 | |
|     }
 | |
|     timeoutId = setTimeout(() => {
 | |
|         console.clear();
 | |
|         buildAuthentik(interfaces);
 | |
|     }, 250);
 | |
| }
 | |
| 
 | |
| if (process.argv.length > 2 && (process.argv[2] === "-h" || process.argv[2] === "--help")) {
 | |
|     console.log(`Build the authentikUI
 | |
| 
 | |
| options:
 | |
|   -w, --watch: Build all ${interfaces.length} interfaces
 | |
|   -p, --proxy: Build only the polyfills and the loading application
 | |
|   -h, --help: This help message
 | |
| `);
 | |
|     process.exit(0);
 | |
| }
 | |
| 
 | |
| if (process.argv.length > 2 && (process.argv[2] === "-w" || process.argv[2] === "--watch")) {
 | |
|     console.log("Watching ./src for changes");
 | |
|     chokidar.watch("./src").on("all", (event, path) => {
 | |
|         if (!["add", "change", "unlink"].includes(event)) {
 | |
|             return;
 | |
|         }
 | |
|         if (!/(\.css|\.ts|\.js)$/.test(path)) {
 | |
|             return;
 | |
|         }
 | |
|         debouncedBuild();
 | |
|     });
 | |
| } else if (process.argv.length > 2 && (process.argv[2] === "-p" || process.argv[2] === "--proxy")) {
 | |
|     // There's no watch-for-proxy, sorry.
 | |
|     await buildAuthentik(
 | |
|         interfaces.filter(([_, dest]) => ["standalone/loading", "."].includes(dest)),
 | |
|     );
 | |
|     process.exit(0);
 | |
| } else {
 | |
|     // And the fallback: just build it.
 | |
|     await buildAuthentik(interfaces);
 | |
| }
 |