You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

736 lines
17 KiB

4 years ago
'use strict'
const t = require('tap')
const test = t.test
const FindMyWay = require('../')
test('the router is an object with methods', t => {
t.plan(4)
const findMyWay = FindMyWay()
t.is(typeof findMyWay.on, 'function')
t.is(typeof findMyWay.off, 'function')
t.is(typeof findMyWay.lookup, 'function')
t.is(typeof findMyWay.find, 'function')
})
test('on throws for invalid method', t => {
t.plan(1)
const findMyWay = FindMyWay()
t.throws(() => {
findMyWay.on('INVALID', '/a/b')
})
})
test('on throws for invalid path', t => {
t.plan(3)
const findMyWay = FindMyWay()
// Non string
t.throws(() => {
findMyWay.on('GET', 1)
})
// Empty
t.throws(() => {
findMyWay.on('GET', '')
})
// Doesn't start with / or *
t.throws(() => {
findMyWay.on('GET', 'invalid')
})
})
test('register a route', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test', () => {
t.ok('inside the handler')
})
findMyWay.lookup({ method: 'GET', url: '/test', headers: {} }, null)
})
test('register a route with multiple methods', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on(['GET', 'POST'], '/test', () => {
t.ok('inside the handler')
})
findMyWay.lookup({ method: 'GET', url: '/test', headers: {} }, null)
findMyWay.lookup({ method: 'POST', url: '/test', headers: {} }, null)
})
test('does not register /test/*/ when ignoreTrailingSlash is true', t => {
t.plan(1)
const findMyWay = FindMyWay({
ignoreTrailingSlash: true
})
findMyWay.on('GET', '/test/*', () => {})
t.is(
findMyWay.routes.filter((r) => r.path.includes('/test')).length,
1
)
})
test('off throws for invalid method', t => {
t.plan(1)
const findMyWay = FindMyWay()
t.throws(() => {
findMyWay.off('INVALID', '/a/b')
})
})
test('off throws for invalid path', t => {
t.plan(3)
const findMyWay = FindMyWay()
// Non string
t.throws(() => {
findMyWay.off('GET', 1)
})
// Empty
t.throws(() => {
findMyWay.off('GET', '')
})
// Doesn't start with / or *
t.throws(() => {
findMyWay.off('GET', 'invalid')
})
})
test('off with nested wildcards with parametric and static', t => {
t.plan(3)
const findMyWay = FindMyWay({
defaultRoute: (req, res) => {
t.fail('we should not be here, the url is: ' + req.url)
}
})
findMyWay.on('GET', '*', (req, res, params) => {
t.is(params['*'], '/foo2/first/second')
})
findMyWay.on('GET', '/foo1/*', () => {})
findMyWay.on('GET', '/foo2/*', () => {})
findMyWay.on('GET', '/foo3/:param', () => {})
findMyWay.on('GET', '/foo3/*', () => {})
findMyWay.on('GET', '/foo4/param/hello/test/long/route', () => {})
var route1 = findMyWay.find('GET', '/foo3/first/second')
t.is(route1.params['*'], 'first/second')
findMyWay.off('GET', '/foo3/*')
var route2 = findMyWay.find('GET', '/foo3/first/second')
t.is(route2.params['*'], '/foo3/first/second')
findMyWay.off('GET', '/foo2/*')
findMyWay.lookup(
{ method: 'GET', url: '/foo2/first/second', headers: {} },
null
)
})
test('off removes all routes when ignoreTrailingSlash is true', t => {
t.plan(6)
const findMyWay = FindMyWay({
ignoreTrailingSlash: true
})
findMyWay.on('GET', '/test1/', () => {})
t.is(findMyWay.routes.length, 2)
findMyWay.on('GET', '/test2', () => {})
t.is(findMyWay.routes.length, 4)
findMyWay.off('GET', '/test1')
t.is(findMyWay.routes.length, 2)
t.is(
findMyWay.routes.filter((r) => r.path === '/test2').length,
1
)
t.is(
findMyWay.routes.filter((r) => r.path === '/test2/').length,
1
)
findMyWay.off('GET', '/test2/')
t.is(findMyWay.routes.length, 0)
})
test('deregister a route without children', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/a', () => {})
findMyWay.on('GET', '/a/b', () => {})
findMyWay.off('GET', '/a/b')
t.ok(findMyWay.find('GET', '/a'))
t.notOk(findMyWay.find('GET', '/a/b'))
})
test('deregister a route with children', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/a', () => {})
findMyWay.on('GET', '/a/b', () => {})
findMyWay.off('GET', '/a')
t.notOk(findMyWay.find('GET', '/a'))
t.ok(findMyWay.find('GET', '/a/b'))
})
test('deregister a route by method', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on(['GET', 'POST'], '/a', () => {})
findMyWay.off('GET', '/a')
t.notOk(findMyWay.find('GET', '/a'))
t.ok(findMyWay.find('POST', '/a'))
})
test('deregister a route with multiple methods', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on(['GET', 'POST'], '/a', () => {})
findMyWay.off(['GET', 'POST'], '/a')
t.notOk(findMyWay.find('GET', '/a'))
t.notOk(findMyWay.find('POST', '/a'))
})
test('reset a router', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on(['GET', 'POST'], '/a', () => {})
findMyWay.reset()
t.notOk(findMyWay.find('GET', '/a'))
t.notOk(findMyWay.find('POST', '/a'))
})
test('default route', t => {
t.plan(1)
const findMyWay = FindMyWay({
defaultRoute: () => {
t.ok('inside the default route')
}
})
findMyWay.lookup({ method: 'GET', url: '/test', headers: {} }, null)
})
test('parametric route', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/:id', (req, res, params) => {
t.is(params.id, 'hello')
})
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
})
test('multiple parametric route', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/:id', (req, res, params) => {
t.is(params.id, 'hello')
})
findMyWay.on('GET', '/other-test/:id', (req, res, params) => {
t.is(params.id, 'world')
})
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/other-test/world', headers: {} }, null)
})
test('multiple parametric route with the same prefix', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/:id', (req, res, params) => {
t.is(params.id, 'hello')
})
findMyWay.on('GET', '/test/:id/world', (req, res, params) => {
t.is(params.id, 'world')
})
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/test/world/world', headers: {} }, null)
})
test('nested parametric route', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/:hello/test/:world', (req, res, params) => {
t.is(params.hello, 'hello')
t.is(params.world, 'world')
})
findMyWay.lookup({ method: 'GET', url: '/test/hello/test/world', headers: {} }, null)
})
test('nested parametric route with same prefix', t => {
t.plan(3)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test', (req, res, params) => {
t.ok('inside route')
})
findMyWay.on('GET', '/test/:hello/test/:world', (req, res, params) => {
t.is(params.hello, 'hello')
t.is(params.world, 'world')
})
findMyWay.lookup({ method: 'GET', url: '/test', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/test/hello/test/world', headers: {} }, null)
})
test('long route', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/abc/def/ghi/lmn/opq/rst/uvz', (req, res, params) => {
t.ok('inside long path')
})
findMyWay.lookup({ method: 'GET', url: '/abc/def/ghi/lmn/opq/rst/uvz', headers: {} }, null)
})
test('long parametric route', t => {
t.plan(3)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/abc/:def/ghi/:lmn/opq/:rst/uvz', (req, res, params) => {
t.is(params.def, 'def')
t.is(params.lmn, 'lmn')
t.is(params.rst, 'rst')
})
findMyWay.lookup({ method: 'GET', url: '/abc/def/ghi/lmn/opq/rst/uvz', headers: {} }, null)
})
test('long parametric route with common prefix', t => {
t.plan(9)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/', (req, res, params) => {
throw new Error('I shoul not be here')
})
findMyWay.on('GET', '/abc', (req, res, params) => {
throw new Error('I shoul not be here')
})
findMyWay.on('GET', '/abc/:def', (req, res, params) => {
t.is(params.def, 'def')
})
findMyWay.on('GET', '/abc/:def/ghi/:lmn', (req, res, params) => {
t.is(params.def, 'def')
t.is(params.lmn, 'lmn')
})
findMyWay.on('GET', '/abc/:def/ghi/:lmn/opq/:rst', (req, res, params) => {
t.is(params.def, 'def')
t.is(params.lmn, 'lmn')
t.is(params.rst, 'rst')
})
findMyWay.on('GET', '/abc/:def/ghi/:lmn/opq/:rst/uvz', (req, res, params) => {
t.is(params.def, 'def')
t.is(params.lmn, 'lmn')
t.is(params.rst, 'rst')
})
findMyWay.lookup({ method: 'GET', url: '/abc/def', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/abc/def/ghi/lmn', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/abc/def/ghi/lmn/opq/rst', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/abc/def/ghi/lmn/opq/rst/uvz', headers: {} }, null)
})
test('common prefix', t => {
t.plan(4)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/f', (req, res, params) => {
t.ok('inside route')
})
findMyWay.on('GET', '/ff', (req, res, params) => {
t.ok('inside route')
})
findMyWay.on('GET', '/ffa', (req, res, params) => {
t.ok('inside route')
})
findMyWay.on('GET', '/ffb', (req, res, params) => {
t.ok('inside route')
})
findMyWay.lookup({ method: 'GET', url: '/f', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/ff', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/ffa', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/ffb', headers: {} }, null)
})
test('wildcard', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/*', (req, res, params) => {
t.is(params['*'], 'hello')
})
findMyWay.lookup(
{ method: 'GET', url: '/test/hello', headers: {} },
null
)
})
test('catch all wildcard', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '*', (req, res, params) => {
t.is(params['*'], '/test/hello')
})
findMyWay.lookup(
{ method: 'GET', url: '/test/hello', headers: {} },
null
)
})
test('find should return the route', t => {
t.plan(1)
const findMyWay = FindMyWay()
const fn = () => {}
findMyWay.on('GET', '/test', fn)
t.deepEqual(
findMyWay.find('GET', '/test'),
{ handler: fn, params: {}, store: null }
)
})
test('find should return the route with params', t => {
t.plan(1)
const findMyWay = FindMyWay()
const fn = () => {}
findMyWay.on('GET', '/test/:id', fn)
t.deepEqual(
findMyWay.find('GET', '/test/hello'),
{ handler: fn, params: { id: 'hello' }, store: null }
)
})
test('find should return a null handler if the route does not exist', t => {
t.plan(1)
const findMyWay = FindMyWay()
t.deepEqual(
findMyWay.find('GET', '/test'),
null
)
})
test('should decode the uri - parametric', t => {
t.plan(1)
const findMyWay = FindMyWay()
const fn = () => {}
findMyWay.on('GET', '/test/:id', fn)
t.deepEqual(
findMyWay.find('GET', '/test/he%2Fllo'),
{ handler: fn, params: { id: 'he/llo' }, store: null }
)
})
test('should decode the uri - wildcard', t => {
t.plan(1)
const findMyWay = FindMyWay()
const fn = () => {}
findMyWay.on('GET', '/test/*', fn)
t.deepEqual(
findMyWay.find('GET', '/test/he%2Fllo'),
{ handler: fn, params: { '*': 'he/llo' }, store: null }
)
})
test('safe decodeURIComponent', t => {
t.plan(1)
const findMyWay = FindMyWay()
const fn = () => {}
findMyWay.on('GET', '/test/:id', fn)
t.deepEqual(
findMyWay.find('GET', '/test/hel%"Flo'),
null
)
})
test('safe decodeURIComponent - nested route', t => {
t.plan(1)
const findMyWay = FindMyWay()
const fn = () => {}
findMyWay.on('GET', '/test/hello/world/:id/blah', fn)
t.deepEqual(
findMyWay.find('GET', '/test/hello/world/hel%"Flo/blah'),
null
)
})
test('safe decodeURIComponent - wildcard', t => {
t.plan(1)
const findMyWay = FindMyWay()
const fn = () => {}
findMyWay.on('GET', '/test/*', fn)
t.deepEqual(
findMyWay.find('GET', '/test/hel%"Flo'),
null
)
})
test('static routes should be inserted before parametric / 1', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/hello', () => {
t.pass('inside correct handler')
})
findMyWay.on('GET', '/test/:id', () => {
t.fail('wrong handler')
})
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
})
test('static routes should be inserted before parametric / 2', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/:id', () => {
t.fail('wrong handler')
})
findMyWay.on('GET', '/test/hello', () => {
t.pass('inside correct handler')
})
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
})
test('static routes should be inserted before parametric / 3', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/:id', () => {
t.fail('wrong handler')
})
findMyWay.on('GET', '/test', () => {
t.ok('inside correct handler')
})
findMyWay.on('GET', '/test/:id', () => {
t.fail('wrong handler')
})
findMyWay.on('GET', '/test/hello', () => {
t.ok('inside correct handler')
})
findMyWay.lookup({ method: 'GET', url: '/test', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
})
test('static routes should be inserted before parametric / 4', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/:id', () => {
t.ok('inside correct handler')
})
findMyWay.on('GET', '/test', () => {
t.fail('wrong handler')
})
findMyWay.on('GET', '/test/:id', () => {
t.ok('inside correct handler')
})
findMyWay.on('GET', '/test/hello', () => {
t.fail('wrong handler')
})
findMyWay.lookup({ method: 'GET', url: '/test/id', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/id', headers: {} }, null)
})
test('Static parametric with shared part of the path', t => {
t.plan(2)
const findMyWay = FindMyWay({
defaultRoute: (req, res) => {
t.is(req.url, '/example/shared/nested/oopss')
}
})
findMyWay.on('GET', '/example/shared/nested/test', (req, res, params) => {
t.fail('We should not be here')
})
findMyWay.on('GET', '/example/:param/nested/oops', (req, res, params) => {
t.is(params.param, 'other')
})
findMyWay.lookup({ method: 'GET', url: '/example/shared/nested/oopss', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/example/other/nested/oops', headers: {} }, null)
})
test('parametric route with different method', t => {
t.plan(2)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/test/:id', (req, res, params) => {
t.is(params.id, 'hello')
})
findMyWay.on('POST', '/test/:other', (req, res, params) => {
t.is(params.other, 'world')
})
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
findMyWay.lookup({ method: 'POST', url: '/test/world', headers: {} }, null)
})
test('params does not keep the object reference', t => {
t.plan(2)
const findMyWay = FindMyWay()
var first = true
findMyWay.on('GET', '/test/:id', (req, res, params) => {
if (first) {
setTimeout(() => {
t.is(params.id, 'hello')
}, 10)
} else {
setTimeout(() => {
t.is(params.id, 'world')
}, 10)
}
first = false
})
findMyWay.lookup({ method: 'GET', url: '/test/hello', headers: {} }, null)
findMyWay.lookup({ method: 'GET', url: '/test/world', headers: {} }, null)
})
test('Unsupported method (static)', t => {
t.plan(1)
const findMyWay = FindMyWay({
defaultRoute: (req, res) => {
t.pass('Everything ok')
}
})
findMyWay.on('GET', '/', (req, res, params) => {
t.fail('We should not be here')
})
findMyWay.lookup({ method: 'TROLL', url: '/', headers: {} }, null)
})
test('Unsupported method (wildcard)', t => {
t.plan(1)
const findMyWay = FindMyWay({
defaultRoute: (req, res) => {
t.pass('Everything ok')
}
})
findMyWay.on('GET', '*', (req, res, params) => {
t.fail('We should not be here')
})
findMyWay.lookup({ method: 'TROLL', url: '/hello/world', headers: {} }, null)
})
test('Unsupported method (static find)', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '/', () => {})
t.deepEqual(findMyWay.find('TROLL', '/'), null)
})
test('Unsupported method (wildcard find)', t => {
t.plan(1)
const findMyWay = FindMyWay()
findMyWay.on('GET', '*', () => {})
t.deepEqual(findMyWay.find('TROLL', '/hello/world'), null)
})
test('register all known HTTP methods', t => {
t.plan(6)
const findMyWay = FindMyWay()
const http = require('http')
const handlers = {}
for (var i in http.METHODS) {
var m = http.METHODS[i]
handlers[m] = function myHandler () {}
findMyWay.on(m, '/test', handlers[m])
}
t.ok(findMyWay.find('COPY', '/test'))
t.equal(findMyWay.find('COPY', '/test').handler, handlers.COPY)
t.ok(findMyWay.find('SUBSCRIBE', '/test'))
t.equal(findMyWay.find('SUBSCRIBE', '/test').handler, handlers.SUBSCRIBE)
t.ok(findMyWay.find('M-SEARCH', '/test'))
t.equal(findMyWay.find('M-SEARCH', '/test').handler, handlers['M-SEARCH'])
})