2020-01-21 09:01:55 -06:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2020-01-09 01:17:44 +11:00
|
|
|
|
|
|
|
import {
|
|
|
|
MediaType,
|
|
|
|
SourceFile,
|
|
|
|
SourceFileJson
|
|
|
|
} from "./compiler_sourcefile.ts";
|
2020-03-05 13:05:41 +01:00
|
|
|
import { normalizeString, CHAR_FORWARD_SLASH } from "./compiler_util.ts";
|
2020-01-09 01:17:44 +11:00
|
|
|
import { cwd } from "./dir.ts";
|
|
|
|
import { sendAsync, sendSync } from "./dispatch_json.ts";
|
|
|
|
import { assert } from "./util.ts";
|
|
|
|
import * as util from "./util.ts";
|
|
|
|
|
|
|
|
/** Resolve a path to the final path segment passed. */
|
|
|
|
function resolvePath(...pathSegments: string[]): string {
|
|
|
|
let resolvedPath = "";
|
|
|
|
let resolvedAbsolute = false;
|
|
|
|
|
|
|
|
for (let i = pathSegments.length - 1; i >= -1 && !resolvedAbsolute; i--) {
|
|
|
|
let path: string;
|
|
|
|
|
|
|
|
if (i >= 0) path = pathSegments[i];
|
|
|
|
else path = cwd();
|
|
|
|
|
|
|
|
// Skip empty entries
|
|
|
|
if (path.length === 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
resolvedPath = `${path}/${resolvedPath}`;
|
2020-03-05 13:05:41 +01:00
|
|
|
resolvedAbsolute = path.charCodeAt(0) === CHAR_FORWARD_SLASH;
|
2020-01-09 01:17:44 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// At this point the path should be resolved to a full absolute path, but
|
|
|
|
// handle relative paths to be safe (might happen when cwd() fails)
|
|
|
|
|
|
|
|
// Normalize the path
|
2020-03-05 13:05:41 +01:00
|
|
|
resolvedPath = normalizeString(
|
2020-01-09 01:17:44 +11:00
|
|
|
resolvedPath,
|
|
|
|
!resolvedAbsolute,
|
|
|
|
"/",
|
2020-03-05 13:05:41 +01:00
|
|
|
code => code === CHAR_FORWARD_SLASH
|
2020-01-09 01:17:44 +11:00
|
|
|
);
|
|
|
|
|
|
|
|
if (resolvedAbsolute) {
|
|
|
|
if (resolvedPath.length > 0) return `/${resolvedPath}`;
|
|
|
|
else return "/";
|
|
|
|
} else if (resolvedPath.length > 0) return resolvedPath;
|
|
|
|
else return ".";
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Resolve a relative specifier based on the referrer. Used when resolving
|
|
|
|
* modules internally within the runtime compiler API. */
|
|
|
|
function resolveSpecifier(specifier: string, referrer: string): string {
|
|
|
|
if (!specifier.startsWith(".")) {
|
|
|
|
return specifier;
|
|
|
|
}
|
|
|
|
const pathParts = referrer.split("/");
|
|
|
|
pathParts.pop();
|
|
|
|
let path = pathParts.join("/");
|
|
|
|
path = path.endsWith("/") ? path : `${path}/`;
|
|
|
|
return resolvePath(path, specifier);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Ops to Rust to resolve modules' URLs. */
|
|
|
|
export function resolveModules(
|
|
|
|
specifiers: string[],
|
|
|
|
referrer?: string
|
|
|
|
): string[] {
|
|
|
|
util.log("compiler_imports::resolveModules", { specifiers, referrer });
|
2020-02-25 09:14:27 -05:00
|
|
|
return sendSync("op_resolve_modules", { specifiers, referrer });
|
2020-01-09 01:17:44 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Ops to Rust to fetch modules meta data. */
|
|
|
|
function fetchSourceFiles(
|
|
|
|
specifiers: string[],
|
|
|
|
referrer?: string
|
|
|
|
): Promise<SourceFileJson[]> {
|
|
|
|
util.log("compiler_imports::fetchSourceFiles", { specifiers, referrer });
|
2020-02-25 09:14:27 -05:00
|
|
|
return sendAsync("op_fetch_source_files", {
|
2020-01-09 01:17:44 +11:00
|
|
|
specifiers,
|
|
|
|
referrer
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Given a filename, determine the media type based on extension. Used when
|
|
|
|
* resolving modules internally in a runtime compile. */
|
|
|
|
function getMediaType(filename: string): MediaType {
|
|
|
|
const maybeExtension = /\.([a-zA-Z]+)$/.exec(filename);
|
|
|
|
if (!maybeExtension) {
|
|
|
|
util.log(`!!! Could not identify valid extension: "${filename}"`);
|
|
|
|
return MediaType.Unknown;
|
|
|
|
}
|
|
|
|
const [, extension] = maybeExtension;
|
|
|
|
switch (extension.toLowerCase()) {
|
|
|
|
case "js":
|
|
|
|
return MediaType.JavaScript;
|
|
|
|
case "jsx":
|
|
|
|
return MediaType.JSX;
|
|
|
|
case "json":
|
|
|
|
return MediaType.Json;
|
|
|
|
case "ts":
|
|
|
|
return MediaType.TypeScript;
|
|
|
|
case "tsx":
|
|
|
|
return MediaType.TSX;
|
|
|
|
case "wasm":
|
|
|
|
return MediaType.Wasm;
|
|
|
|
default:
|
|
|
|
util.log(`!!! Unknown extension: "${extension}"`);
|
|
|
|
return MediaType.Unknown;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Recursively process the imports of modules from within the supplied sources,
|
|
|
|
* generating `SourceFile`s of any imported files.
|
|
|
|
*
|
|
|
|
* Specifiers are supplied in an array of tuples where the first is the
|
|
|
|
* specifier that will be requested in the code and the second is the specifier
|
|
|
|
* that should be actually resolved. */
|
|
|
|
export function processLocalImports(
|
|
|
|
sources: Record<string, string>,
|
|
|
|
specifiers: Array<[string, string]>,
|
2020-02-20 14:58:05 +11:00
|
|
|
referrer?: string,
|
2020-03-03 08:18:27 +11:00
|
|
|
processJsImports = false
|
2020-01-09 01:17:44 +11:00
|
|
|
): string[] {
|
|
|
|
if (!specifiers.length) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
const moduleNames = specifiers.map(
|
|
|
|
referrer
|
|
|
|
? ([, specifier]): string => resolveSpecifier(specifier, referrer)
|
|
|
|
: ([, specifier]): string => specifier
|
|
|
|
);
|
|
|
|
for (let i = 0; i < moduleNames.length; i++) {
|
|
|
|
const moduleName = moduleNames[i];
|
|
|
|
assert(moduleName in sources, `Missing module in sources: "${moduleName}"`);
|
|
|
|
const sourceFile =
|
|
|
|
SourceFile.get(moduleName) ||
|
|
|
|
new SourceFile({
|
|
|
|
url: moduleName,
|
|
|
|
filename: moduleName,
|
|
|
|
sourceCode: sources[moduleName],
|
|
|
|
mediaType: getMediaType(moduleName)
|
|
|
|
});
|
|
|
|
sourceFile.cache(specifiers[i][0], referrer);
|
|
|
|
if (!sourceFile.processed) {
|
2020-02-20 14:58:05 +11:00
|
|
|
processLocalImports(
|
|
|
|
sources,
|
2020-03-03 08:18:27 +11:00
|
|
|
sourceFile.imports(processJsImports),
|
2020-02-20 14:58:05 +11:00
|
|
|
sourceFile.url,
|
2020-03-03 08:18:27 +11:00
|
|
|
processJsImports
|
2020-02-20 14:58:05 +11:00
|
|
|
);
|
2020-01-09 01:17:44 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return moduleNames;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Recursively process the imports of modules, generating `SourceFile`s of any
|
|
|
|
* imported files.
|
|
|
|
*
|
|
|
|
* Specifiers are supplied in an array of tuples where the first is the
|
|
|
|
* specifier that will be requested in the code and the second is the specifier
|
|
|
|
* that should be actually resolved. */
|
|
|
|
export async function processImports(
|
|
|
|
specifiers: Array<[string, string]>,
|
2020-02-20 14:58:05 +11:00
|
|
|
referrer?: string,
|
2020-03-03 08:18:27 +11:00
|
|
|
processJsImports = false
|
2020-01-09 01:17:44 +11:00
|
|
|
): Promise<string[]> {
|
|
|
|
if (!specifiers.length) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
const sources = specifiers.map(([, moduleSpecifier]) => moduleSpecifier);
|
|
|
|
const resolvedSources = resolveModules(sources, referrer);
|
|
|
|
const sourceFiles = await fetchSourceFiles(resolvedSources, referrer);
|
|
|
|
assert(sourceFiles.length === specifiers.length);
|
|
|
|
for (let i = 0; i < sourceFiles.length; i++) {
|
|
|
|
const sourceFileJson = sourceFiles[i];
|
|
|
|
const sourceFile =
|
|
|
|
SourceFile.get(sourceFileJson.url) || new SourceFile(sourceFileJson);
|
|
|
|
sourceFile.cache(specifiers[i][0], referrer);
|
|
|
|
if (!sourceFile.processed) {
|
2020-02-20 14:58:05 +11:00
|
|
|
await processImports(
|
2020-03-03 08:18:27 +11:00
|
|
|
sourceFile.imports(processJsImports),
|
2020-02-20 14:58:05 +11:00
|
|
|
sourceFile.url,
|
2020-03-03 08:18:27 +11:00
|
|
|
processJsImports
|
2020-02-20 14:58:05 +11:00
|
|
|
);
|
2020-01-09 01:17:44 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return resolvedSources;
|
|
|
|
}
|