Compare commits
4 commits
main
...
file-abstr
Author | SHA1 | Date | |
---|---|---|---|
|
004e58078c | ||
|
b98c54477f | ||
|
4948d73fad | ||
|
63e379c146 |
11 changed files with 252 additions and 33 deletions
5
.changeset/tender-doors-poke.md
Normal file
5
.changeset/tender-doors-poke.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Internal refactor for how files are treated
|
|
@ -4,7 +4,7 @@ import { pathToFileURL } from 'node:url';
|
||||||
import { AstroSettings } from '../@types/astro.js';
|
import { AstroSettings } from '../@types/astro.js';
|
||||||
import { StaticBuildOptions } from '../core/build/types.js';
|
import { StaticBuildOptions } from '../core/build/types.js';
|
||||||
import { AstroError, AstroErrorData } from '../core/errors/index.js';
|
import { AstroError, AstroErrorData } from '../core/errors/index.js';
|
||||||
import { rootRelativePath } from '../core/util.js';
|
import File from '../core/file/index.js';
|
||||||
import { ImageService, isLocalService, LocalImageService } from './services/service.js';
|
import { ImageService, isLocalService, LocalImageService } from './services/service.js';
|
||||||
import type { ImageMetadata, ImageTransform } from './types.js';
|
import type { ImageMetadata, ImageTransform } from './types.js';
|
||||||
import { imageMetadata } from './utils/metadata.js';
|
import { imageMetadata } from './utils/metadata.js';
|
||||||
|
@ -127,7 +127,8 @@ export async function emitESMImage(
|
||||||
fileEmitter: any,
|
fileEmitter: any,
|
||||||
settings: AstroSettings
|
settings: AstroSettings
|
||||||
) {
|
) {
|
||||||
const url = pathToFileURL(id);
|
const file = new File(id, settings.config);
|
||||||
|
const url = file.toFileURL();
|
||||||
const meta = await imageMetadata(url);
|
const meta = await imageMetadata(url);
|
||||||
|
|
||||||
if (!meta) {
|
if (!meta) {
|
||||||
|
@ -152,7 +153,7 @@ export async function emitESMImage(
|
||||||
url.searchParams.append('origHeight', meta.height.toString());
|
url.searchParams.append('origHeight', meta.height.toString());
|
||||||
url.searchParams.append('origFormat', meta.format);
|
url.searchParams.append('origFormat', meta.format);
|
||||||
|
|
||||||
meta.src = rootRelativePath(settings.config, url);
|
meta.src = file.toRootRelativePath();
|
||||||
}
|
}
|
||||||
|
|
||||||
return meta;
|
return meta;
|
||||||
|
|
|
@ -8,6 +8,7 @@ import { normalizePath } from 'vite';
|
||||||
import { AstroPluginOptions, ImageTransform } from '../@types/astro';
|
import { AstroPluginOptions, ImageTransform } from '../@types/astro';
|
||||||
import { error } from '../core/logger/core.js';
|
import { error } from '../core/logger/core.js';
|
||||||
import { joinPaths, prependForwardSlash } from '../core/path.js';
|
import { joinPaths, prependForwardSlash } from '../core/path.js';
|
||||||
|
import File from '../core/file/index.js';
|
||||||
import { VIRTUAL_MODULE_ID, VIRTUAL_SERVICE_ID } from './consts.js';
|
import { VIRTUAL_MODULE_ID, VIRTUAL_SERVICE_ID } from './consts.js';
|
||||||
import { emitESMImage, isESMImportedImage } from './internal.js';
|
import { emitESMImage, isESMImportedImage } from './internal.js';
|
||||||
import { isLocalService } from './services/service.js';
|
import { isLocalService } from './services/service.js';
|
||||||
|
@ -76,15 +77,16 @@ export default function assets({
|
||||||
return next();
|
return next();
|
||||||
}
|
}
|
||||||
|
|
||||||
const filePathURL = new URL('.' + filePath, settings.config.root);
|
const file = new File(filePath, settings.config);
|
||||||
const file = await fs.readFile(filePathURL);
|
const filePathURL = file.toFileURL();
|
||||||
|
const buffer = await fs.readFile(file.toFileURL());
|
||||||
|
|
||||||
// Get the file's metadata from the URL
|
// Get the file's metadata from the URL
|
||||||
let meta = getOrigQueryParams(filePathURL.searchParams);
|
let meta = getOrigQueryParams(filePathURL.searchParams);
|
||||||
|
|
||||||
// If we don't have them (ex: the image came from Markdown, let's calculate them again)
|
// If we don't have them (ex: the image came from Markdown, let's calculate them again)
|
||||||
if (!meta) {
|
if (!meta) {
|
||||||
meta = await imageMetadata(filePathURL, file);
|
meta = await imageMetadata(filePathURL, buffer);
|
||||||
|
|
||||||
if (!meta) {
|
if (!meta) {
|
||||||
return next();
|
return next();
|
||||||
|
@ -98,11 +100,11 @@ export default function assets({
|
||||||
}
|
}
|
||||||
|
|
||||||
// if no transforms were added, the original file will be returned as-is
|
// if no transforms were added, the original file will be returned as-is
|
||||||
let data = file;
|
let data = buffer;
|
||||||
let format = meta.format;
|
let format = meta.format;
|
||||||
|
|
||||||
if (transform) {
|
if (transform) {
|
||||||
const result = await globalThis.astroAsset.imageService.transform(file, transform);
|
const result = await globalThis.astroAsset.imageService.transform(buffer, transform);
|
||||||
data = result.data;
|
data = result.data;
|
||||||
format = result.format;
|
format = result.format;
|
||||||
}
|
}
|
||||||
|
@ -201,6 +203,7 @@ export default function assets({
|
||||||
async load(id) {
|
async load(id) {
|
||||||
if (/\.(jpeg|jpg|png|tiff|webp|gif|svg)$/.test(id)) {
|
if (/\.(jpeg|jpg|png|tiff|webp|gif|svg)$/.test(id)) {
|
||||||
const meta = await emitESMImage(id, this.meta.watchMode, this.emitFile, settings);
|
const meta = await emitESMImage(id, this.meta.watchMode, this.emitFile, settings);
|
||||||
|
|
||||||
return `export default ${JSON.stringify(meta)}`;
|
return `export default ${JSON.stringify(meta)}`;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
|
@ -68,10 +68,11 @@ export function astroContentAssetPropagationPlugin({
|
||||||
const { stylesMap, urls } = await getStylesForURL(
|
const { stylesMap, urls } = await getStylesForURL(
|
||||||
pathToFileURL(basePath),
|
pathToFileURL(basePath),
|
||||||
devModuleLoader,
|
devModuleLoader,
|
||||||
'development'
|
'development',
|
||||||
|
settings.config
|
||||||
);
|
);
|
||||||
|
|
||||||
const hoistedScripts = await getScriptsForURL(pathToFileURL(basePath), devModuleLoader);
|
const hoistedScripts = await getScriptsForURL(pathToFileURL(basePath), devModuleLoader, settings.config);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
code: code
|
code: code
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import type { Plugin as VitePlugin } from 'vite';
|
import type { Plugin as VitePlugin } from 'vite';
|
||||||
import type { AstroSettings } from '../../../@types/astro';
|
import type { AstroSettings } from '../../../@types/astro';
|
||||||
import { viteID } from '../../util.js';
|
|
||||||
import type { BuildInternals } from '../internal.js';
|
import type { BuildInternals } from '../internal.js';
|
||||||
import { getPageDataByViteID } from '../internal.js';
|
import { getPageDataByViteID } from '../internal.js';
|
||||||
import { AstroBuildPlugin } from '../plugin';
|
import { AstroBuildPlugin } from '../plugin';
|
||||||
import { StaticBuildOptions } from '../types';
|
import { StaticBuildOptions } from '../types';
|
||||||
|
import File from '../../file/index.js';
|
||||||
|
|
||||||
function virtualHoistedEntry(id: string) {
|
function virtualHoistedEntry(id: string) {
|
||||||
return id.startsWith('/astro/hoisted.js?q=');
|
return id.startsWith('/astro/hoisted.js?q=');
|
||||||
|
@ -66,8 +66,8 @@ export function vitePluginHoistedScripts(
|
||||||
const facadeId = output.facadeModuleId!;
|
const facadeId = output.facadeModuleId!;
|
||||||
const pages = internals.hoistedScriptIdToPagesMap.get(facadeId)!;
|
const pages = internals.hoistedScriptIdToPagesMap.get(facadeId)!;
|
||||||
for (const pathname of pages) {
|
for (const pathname of pages) {
|
||||||
const vid = viteID(new URL('.' + pathname, settings.config.root));
|
const file = new File(pathname, settings.config);
|
||||||
const pageInfo = getPageDataByViteID(internals, vid);
|
const pageInfo = getPageDataByViteID(internals, file.toViteID());
|
||||||
if (pageInfo) {
|
if (pageInfo) {
|
||||||
if (canBeInlined) {
|
if (canBeInlined) {
|
||||||
pageInfo.hoistedScript = {
|
pageInfo.hoistedScript = {
|
||||||
|
|
120
packages/astro/src/core/file/index.ts
Normal file
120
packages/astro/src/core/file/index.ts
Normal file
|
@ -0,0 +1,120 @@
|
||||||
|
import type { AstroConfig } from '../../@types/astro';
|
||||||
|
import { pathToFileURL, fileURLToPath } from 'node:url';
|
||||||
|
import { normalizePath } from 'vite';
|
||||||
|
import { prependForwardSlash } from '../path.js';
|
||||||
|
import { unwrapId, viteID } from '../util.js';
|
||||||
|
|
||||||
|
export type PathType =
|
||||||
|
/**
|
||||||
|
* file:///Users/name/projects/todos/src/pages/index.astro
|
||||||
|
*/
|
||||||
|
'url' |
|
||||||
|
/**
|
||||||
|
* /Users/name/projects/todos/src/pages/index.astro
|
||||||
|
*/
|
||||||
|
'absolute' |
|
||||||
|
/**
|
||||||
|
* /@fs/Users/name/projects/todos/src/pages/index.astro
|
||||||
|
*/
|
||||||
|
'vite-fs-path' |
|
||||||
|
/**
|
||||||
|
* /src/pages/index.astro
|
||||||
|
*/
|
||||||
|
'root-relative' |
|
||||||
|
/**
|
||||||
|
* We don't know
|
||||||
|
*/
|
||||||
|
'unknown';
|
||||||
|
|
||||||
|
class File {
|
||||||
|
public raw: string;
|
||||||
|
public root: URL;
|
||||||
|
public type: PathType;
|
||||||
|
constructor(raw: string | URL, rootOrConfig: URL | Pick<AstroConfig, 'root'>) {
|
||||||
|
this.raw = typeof raw === 'string' ? raw : raw.toString();
|
||||||
|
this.root = rootOrConfig instanceof URL ? rootOrConfig : rootOrConfig.root;
|
||||||
|
this.type = File.getPathType(this.raw, this.root);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert a raw path to a File URL
|
||||||
|
*/
|
||||||
|
toFileURL(): URL {
|
||||||
|
switch(this.type) {
|
||||||
|
case 'url': return new URL(this.raw);
|
||||||
|
case 'absolute': return pathToFileURL(this.raw);
|
||||||
|
case 'vite-fs-path': {
|
||||||
|
const fsPath = this.raw.slice('/@fs'.length);
|
||||||
|
return pathToFileURL(fsPath);
|
||||||
|
}
|
||||||
|
case 'root-relative': {
|
||||||
|
return new URL('.' + this.raw, this.root);
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
throw new Error(`Cannot create file URL for an unknown path type: ${this.raw}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts to a path that is relative to the root, for use in browser paths
|
||||||
|
*/
|
||||||
|
toRootRelativePath() {
|
||||||
|
const url = this.toFileURL();
|
||||||
|
const id = unwrapId(viteID(url));
|
||||||
|
return prependForwardSlash(id.slice(normalizePath(fileURLToPath(this.root)).length));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts to the absolute (string) path. Uses the platform's path separator.
|
||||||
|
*/
|
||||||
|
toAbsolutePath() {
|
||||||
|
return fileURLToPath(this.toFileURL());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts to a path for use in browser, contains the `/@fs` prefix understood by Vite.
|
||||||
|
*/
|
||||||
|
toViteFSPath() {
|
||||||
|
const abs = prependForwardSlash(normalizePath(this.toAbsolutePath()));
|
||||||
|
return '/@fs' + abs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts to an id that is used as the key in Vite's module graph
|
||||||
|
*/
|
||||||
|
toViteID() {
|
||||||
|
return viteID(this.toFileURL());
|
||||||
|
}
|
||||||
|
|
||||||
|
static getPathType(raw: string, root: URL): PathType {
|
||||||
|
if(raw.startsWith('/@fs')) {
|
||||||
|
return 'vite-fs-path';
|
||||||
|
}
|
||||||
|
|
||||||
|
if(raw.startsWith('/')) {
|
||||||
|
const normalRoot = normalizePath(fileURLToPath(root))
|
||||||
|
if(raw.startsWith(normalRoot)) {
|
||||||
|
return 'absolute';
|
||||||
|
} else {
|
||||||
|
return 'root-relative';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(raw.startsWith('file://')) {
|
||||||
|
return 'url';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Windows drive
|
||||||
|
if(/[A-Z]:/.test(raw)) {
|
||||||
|
return 'absolute';
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'unknown';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
File,
|
||||||
|
File as default
|
||||||
|
}
|
|
@ -1,7 +1,7 @@
|
||||||
import type { ModuleLoader } from '../../module-loader/index';
|
import type { ModuleLoader } from '../../module-loader/index';
|
||||||
|
import type { AstroConfig, RuntimeMode } from '../../../@types/astro.js';
|
||||||
|
|
||||||
import { RuntimeMode } from '../../../@types/astro.js';
|
import File from '../../file/index.js';
|
||||||
import { viteID } from '../../util.js';
|
|
||||||
import { isBuildableCSSRequest } from './util.js';
|
import { isBuildableCSSRequest } from './util.js';
|
||||||
import { crawlGraph } from './vite.js';
|
import { crawlGraph } from './vite.js';
|
||||||
|
|
||||||
|
@ -9,12 +9,14 @@ import { crawlGraph } from './vite.js';
|
||||||
export async function getStylesForURL(
|
export async function getStylesForURL(
|
||||||
filePath: URL,
|
filePath: URL,
|
||||||
loader: ModuleLoader,
|
loader: ModuleLoader,
|
||||||
mode: RuntimeMode
|
mode: RuntimeMode,
|
||||||
|
config: AstroConfig,
|
||||||
): Promise<{ urls: Set<string>; stylesMap: Map<string, string> }> {
|
): Promise<{ urls: Set<string>; stylesMap: Map<string, string> }> {
|
||||||
const importedCssUrls = new Set<string>();
|
const importedCssUrls = new Set<string>();
|
||||||
const importedStylesMap = new Map<string, string>();
|
const importedStylesMap = new Map<string, string>();
|
||||||
|
|
||||||
for await (const importedModule of crawlGraph(loader, viteID(filePath), true)) {
|
const file = new File(filePath, config);
|
||||||
|
for await (const importedModule of crawlGraph(loader, file.toViteID(), true)) {
|
||||||
if (isBuildableCSSRequest(importedModule.url)) {
|
if (isBuildableCSSRequest(importedModule.url)) {
|
||||||
let ssrModule: Record<string, any>;
|
let ssrModule: Record<string, any>;
|
||||||
try {
|
try {
|
||||||
|
|
|
@ -76,7 +76,7 @@ interface GetScriptsAndStylesParams {
|
||||||
|
|
||||||
async function getScriptsAndStyles({ env, filePath }: GetScriptsAndStylesParams) {
|
async function getScriptsAndStyles({ env, filePath }: GetScriptsAndStylesParams) {
|
||||||
// Add hoisted script tags
|
// Add hoisted script tags
|
||||||
const scripts = await getScriptsForURL(filePath, env.loader);
|
const scripts = await getScriptsForURL(filePath, env.loader, env.settings.config);
|
||||||
|
|
||||||
// Inject HMR scripts
|
// Inject HMR scripts
|
||||||
if (isPage(filePath, env.settings) && env.mode === 'development') {
|
if (isPage(filePath, env.settings) && env.mode === 'development') {
|
||||||
|
@ -109,7 +109,7 @@ async function getScriptsAndStyles({ env, filePath }: GetScriptsAndStylesParams)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pass framework CSS in as style tags to be appended to the page.
|
// Pass framework CSS in as style tags to be appended to the page.
|
||||||
const { urls: styleUrls, stylesMap } = await getStylesForURL(filePath, env.loader, env.mode);
|
const { urls: styleUrls, stylesMap } = await getStylesForURL(filePath, env.loader, env.mode, env.settings.config);
|
||||||
let links = new Set<SSRElement>();
|
let links = new Set<SSRElement>();
|
||||||
[...styleUrls].forEach((href) => {
|
[...styleUrls].forEach((href) => {
|
||||||
links.add({
|
links.add({
|
||||||
|
|
|
@ -1,17 +1,18 @@
|
||||||
import type { SSRElement } from '../../../@types/astro';
|
import type { AstroConfig, SSRElement } from '../../../@types/astro';
|
||||||
import type { PluginMetadata as AstroPluginMetadata } from '../../../vite-plugin-astro/types';
|
import type { PluginMetadata as AstroPluginMetadata } from '../../../vite-plugin-astro/types';
|
||||||
import type { ModuleInfo, ModuleLoader } from '../../module-loader/index';
|
import type { ModuleInfo, ModuleLoader } from '../../module-loader/index';
|
||||||
|
|
||||||
import { viteID } from '../../util.js';
|
import File from '../../file/index.js';
|
||||||
import { createModuleScriptElementWithSrc } from '../ssr-element.js';
|
import { createModuleScriptElementWithSrc } from '../ssr-element.js';
|
||||||
import { crawlGraph } from './vite.js';
|
import { crawlGraph } from './vite.js';
|
||||||
|
|
||||||
export async function getScriptsForURL(
|
export async function getScriptsForURL(
|
||||||
filePath: URL,
|
filePath: URL,
|
||||||
loader: ModuleLoader
|
loader: ModuleLoader,
|
||||||
|
config: AstroConfig,
|
||||||
): Promise<Set<SSRElement>> {
|
): Promise<Set<SSRElement>> {
|
||||||
const elements = new Set<SSRElement>();
|
const elements = new Set<SSRElement>();
|
||||||
const rootID = viteID(filePath);
|
const rootID = new File(filePath, config).toViteID();
|
||||||
const modInfo = loader.getModuleInfo(rootID);
|
const modInfo = loader.getModuleInfo(rootID);
|
||||||
addHoistedScripts(elements, modInfo);
|
addHoistedScripts(elements, modInfo);
|
||||||
for await (const moduleNode of crawlGraph(loader, rootID, true)) {
|
for await (const moduleNode of crawlGraph(loader, rootID, true)) {
|
||||||
|
|
|
@ -151,16 +151,6 @@ export function relativeToSrcDir(config: AstroConfig, idOrUrl: URL | string) {
|
||||||
return id.slice(slash(fileURLToPath(config.srcDir)).length);
|
return id.slice(slash(fileURLToPath(config.srcDir)).length);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function rootRelativePath(config: AstroConfig, idOrUrl: URL | string) {
|
|
||||||
let id: string;
|
|
||||||
if (typeof idOrUrl !== 'string') {
|
|
||||||
id = unwrapId(viteID(idOrUrl));
|
|
||||||
} else {
|
|
||||||
id = idOrUrl;
|
|
||||||
}
|
|
||||||
return prependForwardSlash(id.slice(normalizePath(fileURLToPath(config.root)).length));
|
|
||||||
}
|
|
||||||
|
|
||||||
export function emoji(char: string, fallback: string) {
|
export function emoji(char: string, fallback: string) {
|
||||||
return process.platform !== 'win32' ? char : fallback;
|
return process.platform !== 'win32' ? char : fallback;
|
||||||
}
|
}
|
||||||
|
|
96
packages/astro/test/units/file/file.test.js
Normal file
96
packages/astro/test/units/file/file.test.js
Normal file
|
@ -0,0 +1,96 @@
|
||||||
|
import { expect } from 'chai';
|
||||||
|
import { fileURLToPath, pathToFileURL } from 'node:url';
|
||||||
|
|
||||||
|
import File from '../../../dist/core/file/index.js';
|
||||||
|
|
||||||
|
|
||||||
|
describe('File abstraction', () => {
|
||||||
|
describe('constructor', () => {
|
||||||
|
it('can take a file url', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let file = new File(import.meta.url, root);
|
||||||
|
expect(file.type).to.equal('url');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('can take an absolute path', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = fileURLToPath(import.meta.url);
|
||||||
|
let file = new File(path, root);
|
||||||
|
expect(file.type).to.equal('absolute');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('can take a Vite fs path', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = '/@fs/some/path';
|
||||||
|
let file = new File(path, root);
|
||||||
|
expect(file.type).to.equal('vite-fs-path');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('can take a path relative to the root', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = '/src/pages/index.astro';
|
||||||
|
let file = new File(path, root);
|
||||||
|
expect(file.type).to.equal('root-relative');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Defaults to unknown', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = 'some-virtual-id';
|
||||||
|
let file = new File(path, root);
|
||||||
|
expect(file.type).to.equal('unknown');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can take an AstroConfig as the second argument', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = fileURLToPath(import.meta.url);
|
||||||
|
let file = new File(path, { root });
|
||||||
|
expect(file.type).to.equal('absolute');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('toFileURL', () => {
|
||||||
|
it('converts from a file URL', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let file = new File(import.meta.url, root);
|
||||||
|
let url = file.toFileURL();
|
||||||
|
expect(url.toString()).to.equal(import.meta.url);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('converts from an absolute path', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = fileURLToPath(import.meta.url);
|
||||||
|
let file = new File(path, root);
|
||||||
|
let url = file.toFileURL();
|
||||||
|
expect(url.toString()).to.equal(pathToFileURL(path).toString());
|
||||||
|
});
|
||||||
|
|
||||||
|
it('converts from an Vite fs path', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = '/@fs/some/path';
|
||||||
|
let file = new File(path, root);
|
||||||
|
expect(file.toFileURL().toString()).to.equal('file:///some/path');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('converts from a path relative to the root', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = '/src/pages/index.astro';
|
||||||
|
let file = new File(path, root);
|
||||||
|
|
||||||
|
let expected = new URL('.' + path, root).toString();
|
||||||
|
expect(file.toFileURL().toString()).to.equal(expected);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Throws converting an unknown', () => {
|
||||||
|
let root = new URL('./', import.meta.url);
|
||||||
|
let path = 'some-virtual-id';
|
||||||
|
let file = new File(path, root);
|
||||||
|
try {
|
||||||
|
file.toFileURL();
|
||||||
|
expect(false).to.equal(true);
|
||||||
|
} catch(err) {
|
||||||
|
expect(err).to.be.an.instanceOf(Error);
|
||||||
|
expect(err.message.startsWith('Cannot create file URL')).to.equal(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})
|
||||||
|
});
|
Loading…
Reference in a new issue