Improve readability and usability

This commit is contained in:
Antoine Bartuccio 2024-11-21 00:33:40 +01:00
parent 8fc1a754de
commit 0739ce2fb4
4 changed files with 110 additions and 71 deletions

View File

@ -3,8 +3,9 @@ from pathlib import Path
from django.contrib.staticfiles.apps import StaticFilesConfig from django.contrib.staticfiles.apps import StaticFilesConfig
GENERATED_ROOT = Path(__file__).parent.resolve() / "generated" GENERATED_ROOT = Path(__file__).parent.resolve() / "generated"
BUNDLED_ROOT = GENERATED_ROOT / "bundled" BUNDLED_FOLDER_NAME = "bundled"
IGNORE_PATTERNS_BUNDLED = ["bundled/*"] BUNDLED_ROOT = GENERATED_ROOT / BUNDLED_FOLDER_NAME
IGNORE_PATTERNS_BUNDLED = [f"{BUNDLED_FOLDER_NAME}/*"]
IGNORE_PATTERNS_SCSS = ["*.scss"] IGNORE_PATTERNS_SCSS = ["*.scss"]
IGNORE_PATTERNS_TYPESCRIPT = ["*.ts"] IGNORE_PATTERNS_TYPESCRIPT = ["*.ts"]
IGNORE_PATTERNS = [ IGNORE_PATTERNS = [

View File

@ -12,28 +12,64 @@ import sass
from django.conf import settings from django.conf import settings
from sith.urls import api from sith.urls import api
from staticfiles.apps import BUNDLED_ROOT, GENERATED_ROOT from staticfiles.apps import BUNDLED_FOLDER_NAME, BUNDLED_ROOT, GENERATED_ROOT
@dataclass @dataclass
class JsBundlerManifestEntry: class JsBundlerManifestEntry:
out: str
src: str src: str
out: str
@classmethod @classmethod
def from_json_entry(cls, entry: dict[str, any]) -> list[Self]: def from_json_entry(cls, entry: dict[str, any]) -> list[Self]:
# We have two parts for a manifest entry
# The `src` element which is what the user asks django as a static
# The `out` element which is it's real name in the output static folder
# For the src part:
# The manifest file contains the path of the file relative to the project root
# We want the relative path of the file inside their respective static folder
# because that's what the user types when importing statics and that's what django gives us
# This is really similar to what we are doing in the bundler, it uses a similar algorithm
# Example:
# core/static/bundled/alpine-index.js -> bundled/alpine-index.js
# core/static/bundled/components/include-index.ts -> core/static/bundled/components/include-index.ts
def get_relative_src_name(name: str) -> str:
original_path = Path(name)
relative_path: list[str] = []
for directory in reversed(original_path.parts):
relative_path.append(directory)
# Contrary to the bundler algorithm, we do want to keep the bundled prefix
if directory == BUNDLED_FOLDER_NAME:
break
return str(Path(*reversed(relative_path)))
# For the out part:
# The bundler is configured to output files in generated/bundled and considers this folders as it's root
# Thus, the output name doesn't contain the `bundled` prefix that we need, we add it ourselves
ret = [ ret = [
cls( cls(
out=str(Path("bundled") / entry["file"]), src=get_relative_src_name(entry["src"]),
src=str(Path(*Path(entry["src"]).parts[2:])), out=str(Path(BUNDLED_FOLDER_NAME) / entry["file"]),
) )
] ]
def remove_hash(path: Path) -> str:
# Hashes are configured to be surrounded by `.`
# Filenames are like this path/to/file.hash.ext
unhashed = ".".join(path.stem.split(".")[:-1])
return str(path.with_stem(unhashed))
# CSS files generated by entrypoints don't have their own entry in the manifest
# They are however listed as an attribute of the entry point that generates them
# Their listed name is the one that has been generated inside the generated/bundled folder
# We prefix it with `bundled` and then generate an `src` name by removing the hash
for css in entry.get("css", []): for css in entry.get("css", []):
path = Path("bundled") / css path = Path(BUNDLED_FOLDER_NAME) / css
ret.append( ret.append(
cls( cls(
src=remove_hash(path),
out=str(path), out=str(path),
src=str(path.with_stem(entry["name"])),
) )
) )
return ret return ret
@ -77,7 +113,7 @@ class JSBundler:
def is_in_bundle(name: str | None) -> bool: def is_in_bundle(name: str | None) -> bool:
if name is None: if name is None:
return False return False
return name.startswith("bundled/") return Path(name).parts[0] == BUNDLED_FOLDER_NAME
class Scss: class Scss:
@ -120,7 +156,7 @@ class JS:
p p
for p in settings.STATIC_ROOT.rglob("*.js") for p in settings.STATIC_ROOT.rglob("*.js")
if ".min" not in p.suffixes if ".min" not in p.suffixes
and (settings.STATIC_ROOT / "bundled") not in p.parents and (settings.STATIC_ROOT / BUNDLED_FOLDER_NAME) not in p.parents
] ]
for path in to_exec: for path in to_exec:
p = path.resolve() p = path.resolve()

View File

