From ba57d78d1b655097cba9e7437dd051269ee82419 Mon Sep 17 00:00:00 2001 From: Chad Hietala Date: Tue, 12 Jan 2021 16:01:27 -0500 Subject: [PATCH 1/2] bugfix: Ensure Component Lookup Is Well Formed This PR introduces an assertion during the component name refinement to assert that the lookup syntax is well formed. If a developer accidentally types ":" instead of "::" you will end up with a runtime erorr that looks like: ``` Uncaught Error: Assertion Failed: fullName must be a proper full name at assert (vendor.js:52729) at Container.lookup (vendor.js:16581) at Class.lookup (vendor.js:43950) at layoutFor (vendor.js:28680) at lookupComponentPair (vendor.js:28697) at lookupComponent (vendor.js:28711) at RuntimeResolver._lookupComponentDefinition (vendor.js:28992) at RuntimeResolver.lookupComponentHandle (vendor.js:28837) at CompileTimeLookup.lookupComponentDefinition (vendor.js:25065) at LazyCompiler.resolveLayoutForTag (vendor.js:60366) ``` This message is not actionable for developers. This PR will causes a compile time error to occur with the following information: ``` Malformed component lookup in "test.js". Got but you must use \"::\" to indicate a lookup ``` --- .../lib/system/compile-options.ts | 10 ++++++++++ .../tests/system/compile_options_test.js | 10 ++++++++++ 2 files changed, 20 insertions(+) diff --git a/packages/ember-template-compiler/lib/system/compile-options.ts b/packages/ember-template-compiler/lib/system/compile-options.ts index 1e9e2129cfc..6a3d6cccfc4 100644 --- a/packages/ember-template-compiler/lib/system/compile-options.ts +++ b/packages/ember-template-compiler/lib/system/compile-options.ts @@ -1,4 +1,5 @@ import { EMBER_STRICT_MODE } from '@ember/canary-features'; +import { assert } from '@ember/debug'; import { assign } from '@ember/polyfills'; import { PrecompileOptions } from '@glimmer/compiler'; import { AST, ASTPlugin, ASTPluginEnvironment, Syntax } from '@glimmer/syntax'; @@ -8,6 +9,10 @@ import COMPONENT_NAME_SIMPLE_DASHERIZE_CACHE from './dasherize-component-name'; let USER_PLUGINS: PluginFunc[] = []; +function malformedComponentLookup(string: string) { + return string.indexOf('::') === -1 && string.indexOf(':') > -1; +} + export default function compileOptions( _options: Partial = {} ): PrecompileOptions { @@ -16,6 +21,11 @@ export default function compileOptions( _options, { customizeComponentName(tagname: string): string { + assert( + `Malformed component lookup in "${_options.moduleName}". Got <${tagname} /> but you must use "::" to indicate a lookup`, + !malformedComponentLookup(tagname) + ); + return COMPONENT_NAME_SIMPLE_DASHERIZE_CACHE.get(tagname); }, } diff --git a/packages/ember-template-compiler/tests/system/compile_options_test.js b/packages/ember-template-compiler/tests/system/compile_options_test.js index 1ee2968adf8..cfbccbe8317 100644 --- a/packages/ember-template-compiler/tests/system/compile_options_test.js +++ b/packages/ember-template-compiler/tests/system/compile_options_test.js @@ -15,6 +15,16 @@ moduleFor( assert.notEqual(compileOptions(), compileOptions()); } + ['@test customizeComponentName asserts name is well formed'](assert) { + let options = compileOptions({ moduleName: 'test.js' }); + + expectAssertion(() => { + options.customizeComponentName('Foo:Bar'); + }, /Malformed component lookup in "test.js". Got but you must use "::" to indicate a lookup/); + + assert.ok(options.customizeComponentName('Foo::Bar')); + } + ['@test has default AST plugins in resolution mode'](assert) { assert.expect(RESOLUTION_MODE_TRANSFORMS.length); From c874f1599045e2ac6e6cccd1701a02e66ae4d43b Mon Sep 17 00:00:00 2001 From: Chad Hietala Date: Thu, 14 Jan 2021 10:38:32 -0500 Subject: [PATCH 2/2] Update packages/ember-template-compiler/lib/system/compile-options.ts Co-authored-by: Godfrey Chan --- packages/ember-template-compiler/lib/system/compile-options.ts | 2 +- .../tests/system/compile_options_test.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/ember-template-compiler/lib/system/compile-options.ts b/packages/ember-template-compiler/lib/system/compile-options.ts index 6a3d6cccfc4..36059842997 100644 --- a/packages/ember-template-compiler/lib/system/compile-options.ts +++ b/packages/ember-template-compiler/lib/system/compile-options.ts @@ -22,7 +22,7 @@ export default function compileOptions( { customizeComponentName(tagname: string): string { assert( - `Malformed component lookup in "${_options.moduleName}". Got <${tagname} /> but you must use "::" to indicate a lookup`, + `You tried to invoke a component named <${tagname} /> in "${_options.moduleName}", but that is not a valid name for a component. Did you mean to use the "::" syntax for nested components?`, !malformedComponentLookup(tagname) ); diff --git a/packages/ember-template-compiler/tests/system/compile_options_test.js b/packages/ember-template-compiler/tests/system/compile_options_test.js index cfbccbe8317..d093f9d84da 100644 --- a/packages/ember-template-compiler/tests/system/compile_options_test.js +++ b/packages/ember-template-compiler/tests/system/compile_options_test.js @@ -20,7 +20,7 @@ moduleFor( expectAssertion(() => { options.customizeComponentName('Foo:Bar'); - }, /Malformed component lookup in "test.js". Got but you must use "::" to indicate a lookup/); + }, /You tried to invoke a component named in "test.js", but that is not a valid name for a component. Did you mean to use the "::" syntax for nested components\?/); assert.ok(options.customizeComponentName('Foo::Bar')); }