fix(router): improve error messages for routes with no config
Closes #2323
This commit is contained in:
parent
ccb41632c7
commit
8bdca5c03e
|
@ -17,7 +17,8 @@ import {
|
||||||
isStringMap,
|
isStringMap,
|
||||||
isFunction,
|
isFunction,
|
||||||
StringWrapper,
|
StringWrapper,
|
||||||
BaseException
|
BaseException,
|
||||||
|
getTypeNameForDebugging
|
||||||
} from 'angular2/src/facade/lang';
|
} from 'angular2/src/facade/lang';
|
||||||
import {RouteConfig} from './route_config_impl';
|
import {RouteConfig} from './route_config_impl';
|
||||||
import {reflector} from 'angular2/src/reflection/reflection';
|
import {reflector} from 'angular2/src/reflection/reflection';
|
||||||
|
@ -154,6 +155,9 @@ export class RouteRegistry {
|
||||||
let componentCursor = parentComponent;
|
let componentCursor = parentComponent;
|
||||||
for (let i = 0; i < linkParams.length; i += 1) {
|
for (let i = 0; i < linkParams.length; i += 1) {
|
||||||
let segment = linkParams[i];
|
let segment = linkParams[i];
|
||||||
|
if (isBlank(componentCursor)) {
|
||||||
|
throw new BaseException(`Could not find route named "${segment}".`);
|
||||||
|
}
|
||||||
if (!isString(segment)) {
|
if (!isString(segment)) {
|
||||||
throw new BaseException(`Unexpected segment "${segment}" in link DSL. Expected a string.`);
|
throw new BaseException(`Unexpected segment "${segment}" in link DSL. Expected a string.`);
|
||||||
} else if (segment == '' || segment == '.' || segment == '..') {
|
} else if (segment == '' || segment == '.' || segment == '..') {
|
||||||
|
@ -170,9 +174,13 @@ export class RouteRegistry {
|
||||||
|
|
||||||
var componentRecognizer = this._rules.get(componentCursor);
|
var componentRecognizer = this._rules.get(componentCursor);
|
||||||
if (isBlank(componentRecognizer)) {
|
if (isBlank(componentRecognizer)) {
|
||||||
throw new BaseException(`Could not find route config for "${segment}".`);
|
throw new BaseException(`Component "${getTypeNameForDebugging(componentCursor)}" has no route config.`);
|
||||||
}
|
}
|
||||||
var response = componentRecognizer.generate(segment, params);
|
var response = componentRecognizer.generate(segment, params);
|
||||||
|
if (isBlank(response)) {
|
||||||
|
throw new BaseException(
|
||||||
|
`Component "${getTypeNameForDebugging(componentCursor)}" has no route named "${segment}".`);
|
||||||
|
}
|
||||||
url += response['url'];
|
url += response['url'];
|
||||||
componentCursor = response['nextComponent'];
|
componentCursor = response['nextComponent'];
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,143 +11,146 @@ import {
|
||||||
} from 'angular2/test_lib';
|
} from 'angular2/test_lib';
|
||||||
|
|
||||||
import {Promise, PromiseWrapper} from 'angular2/src/facade/async';
|
import {Promise, PromiseWrapper} from 'angular2/src/facade/async';
|
||||||
import {ListWrapper} from 'angular2/src/facade/collection';
|
|
||||||
|
|
||||||
import {RouteRegistry} from 'angular2/src/router/route_registry';
|
import {RouteRegistry} from 'angular2/src/router/route_registry';
|
||||||
import {RouteConfig} from 'angular2/src/router/route_config_decorator';
|
import {RouteConfig} from 'angular2/src/router/route_config_decorator';
|
||||||
|
|
||||||
export function main() {
|
export function main() {
|
||||||
describe('RouteRegistry', () => {
|
describe('RouteRegistry', () => {
|
||||||
var registry, rootHostComponent = new Object();
|
var registry;
|
||||||
|
|
||||||
beforeEach(() => { registry = new RouteRegistry(); });
|
beforeEach(() => { registry = new RouteRegistry(); });
|
||||||
|
|
||||||
it('should match the full URL', inject([AsyncTestCompleter], (async) => {
|
it('should match the full URL', inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent, {'path': '/', 'component': DummyCompA});
|
registry.config(RootHostCmp, {'path': '/', 'component': DummyCmpA});
|
||||||
registry.config(rootHostComponent, {'path': '/test', 'component': DummyCompB});
|
registry.config(RootHostCmp, {'path': '/test', 'component': DummyCmpB});
|
||||||
|
|
||||||
registry.recognize('/test', rootHostComponent)
|
registry.recognize('/test', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyCompB);
|
expect(instruction.component).toBe(DummyCmpB);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should generate URLs starting at the given component', () => {
|
it('should generate URLs starting at the given component', () => {
|
||||||
registry.config(rootHostComponent,
|
registry.config(RootHostCmp,
|
||||||
{'path': '/first/...', 'component': DummyParentComp, 'as': 'firstCmp'});
|
{'path': '/first/...', 'component': DummyParentCmp, 'as': 'firstCmp'});
|
||||||
|
|
||||||
expect(registry.generate(['firstCmp', 'secondCmp'], rootHostComponent))
|
expect(registry.generate(['firstCmp', 'secondCmp'], RootHostCmp)).toEqual('first/second');
|
||||||
.toEqual('first/second');
|
expect(registry.generate(['secondCmp'], DummyParentCmp)).toEqual('second');
|
||||||
expect(registry.generate(['secondCmp'], DummyParentComp)).toEqual('second');
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should generate URLs with params', () => {
|
it('should generate URLs with params', () => {
|
||||||
registry.config(
|
registry.config(
|
||||||
rootHostComponent,
|
RootHostCmp,
|
||||||
{'path': '/first/:param/...', 'component': DummyParentParamComp, 'as': 'firstCmp'});
|
{'path': '/first/:param/...', 'component': DummyParentParamCmp, 'as': 'firstCmp'});
|
||||||
|
|
||||||
var url = registry.generate(['firstCmp', {param: 'one'}, 'secondCmp', {param: 'two'}],
|
var url =
|
||||||
rootHostComponent);
|
registry.generate(['firstCmp', {param: 'one'}, 'secondCmp', {param: 'two'}], RootHostCmp);
|
||||||
expect(url).toEqual('first/one/second/two');
|
expect(url).toEqual('first/one/second/two');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should generate URLs of loaded components after they are loaded',
|
it('should generate URLs of loaded components after they are loaded',
|
||||||
inject([AsyncTestCompleter], (async) => {
|
inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent, {
|
registry.config(RootHostCmp, {
|
||||||
'path': '/first/...',
|
'path': '/first/...',
|
||||||
'component': {'type': 'loader', 'loader': AsyncParentLoader},
|
'component': {'type': 'loader', 'loader': AsyncParentLoader},
|
||||||
'as': 'firstCmp'
|
'as': 'firstCmp'
|
||||||
});
|
});
|
||||||
|
|
||||||
expect(() => registry.generate(['firstCmp', 'secondCmp'], rootHostComponent))
|
expect(() => registry.generate(['firstCmp', 'secondCmp'], RootHostCmp))
|
||||||
.toThrowError('Could not find route config for "secondCmp".');
|
.toThrowError('Could not find route named "secondCmp".');
|
||||||
|
|
||||||
registry.recognize('/first/second', rootHostComponent)
|
registry.recognize('/first/second', RootHostCmp)
|
||||||
.then((_) => {
|
.then((_) => {
|
||||||
expect(registry.generate(['firstCmp', 'secondCmp'], rootHostComponent))
|
expect(registry.generate(['firstCmp', 'secondCmp'], RootHostCmp))
|
||||||
.toEqual('first/second');
|
.toEqual('first/second');
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should prefer static segments to dynamic', inject([AsyncTestCompleter], (async) => {
|
|
||||||
registry.config(rootHostComponent, {'path': '/:site', 'component': DummyCompB});
|
|
||||||
registry.config(rootHostComponent, {'path': '/home', 'component': DummyCompA});
|
|
||||||
|
|
||||||
registry.recognize('/home', rootHostComponent)
|
it('should throw when generating a url and a parent has no config', () => {
|
||||||
|
expect(() => registry.generate(['firstCmp', 'secondCmp'], RootHostCmp))
|
||||||
|
.toThrowError('Component "RootHostCmp" has no route config.');
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
it('should prefer static segments to dynamic', inject([AsyncTestCompleter], (async) => {
|
||||||
|
registry.config(RootHostCmp, {'path': '/:site', 'component': DummyCmpB});
|
||||||
|
registry.config(RootHostCmp, {'path': '/home', 'component': DummyCmpA});
|
||||||
|
|
||||||
|
registry.recognize('/home', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyCompA);
|
expect(instruction.component).toBe(DummyCmpA);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should prefer dynamic segments to star', inject([AsyncTestCompleter], (async) => {
|
it('should prefer dynamic segments to star', inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent, {'path': '/:site', 'component': DummyCompA});
|
registry.config(RootHostCmp, {'path': '/:site', 'component': DummyCmpA});
|
||||||
registry.config(rootHostComponent, {'path': '/*site', 'component': DummyCompB});
|
registry.config(RootHostCmp, {'path': '/*site', 'component': DummyCmpB});
|
||||||
|
|
||||||
registry.recognize('/home', rootHostComponent)
|
registry.recognize('/home', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyCompA);
|
expect(instruction.component).toBe(DummyCmpA);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should prefer routes with more dynamic segments', inject([AsyncTestCompleter], (async) => {
|
it('should prefer routes with more dynamic segments', inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent, {'path': '/:first/*rest', 'component': DummyCompA});
|
registry.config(RootHostCmp, {'path': '/:first/*rest', 'component': DummyCmpA});
|
||||||
registry.config(rootHostComponent, {'path': '/*all', 'component': DummyCompB});
|
registry.config(RootHostCmp, {'path': '/*all', 'component': DummyCmpB});
|
||||||
|
|
||||||
registry.recognize('/some/path', rootHostComponent)
|
registry.recognize('/some/path', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyCompA);
|
expect(instruction.component).toBe(DummyCmpA);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should prefer routes with more static segments', inject([AsyncTestCompleter], (async) => {
|
it('should prefer routes with more static segments', inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent, {'path': '/first/:second', 'component': DummyCompA});
|
registry.config(RootHostCmp, {'path': '/first/:second', 'component': DummyCmpA});
|
||||||
registry.config(rootHostComponent, {'path': '/:first/:second', 'component': DummyCompB});
|
registry.config(RootHostCmp, {'path': '/:first/:second', 'component': DummyCmpB});
|
||||||
|
|
||||||
registry.recognize('/first/second', rootHostComponent)
|
registry.recognize('/first/second', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyCompA);
|
expect(instruction.component).toBe(DummyCmpA);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should prefer routes with static segments before dynamic segments',
|
it('should prefer routes with static segments before dynamic segments',
|
||||||
inject([AsyncTestCompleter], (async) => {
|
inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent,
|
registry.config(RootHostCmp, {'path': '/first/second/:third', 'component': DummyCmpB});
|
||||||
{'path': '/first/second/:third', 'component': DummyCompB});
|
registry.config(RootHostCmp, {'path': '/first/:second/third', 'component': DummyCmpA});
|
||||||
registry.config(rootHostComponent,
|
|
||||||
{'path': '/first/:second/third', 'component': DummyCompA});
|
|
||||||
|
|
||||||
registry.recognize('/first/second/third', rootHostComponent)
|
registry.recognize('/first/second/third', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyCompB);
|
expect(instruction.component).toBe(DummyCmpB);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should match the full URL using child components', inject([AsyncTestCompleter], (async) => {
|
it('should match the full URL using child components', inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent, {'path': '/first/...', 'component': DummyParentComp});
|
registry.config(RootHostCmp, {'path': '/first/...', 'component': DummyParentCmp});
|
||||||
|
|
||||||
registry.recognize('/first/second', rootHostComponent)
|
registry.recognize('/first/second', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyParentComp);
|
expect(instruction.component).toBe(DummyParentCmp);
|
||||||
expect(instruction.child.component).toBe(DummyCompB);
|
expect(instruction.child.component).toBe(DummyCmpB);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should match the URL using async child components',
|
it('should match the URL using async child components',
|
||||||
inject([AsyncTestCompleter], (async) => {
|
inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(rootHostComponent, {'path': '/first/...', 'component': DummyAsyncComp});
|
registry.config(RootHostCmp, {'path': '/first/...', 'component': DummyAsyncCmp});
|
||||||
|
|
||||||
registry.recognize('/first/second', rootHostComponent)
|
registry.recognize('/first/second', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyAsyncComp);
|
expect(instruction.component).toBe(DummyAsyncCmp);
|
||||||
expect(instruction.child.component).toBe(DummyCompB);
|
expect(instruction.child.component).toBe(DummyCmpB);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
@ -155,67 +158,67 @@ export function main() {
|
||||||
it('should match the URL using an async parent component',
|
it('should match the URL using an async parent component',
|
||||||
inject([AsyncTestCompleter], (async) => {
|
inject([AsyncTestCompleter], (async) => {
|
||||||
registry.config(
|
registry.config(
|
||||||
rootHostComponent,
|
RootHostCmp,
|
||||||
{'path': '/first/...', 'component': {'loader': AsyncParentLoader, 'type': 'loader'}});
|
{'path': '/first/...', 'component': {'loader': AsyncParentLoader, 'type': 'loader'}});
|
||||||
|
|
||||||
registry.recognize('/first/second', rootHostComponent)
|
registry.recognize('/first/second', RootHostCmp)
|
||||||
.then((instruction) => {
|
.then((instruction) => {
|
||||||
expect(instruction.component).toBe(DummyParentComp);
|
expect(instruction.component).toBe(DummyParentCmp);
|
||||||
expect(instruction.child.component).toBe(DummyCompB);
|
expect(instruction.child.component).toBe(DummyCmpB);
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should throw when a config does not have a component or redirectTo property', () => {
|
it('should throw when a config does not have a component or redirectTo property', () => {
|
||||||
expect(() => registry.config(rootHostComponent, {'path': '/some/path'}))
|
expect(() => registry.config(RootHostCmp, {'path': '/some/path'}))
|
||||||
.toThrowError(
|
.toThrowError(
|
||||||
'Route config should contain exactly one \'component\', or \'redirectTo\' property');
|
'Route config should contain exactly one \'component\', or \'redirectTo\' property');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when a config has an invalid component type', () => {
|
it('should throw when a config has an invalid component type', () => {
|
||||||
expect(() => registry.config(
|
expect(() => registry.config(
|
||||||
rootHostComponent,
|
RootHostCmp,
|
||||||
{'path': '/some/path', 'component': {'type': 'intentionallyWrongComponentType'}}))
|
{'path': '/some/path', 'component': {'type': 'intentionallyWrongComponentType'}}))
|
||||||
.toThrowError('Invalid component type \'intentionallyWrongComponentType\'');
|
.toThrowError('Invalid component type \'intentionallyWrongComponentType\'');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when a parent config is missing the `...` suffix any of its children add routes',
|
it('should throw when a parent config is missing the `...` suffix any of its children add routes',
|
||||||
() => {
|
() => {
|
||||||
expect(() =>
|
expect(() => registry.config(RootHostCmp, {'path': '/', 'component': DummyParentCmp}))
|
||||||
registry.config(rootHostComponent, {'path': '/', 'component': DummyParentComp}))
|
|
||||||
.toThrowError(
|
.toThrowError(
|
||||||
'Child routes are not allowed for "/". Use "..." on the parent\'s route path.');
|
'Child routes are not allowed for "/". Use "..." on the parent\'s route path.');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when a parent config is missing the `...` suffix any of its children add routes',
|
it('should throw when a parent config uses `...` suffix before the end of the route', () => {
|
||||||
() => {
|
expect(() => registry.config(RootHostCmp,
|
||||||
expect(() => registry.config(rootHostComponent,
|
{'path': '/home/.../fun/', 'component': DummyParentCmp}))
|
||||||
{'path': '/home/.../fun/', 'component': DummyParentComp}))
|
.toThrowError('Unexpected "..." before the end of the path for "home/.../fun/".');
|
||||||
.toThrowError('Unexpected "..." before the end of the path for "home/.../fun/".');
|
});
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function AsyncParentLoader() {
|
function AsyncParentLoader() {
|
||||||
return PromiseWrapper.resolve(DummyParentComp);
|
return PromiseWrapper.resolve(DummyParentCmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
function AsyncChildLoader() {
|
function AsyncChildLoader() {
|
||||||
return PromiseWrapper.resolve(DummyCompB);
|
return PromiseWrapper.resolve(DummyCmpB);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class RootHostCmp {}
|
||||||
|
|
||||||
@RouteConfig([{'path': '/second', 'component': {'loader': AsyncChildLoader, 'type': 'loader'}}])
|
@RouteConfig([{'path': '/second', 'component': {'loader': AsyncChildLoader, 'type': 'loader'}}])
|
||||||
class DummyAsyncComp {
|
class DummyAsyncCmp {
|
||||||
}
|
}
|
||||||
|
|
||||||
class DummyCompA {}
|
class DummyCmpA {}
|
||||||
class DummyCompB {}
|
class DummyCmpB {}
|
||||||
|
|
||||||
@RouteConfig([{'path': '/second', 'component': DummyCompB, 'as': 'secondCmp'}])
|
@RouteConfig([{'path': '/second', 'component': DummyCmpB, 'as': 'secondCmp'}])
|
||||||
class DummyParentComp {
|
class DummyParentCmp {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@RouteConfig([{'path': '/second/:param', 'component': DummyCompB, 'as': 'secondCmp'}])
|
@RouteConfig([{'path': '/second/:param', 'component': DummyCmpB, 'as': 'secondCmp'}])
|
||||||
class DummyParentParamComp {
|
class DummyParentParamCmp {
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue