diff --git a/.changeset/thirty-phones-pretend.md b/.changeset/thirty-phones-pretend.md new file mode 100644 index 0000000..0ced225 --- /dev/null +++ b/.changeset/thirty-phones-pretend.md @@ -0,0 +1,5 @@ +--- +"arc-vite": minor +--- + +Add support for other Vite plugins registering additional files to check when determining if a module is adaptive. diff --git a/.eslintrc.json b/.eslintrc.json index 3d57b03..1f74204 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -10,23 +10,31 @@ }, "settings": { "import/resolver": { - "typescript": { - "project": true - } + "typescript": true } }, "env": { "node": true, - "browser": true, - "es2024": true + "es2024": true, + "browser": true }, "rules": { - "import/order": ["error"], - "sort-imports": [ + "import/order": [ "error", { - "allowSeparatedGroups": true, - "ignoreDeclarationSort": true + "groups": [ + ["builtin"], + ["external"], + "internal", + "parent", + "sibling", + "index" + ], + "alphabetize": { + "order": "asc", + "orderImportKind": "asc", + "caseInsensitive": true + } } ] }, @@ -36,9 +44,9 @@ "parser": "@typescript-eslint/parser", "extends": [ "eslint:recommended", + "plugin:@typescript-eslint/recommended", "plugin:import/recommended", "plugin:import/typescript", - "plugin:@typescript-eslint/recommended", "prettier" ], "rules": { diff --git a/package-lock.json b/package-lock.json index 0e7893c..0120832 100644 --- a/package-lock.json +++ b/package-lock.json @@ -18,26 +18,26 @@ "@changesets/changelog-github": "^0.4.8", "@changesets/cli": "^2.26.2", "@marko/compiler": "^5.33.7", - "@marko/vite": "^4.0.0", + "@marko/vite": "^4.0.1", "@playwright/test": "^1.39.0", "@types/estree": "^1.0.5", "@types/node": "^20.9.0", "@types/serve-handler": "^6.1.4", - "@typescript-eslint/eslint-plugin": "^6.10.0", - "@typescript-eslint/parser": "^6.10.0", + "@typescript-eslint/eslint-plugin": "^6.11.0", + "@typescript-eslint/parser": "^6.11.0", "c8": "^8.0.1", "eslint": "^8.53.0", "eslint-config-prettier": "^9.0.0", "eslint-import-resolver-typescript": "^3.6.1", "eslint-plugin-import": "^2.29.0", "husky": "^8.0.3", - "lint-staged": "^15.0.2", + "lint-staged": "^15.1.0", "marko": "^5.31.17", "playwright": "^1.39.0", "prettier": "^3.1.0", "serve-handler": "^6.1.5", "sort-package-json": "^2.6.0", - "tsx": "^4.1.0", + "tsx": "^4.1.1", "typescript": "^5.2.2", "vite": "^4.5.0" }, @@ -839,6 +839,54 @@ "url": "https://github.com/prettier/prettier?sponsor=1" } }, + "node_modules/@esbuild/android-arm": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.18.20.tgz", + "integrity": "sha512-fyi7TDI/ijKKNZTUJAQqiG5T7YjJXgnzkURqmGj13C6dCqckZBLdl4h7bkhHt/t0WP+zO9/zwroDvANaOqO5Sw==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.18.20.tgz", + "integrity": "sha512-Nz4rJcchGDtENV0eMKUNa6L12zz2zBDXuhj/Vjh18zGqB44Bi7MBMSXjgunJgjRhCmKOjnPuZp4Mb6OKqtMHLQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.18.20.tgz", + "integrity": "sha512-8GDdlePJA8D6zlZYJV/jnrRAi6rOiNaCC/JclcXpB+KIuvfBN4owLtgzY2bsxnx666XjJx2kDPUmnTtR8qKQUg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, "node_modules/@esbuild/darwin-arm64": { "version": "0.18.20", "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.18.20.tgz", @@ -855,6 +903,294 @@ "node": ">=12" } }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.18.20.tgz", + "integrity": "sha512-pc5gxlMDxzm513qPGbCbDukOdsGtKhfxD1zJKXjCCcU7ju50O7MeAZ8c4krSJcOIJGFR+qx21yMMVYwiQvyTyQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.18.20.tgz", + "integrity": "sha512-yqDQHy4QHevpMAaxhhIwYPMv1NECwOvIpGCZkECn8w2WFHXjEwrBn3CeNIYsibZ/iZEUemj++M26W3cNR5h+Tw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.18.20.tgz", + "integrity": "sha512-tgWRPPuQsd3RmBZwarGVHZQvtzfEBOreNuxEMKFcd5DaDn2PbBxfwLcj4+aenoh7ctXcbXmOQIn8HI6mCSw5MQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.18.20.tgz", + "integrity": "sha512-/5bHkMWnq1EgKr1V+Ybz3s1hWXok7mDFUMQ4cG10AfW3wL02PSZi5kFpYKrptDsgb2WAJIvRcDm+qIvXf/apvg==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.18.20.tgz", + "integrity": "sha512-2YbscF+UL7SQAVIpnWvYwM+3LskyDmPhe31pE7/aoTMFKKzIc9lLbyGUpmmb8a8AixOL61sQ/mFh3jEjHYFvdA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.18.20.tgz", + "integrity": "sha512-P4etWwq6IsReT0E1KHU40bOnzMHoH73aXp96Fs8TIT6z9Hu8G6+0SHSw9i2isWrD2nbx2qo5yUqACgdfVGx7TA==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.18.20.tgz", + "integrity": "sha512-nXW8nqBTrOpDLPgPY9uV+/1DjxoQ7DoB2N8eocyq8I9XuqJ7BiAMDMf9n1xZM9TgW0J8zrquIb/A7s3BJv7rjg==", + "cpu": [ + "loong64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.18.20.tgz", + "integrity": "sha512-d5NeaXZcHp8PzYy5VnXV3VSd2D328Zb+9dEq5HE6bw6+N86JVPExrA6O68OPwobntbNJ0pzCpUFZTo3w0GyetQ==", + "cpu": [ + "mips64el" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.18.20.tgz", + "integrity": "sha512-WHPyeScRNcmANnLQkq6AfyXRFr5D6N2sKgkFo2FqguP44Nw2eyDlbTdZwd9GYk98DZG9QItIiTlFLHJHjxP3FA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.18.20.tgz", + "integrity": "sha512-WSxo6h5ecI5XH34KC7w5veNnKkju3zBRLEQNY7mv5mtBmrP/MjNBCAlsM2u5hDBlS3NGcTQpoBvRzqBcRtpq1A==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.18.20.tgz", + "integrity": "sha512-+8231GMs3mAEth6Ja1iK0a1sQ3ohfcpzpRLH8uuc5/KVDFneH6jtAJLFGafpzpMRO6DzJ6AvXKze9LfFMrIHVQ==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.18.20.tgz", + "integrity": "sha512-UYqiqemphJcNsFEskc73jQ7B9jgwjWrSayxawS6UVFZGWrAAtkzjxSqnoclCXxWtfwLdzU+vTpcNYhpn43uP1w==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.18.20.tgz", + "integrity": "sha512-iO1c++VP6xUBUmltHZoMtCUdPlnPGdBom6IrO4gyKPFFVBKioIImVooR5I83nTew5UOYrk3gIJhbZh8X44y06A==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.18.20.tgz", + "integrity": "sha512-e5e4YSsuQfX4cxcygw/UCPIEP6wbIL+se3sxPdCiMbFLBWu0eiZOJ7WoD+ptCLrmjZBK1Wk7I6D/I3NglUGOxg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.18.20.tgz", + "integrity": "sha512-kDbFRFp0YpTQVVrqUd5FTYmWo45zGaXe0X8E1G/LKFC0v8x0vWrhOWSLITcCn63lmZIxfOMXtCfti/RxN/0wnQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.18.20.tgz", + "integrity": "sha512-ddYFR6ItYgoaq4v4JmQQaAI5s7npztfV4Ag6NrhiaW0RrnOXqBkgwZLofVTlq1daVTQNhtI5oieTvkRPfZrePg==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.18.20.tgz", + "integrity": "sha512-Wv7QBi3ID/rROT08SABTS7eV4hX26sVduqDOTe1MvGMjNd3EjOz4b7zeexIR62GTIEKrfJXKL9LFxTYgkyeu7g==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.18.20", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.18.20.tgz", + "integrity": "sha512-kTdfRcSiDfQca/y9QIkng02avJ+NCaQvrMejlsB3RRv5sE9rRoeBPISaZpKxHELzRxZyLvNts1P27W3wV+8geQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, "node_modules/@eslint-community/eslint-utils": { "version": "4.4.0", "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", @@ -1185,9 +1521,9 @@ } }, "node_modules/@marko/vite": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/@marko/vite/-/vite-4.0.0.tgz", - "integrity": "sha512-yeYTimM9xQxX1ct6EGTJmJG1rcQk/+DmcB1ErBW1qJB1h8o8qHk3sAGFw/faPrgODneD/V9wtwc1Viih14T61A==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/@marko/vite/-/vite-4.0.1.tgz", + "integrity": "sha512-vK8jBM2xGj/FkY9Dq/9CYhmdeOKt5BCj7xZxIlP3iInoS6cm5qN+mvfU2foxGXZ57/m83P44SWTFRfvP8KqeNA==", "dev": true, "dependencies": { "anymatch": "^3.1.3", @@ -1322,16 +1658,16 @@ } }, "node_modules/@typescript-eslint/eslint-plugin": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.10.0.tgz", - "integrity": "sha512-uoLj4g2OTL8rfUQVx2AFO1hp/zja1wABJq77P6IclQs6I/m9GLrm7jCdgzZkvWdDCQf1uEvoa8s8CupsgWQgVg==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.11.0.tgz", + "integrity": "sha512-uXnpZDc4VRjY4iuypDBKzW1rz9T5YBBK0snMn8MaTSNd2kMlj50LnLBABELjJiOL5YHk7ZD8hbSpI9ubzqYI0w==", "dev": true, "dependencies": { "@eslint-community/regexpp": "^4.5.1", - "@typescript-eslint/scope-manager": "6.10.0", - "@typescript-eslint/type-utils": "6.10.0", - "@typescript-eslint/utils": "6.10.0", - "@typescript-eslint/visitor-keys": "6.10.0", + "@typescript-eslint/scope-manager": "6.11.0", + "@typescript-eslint/type-utils": "6.11.0", + "@typescript-eslint/utils": "6.11.0", + "@typescript-eslint/visitor-keys": "6.11.0", "debug": "^4.3.4", "graphemer": "^1.4.0", "ignore": "^5.2.4", @@ -1357,15 +1693,15 @@ } }, "node_modules/@typescript-eslint/parser": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.10.0.tgz", - "integrity": "sha512-+sZwIj+s+io9ozSxIWbNB5873OSdfeBEH/FR0re14WLI6BaKuSOnnwCJ2foUiu8uXf4dRp1UqHP0vrZ1zXGrog==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.11.0.tgz", + "integrity": "sha512-+whEdjk+d5do5nxfxx73oanLL9ghKO3EwM9kBCkUtWMRwWuPaFv9ScuqlYfQ6pAD6ZiJhky7TZ2ZYhrMsfMxVQ==", "dev": true, "dependencies": { - "@typescript-eslint/scope-manager": "6.10.0", - "@typescript-eslint/types": "6.10.0", - "@typescript-eslint/typescript-estree": "6.10.0", - "@typescript-eslint/visitor-keys": "6.10.0", + "@typescript-eslint/scope-manager": "6.11.0", + "@typescript-eslint/types": "6.11.0", + "@typescript-eslint/typescript-estree": "6.11.0", + "@typescript-eslint/visitor-keys": "6.11.0", "debug": "^4.3.4" }, "engines": { @@ -1385,13 +1721,13 @@ } }, "node_modules/@typescript-eslint/scope-manager": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.10.0.tgz", - "integrity": "sha512-TN/plV7dzqqC2iPNf1KrxozDgZs53Gfgg5ZHyw8erd6jd5Ta/JIEcdCheXFt9b1NYb93a1wmIIVW/2gLkombDg==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.11.0.tgz", + "integrity": "sha512-0A8KoVvIURG4uhxAdjSaxy8RdRE//HztaZdG8KiHLP8WOXSk0vlF7Pvogv+vlJA5Rnjj/wDcFENvDaHb+gKd1A==", "dev": true, "dependencies": { - "@typescript-eslint/types": "6.10.0", - "@typescript-eslint/visitor-keys": "6.10.0" + "@typescript-eslint/types": "6.11.0", + "@typescript-eslint/visitor-keys": "6.11.0" }, "engines": { "node": "^16.0.0 || >=18.0.0" @@ -1402,13 +1738,13 @@ } }, "node_modules/@typescript-eslint/type-utils": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.10.0.tgz", - "integrity": "sha512-wYpPs3hgTFblMYwbYWPT3eZtaDOjbLyIYuqpwuLBBqhLiuvJ+9sEp2gNRJEtR5N/c9G1uTtQQL5AhV0fEPJYcg==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.11.0.tgz", + "integrity": "sha512-nA4IOXwZtqBjIoYrJcYxLRO+F9ri+leVGoJcMW1uqr4r1Hq7vW5cyWrA43lFbpRvQ9XgNrnfLpIkO3i1emDBIA==", "dev": true, "dependencies": { - "@typescript-eslint/typescript-estree": "6.10.0", - "@typescript-eslint/utils": "6.10.0", + "@typescript-eslint/typescript-estree": "6.11.0", + "@typescript-eslint/utils": "6.11.0", "debug": "^4.3.4", "ts-api-utils": "^1.0.1" }, @@ -1429,9 +1765,9 @@ } }, "node_modules/@typescript-eslint/types": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.10.0.tgz", - "integrity": "sha512-36Fq1PWh9dusgo3vH7qmQAj5/AZqARky1Wi6WpINxB6SkQdY5vQoT2/7rW7uBIsPDcvvGCLi4r10p0OJ7ITAeg==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.11.0.tgz", + "integrity": "sha512-ZbEzuD4DwEJxwPqhv3QULlRj8KYTAnNsXxmfuUXFCxZmO6CF2gM/y+ugBSAQhrqaJL3M+oe4owdWunaHM6beqA==", "dev": true, "engines": { "node": "^16.0.0 || >=18.0.0" @@ -1442,13 +1778,13 @@ } }, "node_modules/@typescript-eslint/typescript-estree": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.10.0.tgz", - "integrity": "sha512-ek0Eyuy6P15LJVeghbWhSrBCj/vJpPXXR+EpaRZqou7achUWL8IdYnMSC5WHAeTWswYQuP2hAZgij/bC9fanBg==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.11.0.tgz", + "integrity": "sha512-Aezzv1o2tWJwvZhedzvD5Yv7+Lpu1by/U1LZ5gLc4tCx8jUmuSCMioPFRjliN/6SJIvY6HpTtJIWubKuYYYesQ==", "dev": true, "dependencies": { - "@typescript-eslint/types": "6.10.0", - "@typescript-eslint/visitor-keys": "6.10.0", + "@typescript-eslint/types": "6.11.0", + "@typescript-eslint/visitor-keys": "6.11.0", "debug": "^4.3.4", "globby": "^11.1.0", "is-glob": "^4.0.3", @@ -1469,17 +1805,17 @@ } }, "node_modules/@typescript-eslint/utils": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.10.0.tgz", - "integrity": "sha512-v+pJ1/RcVyRc0o4wAGux9x42RHmAjIGzPRo538Z8M1tVx6HOnoQBCX/NoadHQlZeC+QO2yr4nNSFWOoraZCAyg==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.11.0.tgz", + "integrity": "sha512-p23ibf68fxoZy605dc0dQAEoUsoiNoP3MD9WQGiHLDuTSOuqoTsa4oAy+h3KDkTcxbbfOtUjb9h3Ta0gT4ug2g==", "dev": true, "dependencies": { "@eslint-community/eslint-utils": "^4.4.0", "@types/json-schema": "^7.0.12", "@types/semver": "^7.5.0", - "@typescript-eslint/scope-manager": "6.10.0", - "@typescript-eslint/types": "6.10.0", - "@typescript-eslint/typescript-estree": "6.10.0", + "@typescript-eslint/scope-manager": "6.11.0", + "@typescript-eslint/types": "6.11.0", + "@typescript-eslint/typescript-estree": "6.11.0", "semver": "^7.5.4" }, "engines": { @@ -1494,12 +1830,12 @@ } }, "node_modules/@typescript-eslint/visitor-keys": { - "version": "6.10.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.10.0.tgz", - "integrity": "sha512-xMGluxQIEtOM7bqFCo+rCMh5fqI+ZxV5RUUOa29iVPz1OgCZrtc7rFnz5cLUazlkPKYqX+75iuDq7m0HQ48nCg==", + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.11.0.tgz", + "integrity": "sha512-+SUN/W7WjBr05uRxPggJPSzyB8zUpaYo2hByKasWbqr3PM8AXfZt8UHdNpBS1v9SA62qnSSMF3380SwDqqprgQ==", "dev": true, "dependencies": { - "@typescript-eslint/types": "6.10.0", + "@typescript-eslint/types": "6.11.0", "eslint-visitor-keys": "^3.4.1" }, "engines": { @@ -2621,9 +2957,9 @@ "dev": true }, "node_modules/electron-to-chromium": { - "version": "1.4.580", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.580.tgz", - "integrity": "sha512-T5q3pjQon853xxxHUq3ZP68ZpvJHuSMY2+BZaW3QzjS4HvNuvsMmZ/+lU+nCrftre1jFZ+OSlExynXWBihnXzw==", + "version": "1.4.582", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.582.tgz", + "integrity": "sha512-89o0MGoocwYbzqUUjc+VNpeOFSOK9nIdC5wY4N+PVUarUK0MtjyTjks75AZS2bW4Kl8MdewdFsWaH0jLy+JNoA==", "dev": true }, "node_modules/emoji-regex": { @@ -3504,9 +3840,9 @@ } }, "node_modules/flat-cache": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.1.1.tgz", - "integrity": "sha512-/qM2b3LUIaIgviBQovTLvijfyOQXPtSRnRK26ksj2J7rzPIecePUIpJsZ4T02Qg+xiAEKIs5K8dsHEd+VaKa/Q==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", "dev": true, "dependencies": { "flatted": "^3.2.9", @@ -3514,7 +3850,7 @@ "rimraf": "^3.0.2" }, "engines": { - "node": ">=12.0.0" + "node": "^10.12.0 || >=12.0.0" } }, "node_modules/flatted": { @@ -3927,9 +4263,9 @@ "dev": true }, "node_modules/htmljs-parser": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/htmljs-parser/-/htmljs-parser-5.5.0.tgz", - "integrity": "sha512-KcJ84GmLFo+EWQ1hYjgLTSro8kEMFuTTdchOUeco5N95peHMtoY5XsX7YeU4oj+tPrvl9OXHlUkJAnY5zshhaQ==", + "version": "5.5.1", + "resolved": "https://registry.npmjs.org/htmljs-parser/-/htmljs-parser-5.5.1.tgz", + "integrity": "sha512-mzTQUcQFDLS6LN20AiPKxG5dXmQOhej67ztIpnDzz5raAxrNsh8A17Wqef7KLWI4yJ4Uolj3syJmddFPdATn2Q==", "dev": true }, "node_modules/htmlparser2": { @@ -4597,9 +4933,9 @@ "dev": true }, "node_modules/lint-staged": { - "version": "15.0.2", - "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.0.2.tgz", - "integrity": "sha512-vnEy7pFTHyVuDmCAIFKR5QDO8XLVlPFQQyujQ/STOxe40ICWqJ6knS2wSJ/ffX/Lw0rz83luRDh+ET7toN+rOw==", + "version": "15.1.0", + "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.1.0.tgz", + "integrity": "sha512-ZPKXWHVlL7uwVpy8OZ7YQjYDAuO5X4kMh0XgZvPNxLcCCngd0PO5jKQyy3+s4TL2EnHoIXIzP1422f/l3nZKMw==", "dev": true, "dependencies": { "chalk": "5.3.0", @@ -4611,7 +4947,7 @@ "micromatch": "4.0.5", "pidtree": "0.6.0", "string-argv": "0.3.2", - "yaml": "2.3.3" + "yaml": "2.3.4" }, "bin": { "lint-staged": "bin/lint-staged.js" @@ -6814,9 +7150,9 @@ } }, "node_modules/tsx": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/tsx/-/tsx-4.1.0.tgz", - "integrity": "sha512-u4l17Yd63Wsk2fzNn1wZCmcS9kwJ/2ysl7wuoVggv2hd3NjLA5JQPpyJMXoWSXOwOvoQUzNcu/sf/35HEsnXsg==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/tsx/-/tsx-4.1.1.tgz", + "integrity": "sha512-zyPn5BFMB0TB5kMLbYPNx4x/oL/oSlaecdKCv6WeJ0TeSEfx8RTJWjuB5TZ2dSewktgfBsBO/HNA9mrMWqLXMA==", "dev": true, "dependencies": { "esbuild": "~0.18.20", @@ -7384,9 +7720,9 @@ "dev": true }, "node_modules/yaml": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.3.3.tgz", - "integrity": "sha512-zw0VAJxgeZ6+++/su5AFoqBbZbrEakwu+X0M5HmcwUiBL7AzcuPKjj5we4xfQLp78LkEMpD0cOnUhmgOVy3KdQ==", + "version": "2.3.4", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.3.4.tgz", + "integrity": "sha512-8aAvwVUSHpfEqTQ4w/KMlf3HcRdt50E5ODIQJBw1fQ5RL34xabzxtUlzTXVqc4rkZsPbvrXKWnABCD7kWSmocA==", "dev": true, "engines": { "node": ">= 14" diff --git a/package.json b/package.json index 54f56eb..6acabb5 100644 --- a/package.json +++ b/package.json @@ -50,26 +50,26 @@ "@changesets/changelog-github": "^0.4.8", "@changesets/cli": "^2.26.2", "@marko/compiler": "^5.33.7", - "@marko/vite": "^4.0.0", + "@marko/vite": "^4.0.1", "@playwright/test": "^1.39.0", "@types/estree": "^1.0.5", "@types/node": "^20.9.0", "@types/serve-handler": "^6.1.4", - "@typescript-eslint/eslint-plugin": "^6.10.0", - "@typescript-eslint/parser": "^6.10.0", + "@typescript-eslint/eslint-plugin": "^6.11.0", + "@typescript-eslint/parser": "^6.11.0", "c8": "^8.0.1", "eslint": "^8.53.0", "eslint-config-prettier": "^9.0.0", "eslint-import-resolver-typescript": "^3.6.1", "eslint-plugin-import": "^2.29.0", "husky": "^8.0.3", - "lint-staged": "^15.0.2", + "lint-staged": "^15.1.0", "marko": "^5.31.17", "playwright": "^1.39.0", "prettier": "^3.1.0", "serve-handler": "^6.1.5", "sort-package-json": "^2.6.0", - "tsx": "^4.1.0", + "tsx": "^4.1.1", "typescript": "^5.2.2", "vite": "^4.5.0" }, diff --git a/src/index.ts b/src/index.ts index 16b507d..a89cacf 100644 --- a/src/index.ts +++ b/src/index.ts @@ -2,9 +2,9 @@ import type * as _arcServer from "arc-server"; import type { Plugin } from "vite"; -import { pluginServe } from "./plugins/serve"; -import { pluginBuildWeb } from "./plugins/build-web"; import { pluginBuildSSR } from "./plugins/build-ssr"; +import { pluginBuildWeb } from "./plugins/build-web"; +import { pluginServe } from "./plugins/serve"; import { type Options, getInternalPluginOptions } from "./utils/options"; export { createFlagSets, hasFlags } from "./utils/flags"; diff --git a/src/plugins/build-ssr.ts b/src/plugins/build-ssr.ts index 4d27a4a..ac54c15 100644 --- a/src/plugins/build-ssr.ts +++ b/src/plugins/build-ssr.ts @@ -1,16 +1,22 @@ import path from "path"; import type { Plugin } from "vite"; -import { type InternalPluginOptions } from "../utils/options"; -import { type Matches, clearCache, getMatches } from "../utils/matches"; -import { indexToId } from "../utils/index-to-id"; +import { getAdaptiveFS } from "../utils/adaptive-fs"; +import { ensureArcPluginIsFirst } from "../utils/ensure-arc-plugin-is-first"; import { decodeFileName, encodeFileName } from "../utils/filename-encoding"; +import { indexToId } from "../utils/index-to-id"; import { isCssFile } from "../utils/is-css-file"; +import { type Matches, clearCache } from "../utils/matches"; +import { type InternalPluginOptions } from "../utils/options"; import { toPosix } from "../utils/to-posix"; -import { ensureArcPluginIsFirst } from "../utils/ensure-arc-plugin-is-first"; +import { + decodeArcVirtualMatch, + getVirtualMatches, + isArcVirtualMatch, +} from "../utils/virtual-matches"; const virtualArcServerModuleId = "\0arc-server-virtual"; const arcPrefix = "\0arc-"; -const arcSuffix = ".js"; +const arcSuffix = ".mjs"; const arcProxyPrefix = `${arcPrefix}proxy:`; // TODO: with some tweaks this plugin might work in a test env. @@ -62,16 +68,12 @@ export function pluginBuildSSR({ ...options, skipSelf: true, }); - if (resolved) { - const { id } = resolved; - if (path.isAbsolute(id)) { - const matches = getMatches(id, flagSets); - if (matches) { - adaptiveMatchesForId.set(id, matches); - return { - id: encodeArcProxyId(id), - }; - } + if (resolved && !resolved.external) { + const matches = await getVirtualMatches(this, flagSets, resolved); + if (matches) { + const { id } = resolved; + adaptiveMatchesForId.set(id, matches); + return { id: encodeArcProxyId(id) }; } } @@ -115,7 +117,7 @@ function partsToString(parts, base, injectAttrs) { }; } - if (isArcProxyId(arcProxyPrefix)) { + if (isArcProxyId(id)) { id = decodeArcProxyId(id); const adaptiveMatches = adaptiveMatchesForId.get(id); if (adaptiveMatches) { @@ -193,6 +195,17 @@ function partsToString(parts, base, injectAttrs) { }; } } + } else if (isArcVirtualMatch(id)) { + const [adaptiveImport, flagSet] = decodeArcVirtualMatch(id); + const { meta, moduleSideEffects, syntheticNamedExports } = + this.getModuleInfo(adaptiveImport)!; + const code = meta.source as string; + return { + code, + moduleSideEffects, + syntheticNamedExports, + meta: { ...meta, arcFileSystem: getAdaptiveFS(flagSet) }, + }; } return null; diff --git a/src/plugins/build-web.ts b/src/plugins/build-web.ts index 0003577..7910224 100644 --- a/src/plugins/build-web.ts +++ b/src/plugins/build-web.ts @@ -1,19 +1,25 @@ -import path from "path"; import { promises as fs } from "fs"; +import path from "path"; import type * as estree from "estree"; import type { Plugin } from "vite"; -import { type InternalPluginOptions } from "../utils/options"; -import { type Matches, getMatches } from "../utils/matches"; +import { getAdaptiveFS } from "../utils/adaptive-fs"; +import { ensureArcPluginIsFirst } from "../utils/ensure-arc-plugin-is-first"; +import { decodeFileName, encodeFileName } from "../utils/filename-encoding"; import { type FlagSet, compareFlaggedObject, hasFlags } from "../utils/flags"; import { indexToId } from "../utils/index-to-id"; -import { decodeFileName, encodeFileName } from "../utils/filename-encoding"; import { isCssFile } from "../utils/is-css-file"; import { type DocManifest, generateDocManifest, generateInputDoc, } from "../utils/manifest"; -import { ensureArcPluginIsFirst } from "../utils/ensure-arc-plugin-is-first"; +import { type Matches } from "../utils/matches"; +import { type InternalPluginOptions } from "../utils/options"; +import { + decodeArcVirtualMatch, + getVirtualMatches, + isArcVirtualMatch, +} from "../utils/virtual-matches"; const arcPrefix = "\0arc-"; const arcJsSuffix = ".mjs"; @@ -68,28 +74,22 @@ export function pluginBuildWeb({ ...options, skipSelf: true, }); - if (resolved) { - if (path.isAbsolute(resolved.id)) { - const matches = getMatches(resolved.id, flagSets); - if (matches) { - adaptiveMatchesForId.set(resolved.id, matches); + if (resolved && !resolved.external) { + const matches = await getVirtualMatches(this, flagSets, resolved); + if (matches) { + const { id } = resolved; + adaptiveMatchesForId.set(id, matches); - const adaptiveImportsForImporter = - adaptiveImporters.get(importer); - if (adaptiveImportsForImporter) { - adaptiveImportsForImporter.set(source, resolved.id); - } else { - adaptiveImporters.set( - importer, - new Map([[source, resolved.id]]), - ); - } - - return { - id: encodeArcProxyId(resolved.id), - }; + const adaptiveImportsForImporter = + adaptiveImporters.get(importer); + if (adaptiveImportsForImporter) { + adaptiveImportsForImporter.set(source, id); + } else { + adaptiveImporters.set(importer, new Map([[source, id]])); } + + return { id: encodeArcProxyId(id) }; } } @@ -337,6 +337,17 @@ export function pluginBuildWeb({ };\n`, moduleSideEffects: "no-treeshake", }; + } else if (isArcVirtualMatch(id)) { + const [adaptiveImport, flagSet] = decodeArcVirtualMatch(id); + const { meta, moduleSideEffects, syntheticNamedExports } = + this.getModuleInfo(adaptiveImport)!; + const code = meta.source as string; + return { + code, + moduleSideEffects, + syntheticNamedExports, + meta: { ...meta, arcFileSystem: getAdaptiveFS(flagSet) }, + }; } return null; diff --git a/src/plugins/serve.ts b/src/plugins/serve.ts index 8d676e6..5d8f1d9 100644 --- a/src/plugins/serve.ts +++ b/src/plugins/serve.ts @@ -1,10 +1,12 @@ -import path from "path"; +import fs from "fs"; import { createRequire } from "module"; +import path from "path"; import type { Plugin } from "vite"; -import { type InternalPluginOptions } from "../utils/options"; -import { getMatches } from "../utils/matches"; -import { type FlagSet, hasFlags } from "../utils/flags"; +import { patchFS } from "../utils/adaptive-fs"; import { ensureArcPluginIsFirst } from "../utils/ensure-arc-plugin-is-first"; +import { type FlagSet, hasFlags } from "../utils/flags"; +import { getMatches } from "../utils/matches"; +import { type InternalPluginOptions } from "../utils/options"; // TODO: support forced flagset for build plugins @@ -13,6 +15,7 @@ export function pluginServe({ forceFlagSet, }: InternalPluginOptions): Plugin { const flagSet = forceFlagSet?.length ? forceFlagSet : undefined; + let restoreFS: ReturnType | undefined; return { name: "arc-vite:serve", enforce: "pre", @@ -29,6 +32,7 @@ export function pluginServe({ if (!flagSet) return; + restoreFS = patchFS(flagSet, fs); ensureArcPluginIsFirst(config.plugins!); config.cacheDir = path.resolve( `node_modules/.vite/arc/${flagSet.join(".")}`, @@ -85,6 +89,10 @@ export function pluginServe({ return null; }, + buildEnd() { + restoreFS?.(); + restoreFS = undefined; + }, }; } diff --git a/src/tests/basic.test.ts b/src/tests/basic.test.ts index a15edd2..bd8abea 100644 --- a/src/tests/basic.test.ts +++ b/src/tests/basic.test.ts @@ -1,10 +1,10 @@ -import url from "node:url"; import path from "node:path"; import * as t from "node:test"; -import { type Page } from "playwright"; +import url from "node:url"; import { expect } from "@playwright/test"; -import { getPage } from "./utils/get-page"; +import { type Page } from "playwright"; import { createDevServer } from "./utils/dev-server"; +import { getPage } from "./utils/get-page"; import { createProdServer } from "./utils/prod-server"; const fixture = path.join( diff --git a/src/tests/fixtures/basic/src/entry-server.ts b/src/tests/fixtures/basic/src/entry-server.ts index 09ede5d..e0c6e18 100644 --- a/src/tests/fixtures/basic/src/entry-server.ts +++ b/src/tests/fixtures/basic/src/entry-server.ts @@ -1,7 +1,7 @@ -import url from "url"; -import path from "path"; import { promises as fs } from "fs"; import type { IncomingMessage } from "http"; +import path from "path"; +import url from "url"; import { getAssets, withFlags } from "arc-server"; import renderApp from "./entry-web"; diff --git a/src/tests/fixtures/marko/config.ts b/src/tests/fixtures/marko/config.ts index cc01530..55964e7 100644 --- a/src/tests/fixtures/marko/config.ts +++ b/src/tests/fixtures/marko/config.ts @@ -1,5 +1,5 @@ -import { defineConfig } from "vite"; import markoVite from "@marko/vite"; +import { defineConfig } from "vite"; import arcVite from "../../.."; export default () => defineConfig({ diff --git a/src/tests/marko.test.ts b/src/tests/marko.test.ts index f344bc1..c042b7e 100644 --- a/src/tests/marko.test.ts +++ b/src/tests/marko.test.ts @@ -1,10 +1,10 @@ -import url from "node:url"; import path from "node:path"; import * as t from "node:test"; -import { type Page } from "playwright"; +import url from "node:url"; import { expect } from "@playwright/test"; -import { getPage } from "./utils/get-page"; +import { type Page } from "playwright"; import { createDevServer } from "./utils/dev-server"; +import { getPage } from "./utils/get-page"; import { createProdServer } from "./utils/prod-server"; const fixture = path.join( diff --git a/src/tests/utils/prod-server.ts b/src/tests/utils/prod-server.ts index ef8ed8a..0213cbd 100644 --- a/src/tests/utils/prod-server.ts +++ b/src/tests/utils/prod-server.ts @@ -1,8 +1,8 @@ -import path from "node:path"; -import http, { type IncomingMessage } from "node:http"; import events from "node:events"; -import * as vite from "vite"; +import http, { type IncomingMessage } from "node:http"; +import path from "node:path"; import serve from "serve-handler"; +import * as vite from "vite"; export async function createProdServer(fixtureDir: string) { const getConfig: () => vite.UserConfig = ( diff --git a/src/utils/adaptive-fs.ts b/src/utils/adaptive-fs.ts new file mode 100644 index 0000000..8784cab --- /dev/null +++ b/src/utils/adaptive-fs.ts @@ -0,0 +1,131 @@ +import fs from "fs"; +import type { FlagSet } from "./flags"; +import { getMatches } from "./matches"; + +const arcFlagReg = /\[/; +const fsByFlagSet = new Map(); + +export function getAdaptiveFS(flagSet: FlagSet) { + const flags = flagSet.join("."); + let fileSystem = fsByFlagSet.get(flags); + if (!fileSystem) { + fileSystem = createAdaptiveFS(flagSet); + fsByFlagSet.set(flags, fileSystem); + } + + return fileSystem; +} + +export function patchFS(flagSet: FlagSet, afs: any) { + const flagSets = [flagSet]; + + // Sync api + const { readFileSync, statSync, readlinkSync, accessSync, readdirSync } = fs; + afs.readFileSync = (id: string, ...args: any[]) => + readFileSync(getMatch(id), ...args); + afs.statSync = (id: string, ...args: any[]) => + statSync(getMatch(id), ...args); + afs.readlinkSync = (id: string, ...args: any[]) => + readlinkSync(getMatch(id), ...args); + afs.accessSync = (id: string, ...args: any[]) => + accessSync(getMatch(id), ...args); + afs.readdirSync = (id: string, ...args: any[]) => { + const match = getMatch(id); + const entries = readdirSync(id, ...args); + return match === id ? entries : ignoreAdaptiveEntries(entries); + }; + + // Callback api + const { readFile, stat, readlink, access, readdir } = fs as any; + afs.readFile = (id: string, ...args: any[]) => + readFile(getMatch(id), ...args); + afs.stat = (id: string, ...args: any[]) => stat(getMatch(id), ...args); + afs.readlink = (id: string, ...args: any[]) => + readlink(getMatch(id), ...args); + afs.access = (id: string, ...args: any[]) => access(getMatch(id), ...args); + afs.readdir = (id: string, ...args: any[]) => { + const match = getMatch(id); + if (match === id) return readdir(id, ...args); + const cb = args.pop(); + readdir(id, ...args, (err: any, entries: string[]) => { + if (err) return cb(err); + cb(null, ignoreAdaptiveEntries(entries)); + }); + }; + + // Promise api + const { + promises: { + readFile: readFilePromise, + stat: statPromise, + readlink: readlinkPromise, + access: accessPromise, + readdir: readdirPromise, + }, + } = fs; + afs.promises.readFile = (id: string, ...args: any[]) => + readFilePromise(getMatch(id), ...args); + afs.promises.stat = (id: string, ...args: any[]) => + statPromise(getMatch(id), ...args); + afs.promises.readlink = (id: string, ...args: any[]) => + readlinkPromise(getMatch(id), ...args); + afs.promises.access = (id: string, ...args: any[]) => + accessPromise(getMatch(id), ...args); + afs.promises.readdir = (id: string, ...args: any[]) => { + const match = getMatch(id); + const entriesPromise = readdirPromise(id, ...args); + return match === id + ? entriesPromise + : entriesPromise.then(ignoreAdaptiveEntries); + }; + + return () => { + afs.readFileSync = readFileSync; + afs.statSync = statSync; + afs.readlinkSync = readlinkSync; + afs.accessSync = accessSync; + afs.readdirSync = readdirSync; + + afs.readFile = readFile; + afs.stat = stat; + afs.readlink = readlink; + afs.access = access; + afs.readdir = readdir; + + afs.promises.readFile = readFilePromise; + afs.promises.stat = statPromise; + afs.promises.readlink = readlinkPromise; + afs.promises.access = accessPromise; + afs.promises.readdir = readdirPromise; + }; + + function getMatch(id: unknown): string { + if (typeof id !== "string") return id as string; + const match = getMatches(id, flagSets); + return match ? match.alternates[0].value : id; + } +} + +function createAdaptiveFS(flagSet: FlagSet) { + const afs = { ...fs, promises: { ...fs.promises } } as any; + patchFS(flagSet, afs); + return afs as typeof fs; +} + +function ignoreAdaptiveEntries(entries: string[]) { + for (let i = 0; i < entries.length; i++) { + const entry = entries[i]; + if (arcFlagReg.test(entry)) { + const uniqueEntries = entries.slice(0, i); + for (; i < entries.length; i++) { + const entry = entries[i]; + if (!arcFlagReg.test(entry)) { + uniqueEntries.push(entry); + } + } + + return uniqueEntries; + } + } + return entries; +} diff --git a/src/utils/manifest.ts b/src/utils/manifest.ts index 87a7b5b..39b4080 100644 --- a/src/utils/manifest.ts +++ b/src/utils/manifest.ts @@ -1,6 +1,6 @@ -import { Parser } from "htmlparser2"; import { ElementType } from "domelementtype"; import { Comment, DomHandler, Element, Node, Text } from "domhandler"; +import { Parser } from "htmlparser2"; type Serialized = ReturnType; export type DocManifest = { diff --git a/src/utils/matches.ts b/src/utils/matches.ts index 86b16e8..85cb16e 100644 --- a/src/utils/matches.ts +++ b/src/utils/matches.ts @@ -1,3 +1,4 @@ +import fs from "fs"; import path from "path"; import Resolver from "arc-resolver"; import { @@ -9,10 +10,10 @@ import { export type Matches = { default: string; alternates: [Match, ...Match[]] }; export type Match = { flags: FlagSet; value: string }; +export const resolver = new Resolver(fs); type RawMatch = { flags: string[]; value: string }; -const resolver = new Resolver(); const hasQuery = /\?.*$/; export function getMatches( id: string, diff --git a/src/utils/read-once-persisted-store.ts b/src/utils/read-once-persisted-store.ts index 2f78e07..0f22366 100644 --- a/src/utils/read-once-persisted-store.ts +++ b/src/utils/read-once-persisted-store.ts @@ -1,6 +1,6 @@ +import { promises as fs } from "fs"; import os from "os"; import path from "path"; -import { promises as fs } from "fs"; const noop = () => {}; const tmpFile = path.join(os.tmpdir(), "vite-arc-storage.json"); diff --git a/src/utils/virtual-matches.ts b/src/utils/virtual-matches.ts new file mode 100644 index 0000000..1ce7d44 --- /dev/null +++ b/src/utils/virtual-matches.ts @@ -0,0 +1,84 @@ +import path from "path"; +import type { Rollup } from "vite"; +import { decodeFileName, encodeFileName } from "./filename-encoding"; +import { normalizeFlagSets, type FlagSet } from "./flags"; +import { getMatches, type Match } from "./matches"; + +const arcPrefix = "\0arc-"; +const arcVirtualMatchPrefix = `${arcPrefix}match:`; + +export async function getVirtualMatches( + ctx: Rollup.PluginContext, + flagSets: FlagSet[], + resolved: { id: string }, +) { + const { id } = resolved; + if (path.isAbsolute(id)) { + const matches = getMatches(id, flagSets); + if (matches) return matches; + } + + const { meta } = await ctx.load(resolved); + if (Array.isArray(meta.arcScanFiles) && typeof meta.source === "string") { + const matchesFlagSets: FlagSet[] = []; + await Promise.all( + (meta.arcScanFiles as string[]).map(async (id) => { + const matches = await getVirtualMatches(ctx, flagSets, { id }); + if (matches) { + for (const alternate of matches.alternates) { + matchesFlagSets.push(alternate.flags); + } + } + }), + ); + + let alternates: undefined | [Match, ...Match[]]; + for (const flagSet of normalizeFlagSets(matchesFlagSets)) { + const alternate: Match = { + flags: flagSet, + value: encodeArcVirtualMatch(id, flagSet), + }; + + if (alternates) { + alternates.push(alternate); + } else { + alternates = [alternate]; + } + } + + if (alternates) { + return { + default: id, + alternates, + }; + } + } +} + +export function isArcVirtualMatch(id: string) { + return id.startsWith(arcVirtualMatchPrefix); +} + +export function decodeArcVirtualMatch( + id: string, +): readonly [adaptiveImport: string, flagSet: FlagSet] { + const ext = path.extname(id); + const prefixEnd = id.indexOf(":", arcVirtualMatchPrefix.length + 1) + 1; + const adaptiveImport = + decodeFileName(id.slice(arcVirtualMatchPrefix.length, prefixEnd - 1)) + ext; + const flagSet = decodeFileName(id.slice(prefixEnd, -ext.length)).split( + ".", + ) as FlagSet; + return [adaptiveImport, flagSet]; +} + +function encodeArcVirtualMatch(id: string, flagSet: FlagSet) { + const ext = path.extname(id); + return ( + arcVirtualMatchPrefix + + encodeFileName(id.slice(0, -ext.length)) + + ":" + + encodeFileName(flagSet.join(".")) + + ext + ); +}