@ditojs/router
Advanced tools
Comparing version 0.40.0 to 0.41.0
@@ -7,5 +7,4 @@ "use strict"; | ||
exports.default = void 0; | ||
const STATIC = 0; | ||
const PARAM = 1; | ||
const MATCH_ANY = 2; | ||
const PARAM = ':'; | ||
const MATCH_ANY = '*'; | ||
@@ -17,6 +16,5 @@ class Node { | ||
initialize(prefix = '/', type = STATIC, children = [], handler = null, paramNames = null) { | ||
this.label = prefix.charCodeAt(0); | ||
initialize(prefix = '/', children = {}, handler = null, paramNames = null) { | ||
this.label = prefix[0]; | ||
this.prefix = prefix; | ||
this.type = type; | ||
this.children = children; | ||
@@ -29,29 +27,9 @@ this.handler = handler; | ||
addChild(child) { | ||
this.children.push(child); | ||
this.children[child.label] = child; | ||
} | ||
findChild(label, type) { | ||
for (const child of this.children) { | ||
if (child.label === label && child.type === type) { | ||
return child; | ||
} | ||
} | ||
findChild(label) { | ||
return this.children[label]; | ||
} | ||
findChildWithLabel(label) { | ||
for (const child of this.children) { | ||
if (child.label === label) { | ||
return child; | ||
} | ||
} | ||
} | ||
findChildWithType(type) { | ||
for (const child of this.children) { | ||
if (child.type === type) { | ||
return child; | ||
} | ||
} | ||
} | ||
add(path, handler) { | ||
@@ -63,4 +41,4 @@ const paramNames = []; | ||
if (ch === ':') { | ||
this.insert(path.substring(0, pos), STATIC); | ||
if (ch === PARAM) { | ||
this.insert(path.substring(0, pos)); | ||
pos++; | ||
@@ -74,18 +52,18 @@ const start = pos; | ||
if (start === length) { | ||
return this.insert(path, PARAM, paramNames, handler); | ||
return this.insert(path, paramNames, handler); | ||
} | ||
pos = start; | ||
this.insert(path.substring(0, pos), PARAM, paramNames); | ||
} else if (ch === '*') { | ||
this.insert(path.substring(0, pos), STATIC); | ||
paramNames.push('*'); | ||
return this.insert(path, MATCH_ANY, paramNames, handler); | ||
this.insert(path.substring(0, pos), paramNames); | ||
} else if (ch === MATCH_ANY) { | ||
this.insert(path.substring(0, pos)); | ||
paramNames.push(MATCH_ANY); | ||
return this.insert(path, paramNames, handler); | ||
} | ||
} | ||
this.insert(path, STATIC, paramNames, handler); | ||
this.insert(path, paramNames, handler); | ||
} | ||
insert(path, type, paramNames, handler) { | ||
insert(path, paramNames, handler) { | ||
let current = this; | ||
@@ -100,3 +78,3 @@ | ||
while (pos < max && path.charCodeAt(pos) === prefix.charCodeAt(pos)) { | ||
while (pos < max && path[pos] === prefix[pos]) { | ||
pos++; | ||
@@ -106,10 +84,8 @@ } | ||
if (pos < prefix.length) { | ||
const node = new Node(prefix.substring(pos), current.type, current.children, current.handler, current.paramNames); | ||
const node = new Node(prefix.substring(pos), current.children, current.handler, current.paramNames); | ||
current.initialize(prefix.substring(0, pos)); | ||
current.addChild(node); | ||
if (pos === path.length) { | ||
current.type = type; | ||
} else { | ||
const node = new Node(path.substring(pos), type); | ||
if (pos < path.length) { | ||
const node = new Node(path.substring(pos)); | ||
current.addChild(node); | ||
@@ -120,3 +96,3 @@ current = node; | ||
path = path.substring(pos); | ||
const child = current.findChildWithLabel(path.charCodeAt(0)); | ||
const child = current.findChild(path[0]); | ||
@@ -128,3 +104,3 @@ if (child !== undefined) { | ||
const node = new Node(path, type); | ||
const node = new Node(path); | ||
current.addChild(node); | ||
@@ -175,7 +151,7 @@ current = node; | ||
path = path.substring(this.prefix.length); | ||
} else if (this.type !== PARAM) { | ||
} else if (this.label !== PARAM) { | ||
return null; | ||
} | ||
const staticChild = this.findChild(path.charCodeAt(0), STATIC); | ||
const staticChild = this.findChild(path[0]); | ||
@@ -194,3 +170,3 @@ if (staticChild) { | ||
const paramChild = this.findChildWithType(PARAM); | ||
const paramChild = this.findChild(PARAM); | ||
@@ -209,7 +185,7 @@ if (paramChild) { | ||
const anyChild = this.findChildWithType(MATCH_ANY); | ||
const matchAnyChild = this.findChild(MATCH_ANY); | ||
if (anyChild) { | ||
if (matchAnyChild) { | ||
paramValues.push(path); | ||
return anyChild.find('', paramValues); | ||
return matchAnyChild.find('', paramValues); | ||
} | ||
@@ -225,7 +201,5 @@ | ||
const lines = [`${format(prefix, tail, '└── ', '├── ')}${this.type === PARAM ? `${this.prefix}${this.paramName}` : this.prefix}${handler ? ` ${handler}` : ''} children=${this.children.length}`]; | ||
const children = Object.values(this.children); | ||
const lines = [`${format(prefix, tail, '└── ', '├── ')}${this.label === PARAM ? `${this.prefix}${this.paramName}` : this.prefix}${handler ? ` ${handler}` : ''} children=${children.length}`]; | ||
const str = format(prefix, tail, ' ', '│ '); | ||
const { | ||
children | ||
} = this; | ||
@@ -242,2 +216,2 @@ for (let i = 0, l = children.length - 1; i <= l; i++) { | ||
exports.default = Node; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Node.js"],"names":["STATIC","PARAM","MATCH_ANY","Node","constructor","args","initialize","prefix","type","children","handler","paramNames","label","charCodeAt","paramName","addChild","child","push","findChild","findChildWithLabel","findChildWithType","add","path","pos","length","ch","insert","substring","start","match","current","max","node","undefined","find","paramValues","params","i","name","fullMatch","startsWith","staticChild","result","paramChild","pop","anyChild","toString","tail","root","format","on","off","lines","str","l","join"],"mappings":"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","sourcesContent":["// Node Types:\nconst STATIC = 0\nconst PARAM = 1\nconst MATCH_ANY = 2\n\nexport default class Node {\n  constructor(...args) {\n    this.initialize(...args)\n  }\n\n  initialize(\n    prefix = '/',\n    type = STATIC,\n    children = [],\n    handler = null,\n    paramNames = null\n  ) {\n    this.label = prefix.charCodeAt(0)\n    this.prefix = prefix\n    this.type = type\n    this.children = children\n    this.handler = handler\n    this.paramNames = paramNames\n    this.paramName = null\n  }\n\n  addChild(child) {\n    this.children.push(child)\n  }\n\n  findChild(label, type) {\n    for (const child of this.children) {\n      if (child.label === label && child.type === type) {\n        return child\n      }\n    }\n  }\n\n  findChildWithLabel(label) {\n    for (const child of this.children) {\n      if (child.label === label) {\n        return child\n      }\n    }\n  }\n\n  findChildWithType(type) {\n    for (const child of this.children) {\n      if (child.type === type) {\n        return child\n      }\n    }\n  }\n\n  add(path, handler) {\n    const paramNames = []\n    for (let pos = 0, length = path.length; pos < length; pos++) {\n      const ch = path[pos]\n      if (ch === ':') {\n        this.insert(path.substring(0, pos), STATIC)\n        pos++ // Skip colon.\n        const start = pos\n        // Move pos to the next occurrence of the slash or the end:\n        pos += path.substring(pos).match(/^([^/]*)/)[1].length\n\n        paramNames.push(path.substring(start, pos))\n        // Chop out param name from path, but keep colon.\n        path = path.substring(0, start) + path.substring(pos)\n        length = path.length // Update length after changing path.\n\n        if (start === length) {\n          return this.insert(path, PARAM, paramNames, handler)\n        }\n        pos = start\n        this.insert(path.substring(0, pos), PARAM, paramNames)\n      } else if (ch === '*') {\n        this.insert(path.substring(0, pos), STATIC)\n        paramNames.push('*')\n        return this.insert(path, MATCH_ANY, paramNames, handler)\n      }\n    }\n    this.insert(path, STATIC, paramNames, handler)\n  }\n\n  insert(path, type, paramNames, handler) {\n    let current = this\n    while (true) {\n      // Find the position where the path and the node's prefix start diverging.\n      const { prefix } = current\n      let pos = 0\n      const max = path.length < prefix.length ? path.length : prefix.length\n      while (pos < max && path.charCodeAt(pos) === prefix.charCodeAt(pos)) {\n        pos++\n      }\n      if (pos < prefix.length) {\n        // Split node\n        const node = new Node(\n          prefix.substring(pos),\n          current.type,\n          current.children,\n          current.handler,\n          current.paramNames\n        )\n        // Reset parent node and add new node as child to it:\n        current.initialize(prefix.substring(0, pos))\n        current.addChild(node)\n        if (pos === path.length) {\n          // At parent node\n          current.type = type\n        } else {\n          // Create child node\n          const node = new Node(path.substring(pos), type)\n          current.addChild(node)\n          current = node // Switch to child to set handler\n        }\n      } else if (pos < path.length) {\n        path = path.substring(pos)\n        const child = current.findChildWithLabel(path.charCodeAt(0))\n        if (child !== undefined) {\n          // Go deeper\n          current = child\n          continue\n        }\n        // Create child node\n        const node = new Node(path, type)\n        current.addChild(node)\n        current = node // Switch to child to set handler\n      }\n      if (handler) {\n        current.handler = handler\n        current.paramNames = paramNames\n      }\n      if (paramNames) {\n        // Remember the last entry from the list of param names that keeps\n        // growing during parsing as the name of the current node.\n        current.paramName = paramNames[paramNames.length - 1]\n      }\n      break\n    }\n  }\n\n  find(path, paramValues = []) {\n    if (!path || path === this.prefix) {\n      // It's a match!\n      const { handler, paramNames } = this\n      if (handler) {\n        // Convert paramNames and values to params\n        const params = {}\n        let i = 0\n        for (const name of paramNames) {\n          params[name] = paramValues[i++]\n        }\n        return { handler, params }\n      }\n      return null\n    }\n\n    const fullMatch = path.startsWith(this.prefix)\n    if (fullMatch) {\n      path = path.substring(this.prefix.length)\n    } else if (this.type !== PARAM) {\n      // If the path doesn't fully match the prefix, we only need to look\n      // further on param nodes, which can have overlapping static children.\n      return null\n    }\n\n    // Search order: Static > Param > Match-any\n\n    // Static node\n    const staticChild = this.findChild(path.charCodeAt(0), STATIC)\n    if (staticChild) {\n      const result = staticChild.find(path, paramValues)\n      if (result) {\n        return result\n      }\n    }\n\n    // Node not found\n    if (!fullMatch) {\n      return null\n    }\n\n    // Param node\n    const paramChild = this.findChildWithType(PARAM)\n    if (paramChild) {\n      // Find the position of the next slash:\n      const pos = path.match(/^([^/]*)/)[1].length\n      paramValues.push(path.substring(0, pos))\n      const result = paramChild.find(path.substring(pos), paramValues)\n      if (result) {\n        return result\n      }\n      paramValues.pop()\n    }\n\n    // Match-any node\n    const anyChild = this.findChildWithType(MATCH_ANY)\n    if (anyChild) {\n      paramValues.push(path)\n      return anyChild.find('', paramValues) // '' == End\n    }\n\n    return null\n  }\n\n  toString(prefix = '', tail = true, root = true) {\n    const handler = this.handler && `${this.handler.name || 'ƒ'}()`\n    const format = (prefix, tail, on, off) =>\n      root ? '' : `${prefix}${tail ? on : off}`\n    const lines = [\n      `${format(prefix, tail, '└── ', '├── ')}${\n        this.type === PARAM\n          ? `${this.prefix}${this.paramName}`\n          : this.prefix\n      }${\n        handler ? ` ${handler}` : ''\n      } children=${\n        this.children.length\n      }`\n    ]\n\n    const str = format(prefix, tail, '    ', '│   ')\n    const { children } = this\n    for (let i = 0, l = children.length - 1; i <= l; i++) {\n      lines.push(children[i].toString(str, i === l, false))\n    }\n    return lines.join('\\n')\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Node.js"],"names":["PARAM","MATCH_ANY","Node","constructor","args","initialize","prefix","children","handler","paramNames","label","paramName","addChild","child","findChild","add","path","pos","length","ch","insert","substring","start","match","push","current","max","node","undefined","find","paramValues","params","i","name","fullMatch","startsWith","staticChild","result","paramChild","pop","matchAnyChild","toString","tail","root","format","on","off","Object","values","lines","str","l","join"],"mappings":";;;;;;AACA,MAAMA,KAAK,GAAG,GAAd;AACA,MAAMC,SAAS,GAAG,GAAlB;;AAEe,MAAMC,IAAN,CAAW;AACxBC,EAAAA,WAAW,CAAC,GAAGC,IAAJ,EAAU;AACnB,SAAKC,UAAL,CAAgB,GAAGD,IAAnB;AACD;;AAEDC,EAAAA,UAAU,CACRC,MAAM,GAAG,GADD,EAERC,QAAQ,GAAG,EAFH,EAGRC,OAAO,GAAG,IAHF,EAIRC,UAAU,GAAG,IAJL,EAKR;AACA,SAAKC,KAAL,GAAaJ,MAAM,CAAC,CAAD,CAAnB;AACA,SAAKA,MAAL,GAAcA,MAAd;AACA,SAAKC,QAAL,GAAgBA,QAAhB;AACA,SAAKC,OAAL,GAAeA,OAAf;AACA,SAAKC,UAAL,GAAkBA,UAAlB;AACA,SAAKE,SAAL,GAAiB,IAAjB;AACD;;AAEDC,EAAAA,QAAQ,CAACC,KAAD,EAAQ;AAEd,SAAKN,QAAL,CAAcM,KAAK,CAACH,KAApB,IAA6BG,KAA7B;AACD;;AAEDC,EAAAA,SAAS,CAACJ,KAAD,EAAQ;AACf,WAAO,KAAKH,QAAL,CAAcG,KAAd,CAAP;AACD;;AAEDK,EAAAA,GAAG,CAACC,IAAD,EAAOR,OAAP,EAAgB;AACjB,UAAMC,UAAU,GAAG,EAAnB;;AACA,SAAK,IAAIQ,GAAG,GAAG,CAAV,EAAaC,MAAM,GAAGF,IAAI,CAACE,MAAhC,EAAwCD,GAAG,GAAGC,MAA9C,EAAsDD,GAAG,EAAzD,EAA6D;AAC3D,YAAME,EAAE,GAAGH,IAAI,CAACC,GAAD,CAAf;;AACA,UAAIE,EAAE,KAAKnB,KAAX,EAAkB;AAChB,aAAKoB,MAAL,CAAYJ,IAAI,CAACK,SAAL,CAAe,CAAf,EAAkBJ,GAAlB,CAAZ;AACAA,QAAAA,GAAG;AACH,cAAMK,KAAK,GAAGL,GAAd;AAEAA,QAAAA,GAAG,IAAID,IAAI,CAACK,SAAL,CAAeJ,GAAf,EAAoBM,KAApB,CAA0B,UAA1B,EAAsC,CAAtC,EAAyCL,MAAhD;AAEAT,QAAAA,UAAU,CAACe,IAAX,CAAgBR,IAAI,CAACK,SAAL,CAAeC,KAAf,EAAsBL,GAAtB,CAAhB;AAEAD,QAAAA,IAAI,GAAGA,IAAI,CAACK,SAAL,CAAe,CAAf,EAAkBC,KAAlB,IAA2BN,IAAI,CAACK,SAAL,CAAeJ,GAAf,CAAlC;AACAC,QAAAA,MAAM,GAAGF,IAAI,CAACE,MAAd;;AAEA,YAAII,KAAK,KAAKJ,MAAd,EAAsB;AACpB,iBAAO,KAAKE,MAAL,CAAYJ,IAAZ,EAAkBP,UAAlB,EAA8BD,OAA9B,CAAP;AACD;;AACDS,QAAAA,GAAG,GAAGK,KAAN;AAEA,aAAKF,MAAL,CAAYJ,IAAI,CAACK,SAAL,CAAe,CAAf,EAAkBJ,GAAlB,CAAZ,EAAoCR,UAApC;AACD,OAlBD,MAkBO,IAAIU,EAAE,KAAKlB,SAAX,EAAsB;AAC3B,aAAKmB,MAAL,CAAYJ,IAAI,CAACK,SAAL,CAAe,CAAf,EAAkBJ,GAAlB,CAAZ;AACAR,QAAAA,UAAU,CAACe,IAAX,CAAgBvB,SAAhB;AACA,eAAO,KAAKmB,MAAL,CAAYJ,IAAZ,EAAkBP,UAAlB,EAA8BD,OAA9B,CAAP;AACD;AACF;;AACD,SAAKY,MAAL,CAAYJ,IAAZ,EAAkBP,UAAlB,EAA8BD,OAA9B;AACD;;AAEDY,EAAAA,MAAM,CAACJ,IAAD,EAAOP,UAAP,EAAmBD,OAAnB,EAA4B;AAChC,QAAIiB,OAAO,GAAG,IAAd;;AACA,WAAO,IAAP,EAAa;AAEX,YAAM;AAAEnB,QAAAA;AAAF,UAAamB,OAAnB;AACA,UAAIR,GAAG,GAAG,CAAV;AACA,YAAMS,GAAG,GAAGV,IAAI,CAACE,MAAL,GAAcZ,MAAM,CAACY,MAArB,GAA8BF,IAAI,CAACE,MAAnC,GAA4CZ,MAAM,CAACY,MAA/D;;AACA,aAAOD,GAAG,GAAGS,GAAN,IAAaV,IAAI,CAACC,GAAD,CAAJ,KAAcX,MAAM,CAACW,GAAD,CAAxC,EAA+C;AAC7CA,QAAAA,GAAG;AACJ;;AACD,UAAIA,GAAG,GAAGX,MAAM,CAACY,MAAjB,EAAyB;AAEvB,cAAMS,IAAI,GAAG,IAAIzB,IAAJ,CACXI,MAAM,CAACe,SAAP,CAAiBJ,GAAjB,CADW,EAEXQ,OAAO,CAAClB,QAFG,EAGXkB,OAAO,CAACjB,OAHG,EAIXiB,OAAO,CAAChB,UAJG,CAAb;AAOAgB,QAAAA,OAAO,CAACpB,UAAR,CAAmBC,MAAM,CAACe,SAAP,CAAiB,CAAjB,EAAoBJ,GAApB,CAAnB;AACAQ,QAAAA,OAAO,CAACb,QAAR,CAAiBe,IAAjB;;AACA,YAAIV,GAAG,GAAGD,IAAI,CAACE,MAAf,EAAuB;AAErB,gBAAMS,IAAI,GAAG,IAAIzB,IAAJ,CAASc,IAAI,CAACK,SAAL,CAAeJ,GAAf,CAAT,CAAb;AACAQ,UAAAA,OAAO,CAACb,QAAR,CAAiBe,IAAjB;AACAF,UAAAA,OAAO,GAAGE,IAAV;AACD;AACF,OAjBD,MAiBO,IAAIV,GAAG,GAAGD,IAAI,CAACE,MAAf,EAAuB;AAC5BF,QAAAA,IAAI,GAAGA,IAAI,CAACK,SAAL,CAAeJ,GAAf,CAAP;AACA,cAAMJ,KAAK,GAAGY,OAAO,CAACX,SAAR,CAAkBE,IAAI,CAAC,CAAD,CAAtB,CAAd;;AACA,YAAIH,KAAK,KAAKe,SAAd,EAAyB;AAEvBH,UAAAA,OAAO,GAAGZ,KAAV;AACA;AACD;;AAED,cAAMc,IAAI,GAAG,IAAIzB,IAAJ,CAASc,IAAT,CAAb;AACAS,QAAAA,OAAO,CAACb,QAAR,CAAiBe,IAAjB;AACAF,QAAAA,OAAO,GAAGE,IAAV;AACD;;AACD,UAAInB,OAAJ,EAAa;AACXiB,QAAAA,OAAO,CAACjB,OAAR,GAAkBA,OAAlB;AACAiB,QAAAA,OAAO,CAAChB,UAAR,GAAqBA,UAArB;AACD;;AACD,UAAIA,UAAJ,EAAgB;AAGdgB,QAAAA,OAAO,CAACd,SAAR,GAAoBF,UAAU,CAACA,UAAU,CAACS,MAAX,GAAoB,CAArB,CAA9B;AACD;;AACD;AACD;AACF;;AAEDW,EAAAA,IAAI,CAACb,IAAD,EAAOc,WAAW,GAAG,EAArB,EAAyB;AAC3B,QAAI,CAACd,IAAD,IAASA,IAAI,KAAK,KAAKV,MAA3B,EAAmC;AAEjC,YAAM;AAAEE,QAAAA,OAAF;AAAWC,QAAAA;AAAX,UAA0B,IAAhC;;AACA,UAAID,OAAJ,EAAa;AAEX,cAAMuB,MAAM,GAAG,EAAf;AACA,YAAIC,CAAC,GAAG,CAAR;;AACA,aAAK,MAAMC,IAAX,IAAmBxB,UAAnB,EAA+B;AAC7BsB,UAAAA,MAAM,CAACE,IAAD,CAAN,GAAeH,WAAW,CAACE,CAAC,EAAF,CAA1B;AACD;;AACD,eAAO;AAAExB,UAAAA,OAAF;AAAWuB,UAAAA;AAAX,SAAP;AACD;;AACD,aAAO,IAAP;AACD;;AAED,UAAMG,SAAS,GAAGlB,IAAI,CAACmB,UAAL,CAAgB,KAAK7B,MAArB,CAAlB;;AACA,QAAI4B,SAAJ,EAAe;AACblB,MAAAA,IAAI,GAAGA,IAAI,CAACK,SAAL,CAAe,KAAKf,MAAL,CAAYY,MAA3B,CAAP;AACD,KAFD,MAEO,IAAI,KAAKR,KAAL,KAAeV,KAAnB,EAA0B;AAG/B,aAAO,IAAP;AACD;;AAKD,UAAMoC,WAAW,GAAG,KAAKtB,SAAL,CAAeE,IAAI,CAAC,CAAD,CAAnB,CAApB;;AACA,QAAIoB,WAAJ,EAAiB;AACf,YAAMC,MAAM,GAAGD,WAAW,CAACP,IAAZ,CAAiBb,IAAjB,EAAuBc,WAAvB,CAAf;;AACA,UAAIO,MAAJ,EAAY;AACV,eAAOA,MAAP;AACD;AACF;;AAGD,QAAI,CAACH,SAAL,EAAgB;AACd,aAAO,IAAP;AACD;;AAGD,UAAMI,UAAU,GAAG,KAAKxB,SAAL,CAAed,KAAf,CAAnB;;AACA,QAAIsC,UAAJ,EAAgB;AAEd,YAAMrB,GAAG,GAAGD,IAAI,CAACO,KAAL,CAAW,UAAX,EAAuB,CAAvB,EAA0BL,MAAtC;AACAY,MAAAA,WAAW,CAACN,IAAZ,CAAiBR,IAAI,CAACK,SAAL,CAAe,CAAf,EAAkBJ,GAAlB,CAAjB;AACA,YAAMoB,MAAM,GAAGC,UAAU,CAACT,IAAX,CAAgBb,IAAI,CAACK,SAAL,CAAeJ,GAAf,CAAhB,EAAqCa,WAArC,CAAf;;AACA,UAAIO,MAAJ,EAAY;AACV,eAAOA,MAAP;AACD;;AACDP,MAAAA,WAAW,CAACS,GAAZ;AACD;;AAGD,UAAMC,aAAa,GAAG,KAAK1B,SAAL,CAAeb,SAAf,CAAtB;;AACA,QAAIuC,aAAJ,EAAmB;AACjBV,MAAAA,WAAW,CAACN,IAAZ,CAAiBR,IAAjB;AACA,aAAOwB,aAAa,CAACX,IAAd,CAAmB,EAAnB,EAAuBC,WAAvB,CAAP;AACD;;AAED,WAAO,IAAP;AACD;;AAEDW,EAAAA,QAAQ,CAACnC,MAAM,GAAG,EAAV,EAAcoC,IAAI,GAAG,IAArB,EAA2BC,IAAI,GAAG,IAAlC,EAAwC;AAC9C,UAAMnC,OAAO,GAAG,KAAKA,OAAL,IAAiB,GAAE,KAAKA,OAAL,CAAayB,IAAb,IAAqB,GAAI,IAA5D;;AACA,UAAMW,MAAM,GAAG,CAACtC,MAAD,EAASoC,IAAT,EAAeG,EAAf,EAAmBC,GAAnB,KACbH,IAAI,GAAG,EAAH,GAAS,GAAErC,MAAO,GAAEoC,IAAI,GAAGG,EAAH,GAAQC,GAAI,EAD1C;;AAEA,UAAMvC,QAAQ,GAAGwC,MAAM,CAACC,MAAP,CAAc,KAAKzC,QAAnB,CAAjB;AACA,UAAM0C,KAAK,GAAG,CACX,GAAEL,MAAM,CAACtC,MAAD,EAASoC,IAAT,EAAe,MAAf,EAAuB,MAAvB,CAA+B,GACtC,KAAKhC,KAAL,KAAeV,KAAf,GACK,GAAE,KAAKM,MAAO,GAAE,KAAKK,SAAU,EADpC,GAEI,KAAKL,MACV,GACCE,OAAO,GAAI,IAAGA,OAAQ,EAAf,GAAmB,EAC3B,aACCD,QAAQ,CAACW,MACV,EATW,CAAd;AAYA,UAAMgC,GAAG,GAAGN,MAAM,CAACtC,MAAD,EAASoC,IAAT,EAAe,MAAf,EAAuB,MAAvB,CAAlB;;AAEA,SAAK,IAAIV,CAAC,GAAG,CAAR,EAAWmB,CAAC,GAAG5C,QAAQ,CAACW,MAAT,GAAkB,CAAtC,EAAyCc,CAAC,IAAImB,CAA9C,EAAiDnB,CAAC,EAAlD,EAAsD;AACpDiB,MAAAA,KAAK,CAACzB,IAAN,CAAWjB,QAAQ,CAACyB,CAAD,CAAR,CAAYS,QAAZ,CAAqBS,GAArB,EAA0BlB,CAAC,KAAKmB,CAAhC,EAAmC,KAAnC,CAAX;AACD;;AACD,WAAOF,KAAK,CAACG,IAAN,CAAW,IAAX,CAAP;AACD;;AAvMuB","sourcesContent":["// Special Labels:\nconst PARAM = ':'\nconst MATCH_ANY = '*'\n\nexport default class Node {\n  constructor(...args) {\n    this.initialize(...args)\n  }\n\n  initialize(\n    prefix = '/',\n    children = {},\n    handler = null,\n    paramNames = null\n  ) {\n    this.label = prefix[0]\n    this.prefix = prefix\n    this.children = children\n    this.handler = handler\n    this.paramNames = paramNames\n    this.paramName = null\n  }\n\n  addChild(child) {\n    // No two children can have the same label in a prefix-tree so this is fine:\n    this.children[child.label] = child\n  }\n\n  findChild(label) {\n    return this.children[label]\n  }\n\n  add(path, handler) {\n    const paramNames = []\n    for (let pos = 0, length = path.length; pos < length; pos++) {\n      const ch = path[pos]\n      if (ch === PARAM) {\n        this.insert(path.substring(0, pos))\n        pos++ // Skip colon.\n        const start = pos\n        // Move pos to the next occurrence of the slash or the end:\n        pos += path.substring(pos).match(/^([^/]*)/)[1].length\n\n        paramNames.push(path.substring(start, pos))\n        // Chop out param name from path, but keep colon.\n        path = path.substring(0, start) + path.substring(pos)\n        length = path.length // Update length after changing path.\n\n        if (start === length) {\n          return this.insert(path, paramNames, handler)\n        }\n        pos = start\n        // We need to include paramNames here for toString() and nested queries.\n        this.insert(path.substring(0, pos), paramNames)\n      } else if (ch === MATCH_ANY) {\n        this.insert(path.substring(0, pos))\n        paramNames.push(MATCH_ANY)\n        return this.insert(path, paramNames, handler)\n      }\n    }\n    this.insert(path, paramNames, handler)\n  }\n\n  insert(path, paramNames, handler) {\n    let current = this\n    while (true) {\n      // Find the position where the path and the node's prefix start diverging\n      const { prefix } = current\n      let pos = 0\n      const max = path.length < prefix.length ? path.length : prefix.length\n      while (pos < max && path[pos] === prefix[pos]) {\n        pos++\n      }\n      if (pos < prefix.length) {\n        // Split node\n        const node = new Node(\n          prefix.substring(pos),\n          current.children,\n          current.handler,\n          current.paramNames\n        )\n        // Reset parent node and add new node as child to it:\n        current.initialize(prefix.substring(0, pos))\n        current.addChild(node)\n        if (pos < path.length) {\n          // Create child node\n          const node = new Node(path.substring(pos))\n          current.addChild(node)\n          current = node // Switch to child to set handler\n        }\n      } else if (pos < path.length) {\n        path = path.substring(pos)\n        const child = current.findChild(path[0])\n        if (child !== undefined) {\n          // Go deeper\n          current = child\n          continue\n        }\n        // Create child node\n        const node = new Node(path)\n        current.addChild(node)\n        current = node // Switch to child to set handler\n      }\n      if (handler) {\n        current.handler = handler\n        current.paramNames = paramNames\n      }\n      if (paramNames) {\n        // Remember the last entry from the list of param names that keeps\n        // growing during parsing as the name of the current node.\n        current.paramName = paramNames[paramNames.length - 1]\n      }\n      break\n    }\n  }\n\n  find(path, paramValues = []) {\n    if (!path || path === this.prefix) {\n      // It's a match!\n      const { handler, paramNames } = this\n      if (handler) {\n        // Convert paramNames and values to params\n        const params = {}\n        let i = 0\n        for (const name of paramNames) {\n          params[name] = paramValues[i++]\n        }\n        return { handler, params }\n      }\n      return null\n    }\n\n    const fullMatch = path.startsWith(this.prefix)\n    if (fullMatch) {\n      path = path.substring(this.prefix.length)\n    } else if (this.label !== PARAM) {\n      // If the path doesn't fully match the prefix, we only need to look\n      // further on param nodes, which can have overlapping static children.\n      return null\n    }\n\n    // Search order: Static > Param > Match-any\n\n    // Static node\n    const staticChild = this.findChild(path[0])\n    if (staticChild) {\n      const result = staticChild.find(path, paramValues)\n      if (result) {\n        return result\n      }\n    }\n\n    // Node not found\n    if (!fullMatch) {\n      return null\n    }\n\n    // Param node\n    const paramChild = this.findChild(PARAM)\n    if (paramChild) {\n      // Find the position of the next slash:\n      const pos = path.match(/^([^/]*)/)[1].length\n      paramValues.push(path.substring(0, pos))\n      const result = paramChild.find(path.substring(pos), paramValues)\n      if (result) {\n        return result\n      }\n      paramValues.pop()\n    }\n\n    // Match-any node\n    const matchAnyChild = this.findChild(MATCH_ANY)\n    if (matchAnyChild) {\n      paramValues.push(path)\n      return matchAnyChild.find('', paramValues) // '' == End\n    }\n\n    return null\n  }\n\n  toString(prefix = '', tail = true, root = true) {\n    const handler = this.handler && `${this.handler.name || 'ƒ'}()`\n    const format = (prefix, tail, on, off) =>\n      root ? '' : `${prefix}${tail ? on : off}`\n    const children = Object.values(this.children)\n    const lines = [\n      `${format(prefix, tail, '└── ', '├── ')}${\n        this.label === PARAM\n          ? `${this.prefix}${this.paramName}`\n          : this.prefix\n      }${\n        handler ? ` ${handler}` : ''\n      } children=${\n        children.length\n      }`\n    ]\n\n    const str = format(prefix, tail, '    ', '│   ')\n\n    for (let i = 0, l = children.length - 1; i <= l; i++) {\n      lines.push(children[i].toString(str, i === l, false))\n    }\n    return lines.join('\\n')\n  }\n}\n"]} |
{ | ||
"name": "@ditojs/router", | ||
"version": "0.40.0", | ||
"version": "0.41.0", | ||
"description": "Dito.js Router – Dito.js is a declarative and modern web framework, based on Objection.js, Koa.js and Vue.js", | ||
@@ -29,3 +29,3 @@ "main": "lib/index.js", | ||
"devDependencies": { | ||
"@ditojs/utils": "^0.40.0" | ||
"@ditojs/utils": "^0.41.0" | ||
}, | ||
@@ -39,3 +39,3 @@ "keywords": [ | ||
], | ||
"gitHead": "19103cf1701a5b7c2a46c52075bdc160a28bfb97" | ||
"gitHead": "41f5f6724fec81c02f0cf16359bd1d795c3e8aff" | ||
} |
@@ -1,5 +0,4 @@ | ||
// Node Types: | ||
const STATIC = 0 | ||
const PARAM = 1 | ||
const MATCH_ANY = 2 | ||
// Special Labels: | ||
const PARAM = ':' | ||
const MATCH_ANY = '*' | ||
@@ -13,10 +12,8 @@ export default class Node { | ||
prefix = '/', | ||
type = STATIC, | ||
children = [], | ||
children = {}, | ||
handler = null, | ||
paramNames = null | ||
) { | ||
this.label = prefix.charCodeAt(0) | ||
this.label = prefix[0] | ||
this.prefix = prefix | ||
this.type = type | ||
this.children = children | ||
@@ -29,29 +26,10 @@ this.handler = handler | ||
addChild(child) { | ||
this.children.push(child) | ||
// No two children can have the same label in a prefix-tree so this is fine: | ||
this.children[child.label] = child | ||
} | ||
findChild(label, type) { | ||
for (const child of this.children) { | ||
if (child.label === label && child.type === type) { | ||
return child | ||
} | ||
} | ||
findChild(label) { | ||
return this.children[label] | ||
} | ||
findChildWithLabel(label) { | ||
for (const child of this.children) { | ||
if (child.label === label) { | ||
return child | ||
} | ||
} | ||
} | ||
findChildWithType(type) { | ||
for (const child of this.children) { | ||
if (child.type === type) { | ||
return child | ||
} | ||
} | ||
} | ||
add(path, handler) { | ||
@@ -61,4 +39,4 @@ const paramNames = [] | ||
const ch = path[pos] | ||
if (ch === ':') { | ||
this.insert(path.substring(0, pos), STATIC) | ||
if (ch === PARAM) { | ||
this.insert(path.substring(0, pos)) | ||
pos++ // Skip colon. | ||
@@ -75,23 +53,24 @@ const start = pos | ||
if (start === length) { | ||
return this.insert(path, PARAM, paramNames, handler) | ||
return this.insert(path, paramNames, handler) | ||
} | ||
pos = start | ||
this.insert(path.substring(0, pos), PARAM, paramNames) | ||
} else if (ch === '*') { | ||
this.insert(path.substring(0, pos), STATIC) | ||
paramNames.push('*') | ||
return this.insert(path, MATCH_ANY, paramNames, handler) | ||
// We need to include paramNames here for toString() and nested queries. | ||
this.insert(path.substring(0, pos), paramNames) | ||
} else if (ch === MATCH_ANY) { | ||
this.insert(path.substring(0, pos)) | ||
paramNames.push(MATCH_ANY) | ||
return this.insert(path, paramNames, handler) | ||
} | ||
} | ||
this.insert(path, STATIC, paramNames, handler) | ||
this.insert(path, paramNames, handler) | ||
} | ||
insert(path, type, paramNames, handler) { | ||
insert(path, paramNames, handler) { | ||
let current = this | ||
while (true) { | ||
// Find the position where the path and the node's prefix start diverging. | ||
// Find the position where the path and the node's prefix start diverging | ||
const { prefix } = current | ||
let pos = 0 | ||
const max = path.length < prefix.length ? path.length : prefix.length | ||
while (pos < max && path.charCodeAt(pos) === prefix.charCodeAt(pos)) { | ||
while (pos < max && path[pos] === prefix[pos]) { | ||
pos++ | ||
@@ -103,3 +82,2 @@ } | ||
prefix.substring(pos), | ||
current.type, | ||
current.children, | ||
@@ -112,8 +90,5 @@ current.handler, | ||
current.addChild(node) | ||
if (pos === path.length) { | ||
// At parent node | ||
current.type = type | ||
} else { | ||
if (pos < path.length) { | ||
// Create child node | ||
const node = new Node(path.substring(pos), type) | ||
const node = new Node(path.substring(pos)) | ||
current.addChild(node) | ||
@@ -124,3 +99,3 @@ current = node // Switch to child to set handler | ||
path = path.substring(pos) | ||
const child = current.findChildWithLabel(path.charCodeAt(0)) | ||
const child = current.findChild(path[0]) | ||
if (child !== undefined) { | ||
@@ -132,3 +107,3 @@ // Go deeper | ||
// Create child node | ||
const node = new Node(path, type) | ||
const node = new Node(path) | ||
current.addChild(node) | ||
@@ -169,3 +144,3 @@ current = node // Switch to child to set handler | ||
path = path.substring(this.prefix.length) | ||
} else if (this.type !== PARAM) { | ||
} else if (this.label !== PARAM) { | ||
// If the path doesn't fully match the prefix, we only need to look | ||
@@ -179,3 +154,3 @@ // further on param nodes, which can have overlapping static children. | ||
// Static node | ||
const staticChild = this.findChild(path.charCodeAt(0), STATIC) | ||
const staticChild = this.findChild(path[0]) | ||
if (staticChild) { | ||
@@ -194,3 +169,3 @@ const result = staticChild.find(path, paramValues) | ||
// Param node | ||
const paramChild = this.findChildWithType(PARAM) | ||
const paramChild = this.findChild(PARAM) | ||
if (paramChild) { | ||
@@ -208,6 +183,6 @@ // Find the position of the next slash: | ||
// Match-any node | ||
const anyChild = this.findChildWithType(MATCH_ANY) | ||
if (anyChild) { | ||
const matchAnyChild = this.findChild(MATCH_ANY) | ||
if (matchAnyChild) { | ||
paramValues.push(path) | ||
return anyChild.find('', paramValues) // '' == End | ||
return matchAnyChild.find('', paramValues) // '' == End | ||
} | ||
@@ -222,5 +197,6 @@ | ||
root ? '' : `${prefix}${tail ? on : off}` | ||
const children = Object.values(this.children) | ||
const lines = [ | ||
`${format(prefix, tail, '└── ', '├── ')}${ | ||
this.type === PARAM | ||
this.label === PARAM | ||
? `${this.prefix}${this.paramName}` | ||
@@ -231,3 +207,3 @@ : this.prefix | ||
} children=${ | ||
this.children.length | ||
children.length | ||
}` | ||
@@ -237,3 +213,3 @@ ] | ||
const str = format(prefix, tail, ' ', '│ ') | ||
const { children } = this | ||
for (let i = 0, l = children.length - 1; i <= l; i++) { | ||
@@ -240,0 +216,0 @@ lines.push(children[i].toString(str, i === l, false)) |
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
159238
1622