From 7b55d3d43edbe491ed7162075f034ffdb1adbd45 Mon Sep 17 00:00:00 2001 From: Drew Powers <1369770+drwpow@users.noreply.github.com> Date: Thu, 6 May 2021 12:57:14 -0600 Subject: [PATCH] Chore: fix lint rules, add more error handling for unexpected cases (#177) --- .changeset/happy-chicken-whisper.md | 5 ++++ packages/astro-parser/src/parse/state/tag.ts | 2 -- packages/astro/src/ast.ts | 8 ++---- packages/astro/src/build.ts | 7 ++--- packages/astro/src/build/bundle/js.ts | 5 ++-- packages/astro/src/build/page.ts | 28 ++++++++++--------- packages/astro/src/cli.ts | 4 +-- packages/astro/src/compiler/codegen/index.ts | 20 ++++++------- .../astro/src/compiler/transform/doctype.ts | 3 +- .../astro/src/compiler/transform/index.ts | 6 ++-- .../astro/src/compiler/transform/styles.ts | 19 ++++++------- packages/astro/src/logger.ts | 13 ++++----- packages/astro/test/.eslintrc.cjs | 5 ++++ .../astro/test/snowpack-integration.test.js | 2 -- .../src/plugins/PluginHost.ts | 1 - 15 files changed, 63 insertions(+), 65 deletions(-) create mode 100644 .changeset/happy-chicken-whisper.md create mode 100644 packages/astro/test/.eslintrc.cjs diff --git a/.changeset/happy-chicken-whisper.md b/.changeset/happy-chicken-whisper.md new file mode 100644 index 0000000000..326e1e501d --- /dev/null +++ b/.changeset/happy-chicken-whisper.md @@ -0,0 +1,5 @@ +--- +'astro': patch +--- + +chore: Remove non-null assertions diff --git a/packages/astro-parser/src/parse/state/tag.ts b/packages/astro-parser/src/parse/state/tag.ts index a8b919a49d..65bb93b4be 100644 --- a/packages/astro-parser/src/parse/state/tag.ts +++ b/packages/astro-parser/src/parse/state/tag.ts @@ -10,7 +10,6 @@ import { Directive, DirectiveType, TemplateNode, Text } from '../../interfaces.j import fuzzymatch from '../../utils/fuzzymatch.js'; import list from '../../utils/list.js'; -// eslint-disable-next-line no-useless-escape const valid_tag_name = /^\!?[a-zA-Z]{1,}:?[a-zA-Z0-9\-]*/; const meta_tags = new Map([ @@ -393,7 +392,6 @@ function read_attribute(parser: Parser, unique_names: Set) { } } - // eslint-disable-next-line no-useless-escape const name = parser.read_until(/[\s=\/>"']/); if (!name) return null; diff --git a/packages/astro/src/ast.ts b/packages/astro/src/ast.ts index 6e6bd191f1..f80084cb2d 100644 --- a/packages/astro/src/ast.ts +++ b/packages/astro/src/ast.ts @@ -20,10 +20,8 @@ export function getAttrValue(attributes: Attribute[], name: string): string | un /** Set TemplateNode attribute value */ export function setAttrValue(attributes: Attribute[], name: string, value: string): void { const attr = attributes.find((a) => a.name === name); - if (attr) { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - attr.value[0]!.data = value; - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - attr.value[0]!.raw = value; + if (attr && attr.value[0]) { + attr.value[0].data = value; + attr.value[0].raw = value; } } diff --git a/packages/astro/src/build.ts b/packages/astro/src/build.ts index 9661b5cc48..e6a23bc952 100644 --- a/packages/astro/src/build.ts +++ b/packages/astro/src/build.ts @@ -168,9 +168,8 @@ export async function build(astroConfig: AstroConfig): Promise<0 | 1> { info(logging, 'tip', `Set "buildOptions.site" in astro.config.mjs to generate a sitemap.xml, or set "buildOptions.sitemap: false" to disable this message.`); } - timer.write = performance.now(); - // write to disk and free up memory + timer.write = performance.now(); await Promise.all( Object.keys(buildState).map(async (id) => { const outPath = new URL(`.${id}`, dist); @@ -195,7 +194,7 @@ export async function build(astroConfig: AstroConfig): Promise<0 | 1> { publicFiles.map(async (filepath) => { const fileUrl = new URL(`file://${filepath}`); const rel = path.relative(fileURLToPath(pub), fileURLToPath(fileUrl)); - const outPath = new URL(path.join('.', rel), dist); + const outPath = new URL('./' + rel, dist); await fs.promises.mkdir(path.dirname(fileURLToPath(outPath)), { recursive: true }); await fs.promises.copyFile(fileUrl, outPath); }) @@ -209,7 +208,7 @@ export async function build(astroConfig: AstroConfig): Promise<0 | 1> { } /** - * 5. Bundling 2nd pass: On disk + * 5. Bundling 2nd Pass: On disk * Bundle JS, which requires hard files to optimize */ info(logging, 'build', yellow(`! bundling...`)); diff --git a/packages/astro/src/build/bundle/js.ts b/packages/astro/src/build/bundle/js.ts index 0112f024ad..d1d651e240 100644 --- a/packages/astro/src/build/bundle/js.ts +++ b/packages/astro/src/build/bundle/js.ts @@ -72,8 +72,9 @@ export async function bundleJS(imports: Set, { runtime, dist }: BundleOp format: 'esm', exports: 'named', entryFileNames(chunk) { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - return chunk.facadeModuleId!.substr(1); + const { facadeModuleId } = chunk; + if (!facadeModuleId) throw new Error(`facadeModuleId missing: ${chunk.name}`); + return facadeModuleId.substr(1); }, plugins: [ // We are using terser for the demo, but might switch to something else long term diff --git a/packages/astro/src/build/page.ts b/packages/astro/src/build/page.ts index 691b37507f..ae472e0719 100644 --- a/packages/astro/src/build/page.ts +++ b/packages/astro/src/build/page.ts @@ -37,7 +37,7 @@ export function getPageType(filepath: URL): 'collection' | 'static' { /** Build collection */ export async function buildCollectionPage({ astroConfig, filepath, logging, mode, runtime, site, resolvePackageUrl, buildState }: PageBuildOptions): Promise { - const rel = path.relative(fileURLToPath(astroConfig.astroRoot) + '/pages', fileURLToPath(filepath)); // pages/index.astro + const rel = path.posix.relative(fileURLToPath(astroConfig.astroRoot) + '/pages', fileURLToPath(filepath)); // pages/index.astro const pagePath = `/${rel.replace(/\$([^.]+)\.astro$/, '$1')}`; const srcPath = new URL('pages/' + rel, astroConfig.astroRoot); const builtURLs = new Set(); // !important: internal cache that prevents building the same URLs @@ -102,7 +102,7 @@ export async function buildCollectionPage({ astroConfig, filepath, logging, mode /** Build static page */ export async function buildStaticPage({ astroConfig, buildState, filepath, logging, mode, resolvePackageUrl, runtime }: PageBuildOptions): Promise { - const rel = path.relative(fileURLToPath(astroConfig.astroRoot) + '/pages', fileURLToPath(filepath)); // pages/index.astro + const rel = path.posix.relative(fileURLToPath(astroConfig.astroRoot) + '/pages', fileURLToPath(filepath)); // pages/index.astro const pagePath = `/${rel.replace(/\.(astro|md)$/, '')}`; let relPath = path.posix.join('/', rel.replace(/\.(astro|md)$/, '.html')); @@ -251,29 +251,31 @@ async function gatherRuntimes({ astroConfig, buildState, filepath, logging, reso switch (defn.plugin) { case 'preact': { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - imports.add(dynamic.get('preact')!); + const preact = dynamic.get('preact'); + if (!preact) throw new Error(`Unable to load Preact plugin`); + imports.add(preact); rel = rel.replace(/\.[^.]+$/, '.js'); break; } case 'react': { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - imports.add(dynamic.get('react')!); - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - imports.add(dynamic.get('react-dom')!); + const [react, reactDOM] = [dynamic.get('react'), dynamic.get('react-dom')]; + if (!react || !reactDOM) throw new Error(`Unable to load React plugin`); + imports.add(react); + imports.add(reactDOM); rel = rel.replace(/\.[^.]+$/, '.js'); break; } case 'vue': { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - imports.add(dynamic.get('vue')!); + const vue = dynamic.get('vue'); + if (!vue) throw new Error('Unable to load Vue plugin'); + imports.add(vue); rel = rel.replace(/\.[^.]+$/, '.vue.js'); break; } case 'svelte': { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - imports.add(dynamic.get('svelte')!); - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion + const svelte = dynamic.get('svelte'); + if (!svelte) throw new Error('Unable to load Svelte plugin'); + imports.add(svelte); imports.add('/_astro_internal/runtime/svelte.js'); rel = rel.replace(/\.[^.]+$/, '.svelte.js'); break; diff --git a/packages/astro/src/cli.ts b/packages/astro/src/cli.ts index 16f67fd60e..9588b92824 100644 --- a/packages/astro/src/cli.ts +++ b/packages/astro/src/cli.ts @@ -120,8 +120,8 @@ export async function cli(args: string[]) { } case 'build': case 'dev': { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - const cmd = cmdMap.get(state.cmd)!; + const cmd = cmdMap.get(state.cmd); + if (!cmd) throw new Error(`Error running ${state.cmd}`); runCommand(flags._[3], cmd, state.options); } } diff --git a/packages/astro/src/compiler/codegen/index.ts b/packages/astro/src/compiler/codegen/index.ts index a0f8e6b90a..e4cfad1cd2 100644 --- a/packages/astro/src/compiler/codegen/index.ts +++ b/packages/astro/src/compiler/codegen/index.ts @@ -423,21 +423,21 @@ function compileModule(module: Script, state: CodegenState, compileOptions: Comp if (plugin) { componentPlugins.add(plugin); } - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - state.importExportStatements.add(module.content.slice(componentImport.start!, componentImport.end!)); + const { start, end } = componentImport; + state.importExportStatements.add(module.content.slice(start || undefined, end || undefined)); } for (const componentImport of componentExports) { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - state.importExportStatements.add(module.content.slice(componentImport.start!, componentImport.end!)); + const { start, end } = componentImport; + state.importExportStatements.add(module.content.slice(start || undefined, end || undefined)); } if (componentProps.length > 0) { propsStatement = 'let {'; for (const componentExport of componentProps) { propsStatement += `${(componentExport.id as Identifier).name}`; - if (componentExport.init) { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - propsStatement += `= ${babelGenerator(componentExport.init!).code}`; + const { init } = componentExport; + if (init) { + propsStatement += `= ${babelGenerator(init).code}`; } propsStatement += `,`; } @@ -541,14 +541,12 @@ function compileHtml(enterNode: TemplateNode, state: CodegenState, compileOption switch (node.type) { case 'Expression': { let children: string[] = []; - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - for (const child of node.children!) { + for (const child of node.children || []) { children.push(compileHtml(child, state, compileOptions)); } let raw = ''; let nextChildIndex = 0; - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - for (const chunk of node.codeChunks!) { + for (const chunk of node.codeChunks) { raw += chunk; if (nextChildIndex < children.length) { raw += children[nextChildIndex++]; diff --git a/packages/astro/src/compiler/transform/doctype.ts b/packages/astro/src/compiler/transform/doctype.ts index e871f5b485..1a0ff39e12 100644 --- a/packages/astro/src/compiler/transform/doctype.ts +++ b/packages/astro/src/compiler/transform/doctype.ts @@ -21,8 +21,7 @@ export default function (_opts: { filename: string; fileID: string }): Transform name: '!doctype', type: 'Element', }; - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - parent.children!.splice(index, 0, dtNode); + (parent.children || []).splice(index, 0, dtNode); hasDoctype = true; } }, diff --git a/packages/astro/src/compiler/transform/index.ts b/packages/astro/src/compiler/transform/index.ts index 2fe0d0e73a..27f5a3212f 100644 --- a/packages/astro/src/compiler/transform/index.ts +++ b/packages/astro/src/compiler/transform/index.ts @@ -56,8 +56,7 @@ function walkAstWithVisitors(tmpl: TemplateNode, collection: VisitorCollection) walk(tmpl, { enter(node, parent, key, index) { if (collection.enter.has(node.type)) { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - const fns = collection.enter.get(node.type)!; + const fns = collection.enter.get(node.type) || []; for (let fn of fns) { fn.call(this, node, parent, key, index); } @@ -65,8 +64,7 @@ function walkAstWithVisitors(tmpl: TemplateNode, collection: VisitorCollection) }, leave(node, parent, key, index) { if (collection.leave.has(node.type)) { - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - const fns = collection.leave.get(node.type)!; + const fns = collection.leave.get(node.type) || []; for (let fn of fns) { fn.call(this, node, parent, key, index); } diff --git a/packages/astro/src/compiler/transform/styles.ts b/packages/astro/src/compiler/transform/styles.ts index 4e665fc61e..89a8c9c7f8 100644 --- a/packages/astro/src/compiler/transform/styles.ts +++ b/packages/astro/src/compiler/transform/styles.ts @@ -1,3 +1,6 @@ +import type { TransformOptions, Transformer } from '../../@types/transformer'; +import type { TemplateNode } from 'astro-parser'; + import crypto from 'crypto'; import fs from 'fs'; import { createRequire } from 'module'; @@ -8,10 +11,7 @@ import postcss, { Plugin } from 'postcss'; import postcssKeyframes from 'postcss-icss-keyframes'; import findUp from 'find-up'; import sass from 'sass'; -import type { RuntimeMode } from '../../@types/astro'; -import type { TransformOptions, Transformer } from '../../@types/transformer'; -import type { TemplateNode } from 'astro-parser'; -import { debug } from '../../logger.js'; +import { debug, error, LogOptions } from '../../logger.js'; import astroScopedStyles, { NEVER_SCOPED_TAGS } from './postcss-scoped-styles/index.js'; import slash from 'slash'; @@ -64,10 +64,10 @@ const miniCache: StylesMiniCache = { }; export interface TransformStyleOptions { + logging: LogOptions; type?: string; filename: string; scopedClass: string; - mode: RuntimeMode; } /** given a class="" string, does it contain a given class? */ @@ -80,7 +80,7 @@ function hasClass(classList: string, className: string): boolean { } /** Convert styles to scoped CSS */ -async function transformStyle(code: string, { type, filename, scopedClass, mode }: TransformStyleOptions): Promise { +async function transformStyle(code: string, { logging, type, filename, scopedClass }: TransformStyleOptions): Promise { let styleType: StyleType = 'css'; // important: assume CSS as default if (type) { styleType = getStyleType.get(type) || styleType; @@ -128,8 +128,7 @@ async function transformStyle(code: string, { type, filename, scopedClass, mode const tw = require.resolve('tailwindcss', { paths: [import.meta.url, process.cwd()] }); postcssPlugins.push(require(tw) as any); } catch (err) { - // eslint-disable-next-line no-console - console.error(err); + error(logging, 'transform', err); throw new Error(`tailwindcss not installed. Try running \`npm install tailwindcss\` and trying again.`); } } @@ -192,10 +191,10 @@ export default function transformStyles({ compileOptions, filename, fileID }: Tr styleNodes.push(node); styleTransformPromises.push( transformStyle(code, { + logging: compileOptions.logging, type: (langAttr && langAttr.value[0] && langAttr.value[0].data) || undefined, filename, scopedClass, - mode: compileOptions.mode, }) ); return; @@ -246,10 +245,10 @@ export default function transformStyles({ compileOptions, filename, fileID }: Tr styleNodes.push(node); styleTransformPromises.push( transformStyle(code, { + logging: compileOptions.logging, type: (langAttr && langAttr.value[0] && langAttr.value[0].data) || undefined, filename, scopedClass, - mode: compileOptions.mode, }) ); }, diff --git a/packages/astro/src/logger.ts b/packages/astro/src/logger.ts index d5d4fac896..c1b024c0c1 100644 --- a/packages/astro/src/logger.ts +++ b/packages/astro/src/logger.ts @@ -144,22 +144,21 @@ export const logger = { error: error.bind(null, defaultLogOptions), }; -// For silencing libraries that go directly to console.warn +/** For silencing libraries that go directly to console.warn */ export function trapWarn(cb: (...args: any[]) => void = () => {}) { - const warn = console.warn; + /* eslint-disable no-console */ + const consoleWarn = console.warn; console.warn = function (...args: any[]) { cb(...args); }; - return () => (console.warn = warn); + return () => (console.warn = consoleWarn); } - - function padStr(str: string, len: number) { const strLen = stringWidth(str); - if(strLen > len) { + if (strLen > len) { return str.substring(0, len - 3) + '...'; } const spaces = Array.from({ length: len - strLen }, () => ' ').join(''); return str + spaces; -} \ No newline at end of file +} diff --git a/packages/astro/test/.eslintrc.cjs b/packages/astro/test/.eslintrc.cjs new file mode 100644 index 0000000000..0e6acda5e7 --- /dev/null +++ b/packages/astro/test/.eslintrc.cjs @@ -0,0 +1,5 @@ +module.exports = { + rules: { + 'no-console': 'off', + }, +}; diff --git a/packages/astro/test/snowpack-integration.test.js b/packages/astro/test/snowpack-integration.test.js index ebd19f9930..cdccff14a1 100644 --- a/packages/astro/test/snowpack-integration.test.js +++ b/packages/astro/test/snowpack-integration.test.js @@ -27,7 +27,6 @@ SnowpackDev.before(async () => { try { runtime = await createRuntime(astroConfig, { logging }); } catch (err) { - // eslint-disable-next-line no-console console.error(err); setupError = err; } @@ -82,7 +81,6 @@ SnowpackDev('Can load every page', async () => { } if (failed.length > 0) { - // eslint-disable-next-line no-console console.error(failed); } assert.equal(failed.length, 0, 'Failed pages'); diff --git a/tools/astro-languageserver/src/plugins/PluginHost.ts b/tools/astro-languageserver/src/plugins/PluginHost.ts index 037dd6e070..8d59c9c937 100644 --- a/tools/astro-languageserver/src/plugins/PluginHost.ts +++ b/tools/astro-languageserver/src/plugins/PluginHost.ts @@ -4,7 +4,6 @@ import type * as d from './interfaces'; import { flatten } from '../utils'; import { FoldingRange } from 'vscode-languageserver-types'; -// eslint-disable-next-line no-shadow enum ExecuteMode { None, FirstNonNull,