Skip to content

Commit

Permalink
refactor(environment): remove IS_NODE_ENV and delete dead code
Browse files Browse the repository at this point in the history
This removes the `IS_NODE_ENV` variable from our environment module
which previously was used to switch between browser-specific and
node-specific approaches to various things in the Stencil compilation
process.

Following the merge of #4317 we only support one runtime, node, for
compiling Stencil components, so we can assume that `IS_NODE_ENV` is
`true` and delete code accordingly.
  • Loading branch information
alicewriteswrongs committed Jul 7, 2023
1 parent d9e4e1a commit 52cfa4e
Show file tree
Hide file tree
Showing 7 changed files with 72 additions and 170 deletions.
3 changes: 1 addition & 2 deletions src/compiler/build/build-finish.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@ import { isFunction, isRemoteUrl } from '@utils';
import { relative } from 'path';

import type * as d from '../../declarations';
import { IS_NODE_ENV } from '../sys/environment';
import { generateBuildResults } from './build-results';
import { generateBuildStats, writeBuildStats } from './build-stats';

Expand Down Expand Up @@ -122,7 +121,7 @@ const buildDone = async (

if (!config.watch) {
compilerCtx.reset();
if (IS_NODE_ENV && global.gc) {
if (global.gc) {
buildCtx.debug(`triggering forced gc`);
global.gc();
buildCtx.debug(`forced gc finished`);
Expand Down
99 changes: 22 additions & 77 deletions src/compiler/config/load-config.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,8 @@
import { createNodeSys } from '@sys-api-node';
import { buildError, catchError, hasError, isString, normalizePath } from '@utils';
import { dirname } from 'path';
import ts from 'typescript';

import type {
CompilerSystem,
Diagnostic,
LoadConfigInit,
LoadConfigResults,
UnvalidatedConfig,
} from '../../declarations';
import { IS_NODE_ENV } from '../sys/environment';

import type { Diagnostic, LoadConfigInit, LoadConfigResults, UnvalidatedConfig } from '../../declarations';
import { nodeRequire } from '../sys/node-require';
import { validateTsConfig } from '../sys/typescript/typescript-config';
import { validateConfig } from './validate-config';
Expand Down Expand Up @@ -55,7 +47,7 @@ export const loadConfig = async (init: LoadConfigInit = {}): Promise<LoadConfigR
// attached to a configuration entity, validated or otherwise)
const sys = init.sys ?? createNodeSys();

const loadedConfigFile = await loadConfigFile(sys, results.diagnostics, configPath);
const loadedConfigFile = await loadConfigFile(results.diagnostics, configPath);
if (hasError(results.diagnostics)) {
return results;
}
Expand Down Expand Up @@ -116,23 +108,21 @@ export const loadConfig = async (init: LoadConfigInit = {}): Promise<LoadConfigR

/**
* Load a Stencil configuration file from disk
* @param sys the underlying System entity to use to interact with the operating system
* @param diagnostics a series of diagnostics used to track errors & warnings throughout the loading process. Entries
* may be added to this list in the event of an error.
*
* @param diagnostics a series of diagnostics used to track errors & warnings
* throughout the loading process. Entries may be added to this list in the
* event of an error.
* @param configPath the path to the configuration file to load
* @returns an unvalidated configuration. In the event of an error, additional diagnostics may be pushed to the
* provided `diagnostics` argument and `null` will be returned.
* @returns an unvalidated configuration. In the event of an error, additional
* diagnostics may be pushed to the provided `diagnostics` argument and `null`
* will be returned.
*/
const loadConfigFile = async (
sys: CompilerSystem,
diagnostics: Diagnostic[],
configPath: string
): Promise<UnvalidatedConfig | null> => {
const loadConfigFile = async (diagnostics: Diagnostic[], configPath: string): Promise<UnvalidatedConfig | null> => {
let config: UnvalidatedConfig | null = null;

if (isString(configPath)) {
// the passed in config was a string, so it's probably a path to the config we need to load
const configFileData = await evaluateConfigFile(sys, diagnostics, configPath);
const configFileData = await evaluateConfigFile(diagnostics, configPath);
if (hasError(diagnostics)) {
return config;
}
Expand All @@ -152,73 +142,28 @@ const loadConfigFile = async (

/**
* Load the configuration file, based on the environment that Stencil is being run in
* @param sys the underlying System entity to use to interact with the operating system
* @param diagnostics a series of diagnostics used to track errors & warnings throughout the loading process. Entries
* may be added to this list in the event of an error.
*
* @param diagnostics a series of diagnostics used to track errors & warnings
* throughout the loading process. Entries may be added to this list in the
* event of an error.
* @param configFilePath the path to the configuration file to load
* @returns an unvalidated configuration. In the event of an error, additional diagnostics may be pushed to the
* provided `diagnostics` argument and `null` will be returned.
* @returns an unvalidated configuration. In the event of an error, additional
* diagnostics may be pushed to the provided `diagnostics` argument and `null`
* will be returned.
*/
const evaluateConfigFile = async (
sys: CompilerSystem,
diagnostics: Diagnostic[],
configFilePath: string
): Promise<{ config?: UnvalidatedConfig } | null> => {
let configFileData: { config?: UnvalidatedConfig } | null = null;

try {
if (IS_NODE_ENV) {
const results = nodeRequire(configFilePath);
diagnostics.push(...results.diagnostics);
configFileData = results.module;
} else {
// browser environment, can't use node's require() to evaluate
let sourceText = await sys.readFile(configFilePath);
sourceText = transpileTypedConfig(diagnostics, sourceText, configFilePath);
if (hasError(diagnostics)) {
return configFileData;
}

const evalConfig = new Function(`const exports = {}; ${sourceText}; return exports;`);
configFileData = evalConfig();
}
const results = nodeRequire(configFilePath);
diagnostics.push(...results.diagnostics);
configFileData = results.module;
} catch (e: any) {
catchError(diagnostics, e);
}

return configFileData;
};

/**
* Transpiles the provided TypeScript source text into JavaScript.
*
* This function is intended to be used on a `stencil.config.ts` file
*
* @param diagnostics a collection of compiler diagnostics to check as a part of the compilation process
* @param sourceText the text to transpile
* @param filePath the name of the file to transpile
* @returns the transpiled text. If there are any diagnostics in the provided collection, the provided source is returned
*/
const transpileTypedConfig = (diagnostics: Diagnostic[], sourceText: string, filePath: string): string => {
// let's transpile an awesome stencil.config.ts file into
// a boring stencil.config.js file
if (hasError(diagnostics)) {
return sourceText;
}

const opts: ts.TranspileOptions = {
fileName: filePath,
compilerOptions: {
module: ts.ModuleKind.CommonJS,
moduleResolution: ts.ModuleResolutionKind.NodeJs,
esModuleInterop: true,
target: ts.ScriptTarget.ES2015,
allowJs: true,
},
reportDiagnostics: false,
};

const output = ts.transpileModule(sourceText, opts);

return output.outputText;
};
5 changes: 1 addition & 4 deletions src/compiler/optimize/autoprefixer.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { Postcss } from 'postcss';

import type * as d from '../../declarations';
import { IS_NODE_ENV, requireFunc } from '../sys/environment';
import { requireFunc } from '../sys/environment';

type CssProcessor = ReturnType<Postcss>;
let cssProcessor: CssProcessor;
Expand All @@ -22,9 +22,6 @@ export const autoprefixCss = async (cssText: string, opts: boolean | null | d.Au
output: cssText,
diagnostics: [],
};
if (!IS_NODE_ENV) {
return output;
}

try {
const autoprefixerOpts = opts != null && typeof opts === 'object' ? opts : DEFAULT_AUTOPREFIX_OPTIONS;
Expand Down
13 changes: 2 additions & 11 deletions src/compiler/sys/environment.ts
Original file line number Diff line number Diff line change
@@ -1,13 +1,4 @@
export const IS_NODE_ENV =
typeof global !== 'undefined' &&
typeof require === 'function' &&
!!global.process &&
typeof __filename === 'string' &&
(!(global as any as Window).origin || typeof (global as any as Window).origin !== 'string');

export const OS_PLATFORM = IS_NODE_ENV ? process.platform : '';

export const IS_WINDOWS_ENV = OS_PLATFORM === 'win32';
export const IS_WINDOWS_ENV = process.platform === 'win32';

export const IS_CASE_SENSITIVE_FILE_NAMES = !IS_WINDOWS_ENV;

Expand All @@ -21,4 +12,4 @@ export const HAS_WEB_WORKER = IS_BROWSER_ENV && typeof Worker === 'function';

export const IS_FETCH_ENV = typeof fetch === 'function';

export const requireFunc = IS_NODE_ENV ? require : () => {};
export const requireFunc = require;
93 changes: 45 additions & 48 deletions src/compiler/sys/node-require.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@ import { catchError, loadTypeScriptDiagnostic } from '@utils';
import ts from 'typescript';

import type { Diagnostic } from '../../declarations';
import { IS_NODE_ENV } from './environment';

export const nodeRequire = (id: string) => {
const results = {
Expand All @@ -11,61 +10,59 @@ export const nodeRequire = (id: string) => {
diagnostics: [] as Diagnostic[],
};

if (IS_NODE_ENV) {
try {
const fs: typeof import('fs') = require('fs');
const path: typeof import('path') = require('path');
try {
const fs: typeof import('fs') = require('fs');
const path: typeof import('path') = require('path');

results.id = path.resolve(id);
results.id = path.resolve(id);

// ensure we cleared out node's internal require() cache for this file
delete require.cache[results.id];
// ensure we cleared out node's internal require() cache for this file
delete require.cache[results.id];

// let's override node's require for a second
// don't worry, we'll revert this when we're done
require.extensions['.ts'] = (module: NodeJS.Module, fileName: string) => {
let sourceText = fs.readFileSync(fileName, 'utf8');
// let's override node's require for a second
// don't worry, we'll revert this when we're done
require.extensions['.ts'] = (module: NodeJS.Module, fileName: string) => {
let sourceText = fs.readFileSync(fileName, 'utf8');

if (fileName.endsWith('.ts')) {
// looks like we've got a typed config file
// let's transpile it to .js quick
const tsResults = ts.transpileModule(sourceText, {
fileName,
compilerOptions: {
module: ts.ModuleKind.CommonJS,
moduleResolution: ts.ModuleResolutionKind.NodeJs,
esModuleInterop: true,
target: ts.ScriptTarget.ES2017,
allowJs: true,
},
});
sourceText = tsResults.outputText;
if (fileName.endsWith('.ts')) {
// looks like we've got a typed config file
// let's transpile it to .js quick
const tsResults = ts.transpileModule(sourceText, {
fileName,
compilerOptions: {
module: ts.ModuleKind.CommonJS,
moduleResolution: ts.ModuleResolutionKind.NodeJs,
esModuleInterop: true,
target: ts.ScriptTarget.ES2017,
allowJs: true,
},
});
sourceText = tsResults.outputText;

results.diagnostics.push(...tsResults.diagnostics.map(loadTypeScriptDiagnostic));
} else {
// quick hack to turn a modern es module
// into and old school commonjs module
sourceText = sourceText.replace(/export\s+\w+\s+(\w+)/gm, 'exports.$1');
}
results.diagnostics.push(...tsResults.diagnostics.map(loadTypeScriptDiagnostic));
} else {
// quick hack to turn a modern es module
// into and old school commonjs module
sourceText = sourceText.replace(/export\s+\w+\s+(\w+)/gm, 'exports.$1');
}

try {
// we need to coerce because of the requirements for the arguments to
// this function. It's safe enough since it's already wrapped in a
// `try { } catch`.
(module as NodeModuleWithCompile)._compile(sourceText, fileName);
} catch (e: any) {
catchError(results.diagnostics, e);
}
};
try {
// we need to coerce because of the requirements for the arguments to
// this function. It's safe enough since it's already wrapped in a
// `try { } catch`.
(module as NodeModuleWithCompile)._compile(sourceText, fileName);
} catch (e: any) {
catchError(results.diagnostics, e);
}
};

// let's do this!
results.module = require(results.id);
// let's do this!
results.module = require(results.id);

// all set, let's go ahead and reset the require back to the default
require.extensions['.ts'] = undefined;
} catch (e: any) {
catchError(results.diagnostics, e);
}
// all set, let's go ahead and reset the require back to the default
require.extensions['.ts'] = undefined;
} catch (e: any) {
catchError(results.diagnostics, e);
}

return results;
Expand Down
27 changes: 1 addition & 26 deletions src/compiler/sys/typescript/typescript-resolve-module.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,9 @@
import { isRemoteUrl, isString, normalizePath } from '@utils';
import { isString, normalizePath } from '@utils';
import { basename, dirname, isAbsolute, join, resolve } from 'path';
import ts from 'typescript';

import type * as d from '../../../declarations';
import { version } from '../../../version';
import { IS_BROWSER_ENV, IS_NODE_ENV } from '../environment';
import { InMemoryFileSystem } from '../in-memory-fs';
import { resolveRemoteModuleIdSync } from '../resolve/resolve-module-sync';
import {
Expand All @@ -19,28 +18,6 @@ import {
} from '../resolve/resolve-utils';
import { patchTsSystemFileSystem } from './typescript-sys';

// TODO(STENCIL-728): fix typing of `inMemoryFs` parameter in `patchTypescript`, related functions
export const patchTypeScriptResolveModule = (config: d.Config, inMemoryFs: InMemoryFileSystem) => {
let compilerExe: string;
if (config.sys) {
compilerExe = config.sys.getCompilerExecutingPath();
} else if (IS_BROWSER_ENV) {
compilerExe = location.href;
}

if (shouldPatchRemoteTypeScript(compilerExe)) {
const resolveModuleName = ((ts as any).__resolveModuleName = ts.resolveModuleName);

ts.resolveModuleName = (moduleName, containingFile, compilerOptions, host, cache, redirectedReference) => {
const resolvedModule = patchedTsResolveModule(config, inMemoryFs, moduleName, containingFile);
if (resolvedModule) {
return resolvedModule;
}
return resolveModuleName(moduleName, containingFile, compilerOptions, host, cache, redirectedReference);
};
}
};

export const tsResolveModuleName = (
config: d.Config,
compilerCtx: d.CompilerCtx,
Expand Down Expand Up @@ -235,5 +212,3 @@ const getTsResolveExtension = (p: string) => {
}
return ts.Extension.Ts;
};

const shouldPatchRemoteTypeScript = (compilerExe: string) => !IS_NODE_ENV && isRemoteUrl(compilerExe);
2 changes: 0 additions & 2 deletions src/compiler/sys/typescript/typescript-sys.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@ import type * as d from '../../../declarations';
import { IS_CASE_SENSITIVE_FILE_NAMES, IS_WEB_WORKER_ENV } from '../environment';
import { fetchUrlSync } from '../fetch/fetch-module-sync';
import { InMemoryFileSystem } from '../in-memory-fs';
import { patchTypeScriptResolveModule } from './typescript-resolve-module';

// TODO(STENCIL-728): fix typing of `inMemoryFs` parameter in `patchTypescript`, related functions
export const patchTsSystemFileSystem = (
Expand Down Expand Up @@ -197,7 +196,6 @@ export const patchTypescript = (config: d.Config, inMemoryFs: InMemoryFileSystem
patchTsSystemFileSystem(config, config.sys, inMemoryFs, ts.sys);
patchTsSystemWatch(config.sys, ts.sys);
}
patchTypeScriptResolveModule(config, inMemoryFs);
(ts as any).__patched = true;
}
};
Expand Down

0 comments on commit 52cfa4e

Please sign in to comment.