@ -16,9 +16,10 @@ class ManifestPostProcessingStorage(ManifestStaticFilesStorage):
# This name swap has to be done here # This name swap has to be done here
# Otherwise, the manifest isn't aware of the file and can't work properly # Otherwise, the manifest isn't aware of the file and can't work properly
if settings.DEBUG: if settings.DEBUG:
# In production, the bundler manifest is used at compile time, we don't need to convert anything
try: try:
manifest = JSBundler.get_manifest() manifest = JSBundler.get_manifest()
except Exception as e: except FileNotFoundError as e:
raise Exception( raise Exception(
"Error loading manifest file, the bundler seems to be busy" "Error loading manifest file, the bundler seems to be busy"
) from e ) from e
@ -27,7 +28,6 @@ class ManifestPostProcessingStorage(ManifestStaticFilesStorage):
name = converted name = converted
path = Path(name) path = Path(name)
# Call bundler manifest
if path.suffix == ".scss": if path.suffix == ".scss":
# Compile scss files automatically in debug mode # Compile scss files automatically in debug mode
if settings.DEBUG: if settings.DEBUG:

View File

@ -2,7 +2,7 @@
import { parse, resolve } from "node:path"; import { parse, resolve } from "node:path";
import inject from "@rollup/plugin-inject"; import inject from "@rollup/plugin-inject";
import { glob } from "glob"; import { glob } from "glob";
import type { AliasOptions, UserConfig } from "vite"; import { type AliasOptions, type UserConfig, defineConfig } from "vite";
import type { Rollup } from "vite"; import type { Rollup } from "vite";
import { viteStaticCopy } from "vite-plugin-static-copy"; import { viteStaticCopy } from "vite-plugin-static-copy";
import tsconfig from "./tsconfig.json"; import tsconfig from "./tsconfig.json";
@ -44,65 +44,67 @@ function getRelativeAssetPath(path: string): string {
} }
// biome-ignore lint/style/noDefaultExport: this is recommended by documentation // biome-ignore lint/style/noDefaultExport: this is recommended by documentation
export default { export default defineConfig((config: UserConfig) => {
base: "/static/bundled/", return {
appType: "custom", base: "/static/bundled/",
build: { appType: "custom",
outDir: outDir, build: {
manifest: true, // goes into .vite/manifest.json in the build folder outDir: outDir,
modulePreload: false, // would require `import 'vite/modulepreload-polyfill'` to always be injected manifest: true, // goes into .vite/manifest.json in the build folder
emptyOutDir: true, modulePreload: false, // would require `import 'vite/modulepreload-polyfill'` to always be injected
rollupOptions: { emptyOutDir: config.mode === "production", // Avoid rebuilding everything in dev mode
input: collectedFiles, rollupOptions: {
output: { input: collectedFiles,
// Mirror architecture of static folders in generated .js and .css output: {
entryFileNames: (chunkInfo: Rollup.PreRenderedChunk) => { // Mirror architecture of static folders in generated .js and .css
if (chunkInfo.facadeModuleId !== null) { entryFileNames: (chunkInfo: Rollup.PreRenderedChunk) => {
return `${getRelativeAssetPath(chunkInfo.facadeModuleId)}.[hash].js`; if (chunkInfo.facadeModuleId !== null) {
} return `${getRelativeAssetPath(chunkInfo.facadeModuleId)}.[hash].js`;
return "[name].[hash].js"; }
return "[name].[hash].js";
},
assetFileNames: (chunkInfo: Rollup.PreRenderedAsset) => {
if (
chunkInfo.names?.length === 1 &&
chunkInfo.originalFileNames?.length === 1 &&
collectedFiles.includes(chunkInfo.originalFileNames[0])
) {
return `${getRelativeAssetPath(chunkInfo.originalFileNames[0])}.[hash][extname]`;
}
return "[name].[hash][extname]";
},
chunkFileNames: "[name].[hash].js",
}, },
assetFileNames: (chunkInfo: Rollup.PreRenderedAsset) => {
if (
chunkInfo.names?.length === 1 &&
chunkInfo.originalFileNames?.length === 1 &&
collectedFiles.includes(chunkInfo.originalFileNames[0])
) {
return `${getRelativeAssetPath(chunkInfo.originalFileNames[0])}.[hash][extname]`;
}
return "[name].[hash][extname]";
},
chunkFileNames: "[name].[hash].js",
}, },
}, },
}, resolve: {
resolve: { alias: getAliases(),
alias: getAliases(), },
},
plugins: [ plugins: [
inject({ inject({
// biome-ignore lint/style/useNamingConvention: that's how it's called // biome-ignore lint/style/useNamingConvention: that's how it's called
Alpine: "alpinejs", Alpine: "alpinejs",
}), }),
viteStaticCopy({ viteStaticCopy({
targets: [ targets: [
{ {
src: resolve(nodeModules, "jquery/dist/jquery.min.js"), src: resolve(nodeModules, "jquery/dist/jquery.min.js"),
dest: vendored, dest: vendored,
}, },
{ {
src: resolve(nodeModules, "jquery-ui/dist/jquery-ui.min.js"), src: resolve(nodeModules, "jquery-ui/dist/jquery-ui.min.js"),
dest: vendored, dest: vendored,
}, },
{ {
src: resolve(nodeModules, "jquery.shorten/src/jquery.shorten.min.js"), src: resolve(nodeModules, "jquery.shorten/src/jquery.shorten.min.js"),
dest: vendored, dest: vendored,
}, },
], ],
}), }),
], ],
optimizeDeps: { optimizeDeps: {
include: ["jquery"], include: ["jquery"],
}, },
} satisfies UserConfig; } satisfies UserConfig;
});