212 lines
27 KiB
JavaScript
212 lines
27 KiB
JavaScript
import { UrlTree, UrlSegment } from './url_tree';
|
|
import { PRIMARY_OUTLET } from './shared';
|
|
import { rootNode, TreeNode } from './utils/tree';
|
|
export class UrlSerializer {
|
|
}
|
|
export class DefaultUrlSerializer {
|
|
parse(url) {
|
|
const p = new UrlParser(url);
|
|
return new UrlTree(p.parseRootSegment(), p.parseQueryParams(), p.parseFragment());
|
|
}
|
|
serialize(tree) {
|
|
const node = serializeUrlTreeNode(rootNode(tree));
|
|
const query = serializeQueryParams(tree.queryParameters);
|
|
const fragment = tree.fragment !== null ? `#${tree.fragment}` : '';
|
|
return `${node}${query}${fragment}`;
|
|
}
|
|
}
|
|
function serializeUrlTreeNode(node) {
|
|
return `${serializeSegment(node.value)}${serializeChildren(node)}`;
|
|
}
|
|
function serializeUrlTreeNodes(nodes) {
|
|
const primary = serializeSegment(nodes[0].value);
|
|
const secondaryNodes = nodes.slice(1);
|
|
const secondary = secondaryNodes.length > 0 ? `(${secondaryNodes.map(serializeUrlTreeNode).join("//")})` : "";
|
|
const children = serializeChildren(nodes[0]);
|
|
return `${primary}${secondary}${children}`;
|
|
}
|
|
function serializeChildren(node) {
|
|
if (node.children.length > 0) {
|
|
return `/${serializeUrlTreeNodes(node.children)}`;
|
|
}
|
|
else {
|
|
return "";
|
|
}
|
|
}
|
|
export function serializeSegment(segment) {
|
|
const outlet = segment.outlet === PRIMARY_OUTLET ? '' : `${segment.outlet}:`;
|
|
return `${outlet}${segment.path}${serializeParams(segment.parameters)}`;
|
|
}
|
|
function serializeParams(params) {
|
|
return pairs(params).map(p => `;${p.first}=${p.second}`).join("");
|
|
}
|
|
function serializeQueryParams(params) {
|
|
const strs = pairs(params).map(p => `${p.first}=${p.second}`);
|
|
return strs.length > 0 ? `?${strs.join("&")}` : "";
|
|
}
|
|
class Pair {
|
|
constructor(first, second) {
|
|
this.first = first;
|
|
this.second = second;
|
|
}
|
|
}
|
|
function pairs(obj) {
|
|
const res = [];
|
|
for (let prop in obj) {
|
|
if (obj.hasOwnProperty(prop)) {
|
|
res.push(new Pair(prop, obj[prop]));
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
const SEGMENT_RE = /^[^\/\(\)\?;=&#]+/;
|
|
function matchUrlSegment(str) {
|
|
SEGMENT_RE.lastIndex = 0;
|
|
var match = SEGMENT_RE.exec(str);
|
|
return match ? match[0] : '';
|
|
}
|
|
const QUERY_PARAM_VALUE_RE = /^[^\(\)\?;&#]+/;
|
|
function matchUrlQueryParamValue(str) {
|
|
QUERY_PARAM_VALUE_RE.lastIndex = 0;
|
|
const match = QUERY_PARAM_VALUE_RE.exec(str);
|
|
return match ? match[0] : '';
|
|
}
|
|
class UrlParser {
|
|
constructor(remaining) {
|
|
this.remaining = remaining;
|
|
}
|
|
peekStartsWith(str) { return this.remaining.startsWith(str); }
|
|
capture(str) {
|
|
if (!this.remaining.startsWith(str)) {
|
|
throw new Error(`Expected "${str}".`);
|
|
}
|
|
this.remaining = this.remaining.substring(str.length);
|
|
}
|
|
parseRootSegment() {
|
|
if (this.remaining == '' || this.remaining == '/') {
|
|
return new TreeNode(new UrlSegment('', {}, PRIMARY_OUTLET), []);
|
|
}
|
|
else {
|
|
const segments = this.parseSegments(false);
|
|
return new TreeNode(new UrlSegment('', {}, PRIMARY_OUTLET), segments);
|
|
}
|
|
}
|
|
parseSegments(hasOutletName) {
|
|
if (this.remaining.length == 0) {
|
|
return [];
|
|
}
|
|
if (this.peekStartsWith('/')) {
|
|
this.capture('/');
|
|
}
|
|
let path = matchUrlSegment(this.remaining);
|
|
this.capture(path);
|
|
let outletName;
|
|
if (hasOutletName) {
|
|
if (path.indexOf(":") === -1) {
|
|
throw new Error("Not outlet name is provided");
|
|
}
|
|
if (path.indexOf(":") > -1 && hasOutletName) {
|
|
let parts = path.split(":");
|
|
outletName = parts[0];
|
|
path = parts[1];
|
|
}
|
|
}
|
|
else {
|
|
if (path.indexOf(":") > -1) {
|
|
throw new Error("Not outlet name is allowed");
|
|
}
|
|
outletName = PRIMARY_OUTLET;
|
|
}
|
|
let matrixParams = {};
|
|
if (this.peekStartsWith(';')) {
|
|
matrixParams = this.parseMatrixParams();
|
|
}
|
|
let secondary = [];
|
|
if (this.peekStartsWith('(')) {
|
|
secondary = this.parseSecondarySegments();
|
|
}
|
|
let children = [];
|
|
if (this.peekStartsWith('/') && !this.peekStartsWith('//')) {
|
|
this.capture('/');
|
|
children = this.parseSegments(false);
|
|
}
|
|
const segment = new UrlSegment(path, matrixParams, outletName);
|
|
const node = new TreeNode(segment, children);
|
|
return [node].concat(secondary);
|
|
}
|
|
parseQueryParams() {
|
|
var params = {};
|
|
if (this.peekStartsWith('?')) {
|
|
this.capture('?');
|
|
this.parseQueryParam(params);
|
|
while (this.remaining.length > 0 && this.peekStartsWith('&')) {
|
|
this.capture('&');
|
|
this.parseQueryParam(params);
|
|
}
|
|
}
|
|
return params;
|
|
}
|
|
parseFragment() {
|
|
if (this.peekStartsWith('#')) {
|
|
return this.remaining.substring(1);
|
|
}
|
|
else {
|
|
return null;
|
|
}
|
|
}
|
|
parseMatrixParams() {
|
|
var params = {};
|
|
while (this.remaining.length > 0 && this.peekStartsWith(';')) {
|
|
this.capture(';');
|
|
this.parseParam(params);
|
|
}
|
|
return params;
|
|
}
|
|
parseParam(params) {
|
|
var key = matchUrlSegment(this.remaining);
|
|
if (!key) {
|
|
return;
|
|
}
|
|
this.capture(key);
|
|
var value = "true";
|
|
if (this.peekStartsWith('=')) {
|
|
this.capture('=');
|
|
var valueMatch = matchUrlSegment(this.remaining);
|
|
if (valueMatch) {
|
|
value = valueMatch;
|
|
this.capture(value);
|
|
}
|
|
}
|
|
params[key] = value;
|
|
}
|
|
parseQueryParam(params) {
|
|
var key = matchUrlSegment(this.remaining);
|
|
if (!key) {
|
|
return;
|
|
}
|
|
this.capture(key);
|
|
var value = "true";
|
|
if (this.peekStartsWith('=')) {
|
|
this.capture('=');
|
|
var valueMatch = matchUrlQueryParamValue(this.remaining);
|
|
if (valueMatch) {
|
|
value = valueMatch;
|
|
this.capture(value);
|
|
}
|
|
}
|
|
params[key] = value;
|
|
}
|
|
parseSecondarySegments() {
|
|
var segments = [];
|
|
this.capture('(');
|
|
while (!this.peekStartsWith(')') && this.remaining.length > 0) {
|
|
segments = segments.concat(this.parseSegments(true));
|
|
if (this.peekStartsWith('//')) {
|
|
this.capture('//');
|
|
}
|
|
}
|
|
this.capture(')');
|
|
return segments;
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"url_serializer.js","sourceRoot":"","sources":["../../../src/url_serializer.ts"],"names":[],"mappings":"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","sourcesContent":["import { UrlTree, UrlSegment } from './url_tree';\nimport { PRIMARY_OUTLET } from './shared';\nimport { rootNode, TreeNode } from './utils/tree';\n\n/**\n * Defines a way to serialize/deserialize a url tree.\n */\nexport abstract class UrlSerializer {\n  /**\n   * Parse a url into a {@Link UrlTree}\n   */\n  abstract parse(url: string): UrlTree;\n\n  /**\n   * Converts a {@Link UrlTree} into a url\n   */\n  abstract serialize(tree: UrlTree): string;\n}\n\n/**\n * A default implementation of the serialization.\n */\nexport class DefaultUrlSerializer implements UrlSerializer {\n  parse(url: string): UrlTree {\n    const p = new UrlParser(url);\n    return new UrlTree(p.parseRootSegment(), p.parseQueryParams(), p.parseFragment());\n  }\n\n  serialize(tree: UrlTree): string { \n    const node = serializeUrlTreeNode(rootNode(tree));\n    const query = serializeQueryParams(tree.queryParameters);\n    const fragment = tree.fragment !== null ? `#${tree.fragment}` : '';\n    return `${node}${query}${fragment}`;\n  }\n}\n\nfunction serializeUrlTreeNode(node: TreeNode<UrlSegment>): string {\n  return `${serializeSegment(node.value)}${serializeChildren(node)}`;\n}\n\nfunction serializeUrlTreeNodes(nodes: TreeNode<UrlSegment>[]): string {\n  const primary = serializeSegment(nodes[0].value);\n  const secondaryNodes = nodes.slice(1);\n  const secondary = secondaryNodes.length > 0 ? `(${secondaryNodes.map(serializeUrlTreeNode).join(\"//\")})` : \"\";\n  const children = serializeChildren(nodes[0]);\n  return `${primary}${secondary}${children}`;\n}\n\nfunction serializeChildren(node: TreeNode<UrlSegment>): string {\n  if (node.children.length > 0) {\n    return `/${serializeUrlTreeNodes(node.children)}`;\n  } else {\n    return \"\";\n  }\n}\n\nexport function serializeSegment(segment: UrlSegment): string {\n  const outlet = segment.outlet === PRIMARY_OUTLET ? '' : `${segment.outlet}:`;\n  return `${outlet}${segment.path}${serializeParams(segment.parameters)}`;\n}\n\nfunction serializeParams(params: {[key: string]: string}): string {\n  return pairs(params).map(p => `;${p.first}=${p.second}`).join(\"\");\n}\n\nfunction serializeQueryParams(params: {[key: string]: string}): string {\n  const strs = pairs(params).map(p => `${p.first}=${p.second}`);\n  return strs.length > 0 ? `?${strs.join(\"&\")}` : \"\";\n}\n\nclass Pair<A,B> { constructor(public first:A, public second:B) {} }\nfunction pairs<T>(obj: {[key: string]: T}):Pair<string,T>[] {\n  const res = [];\n  for (let prop in obj) {\n    if (obj.hasOwnProperty(prop)) {\n      res.push(new Pair<string, T>(prop, obj[prop]));\n    }\n  }\n  return res;\n}\n\nconst SEGMENT_RE = /^[^\\/\\(\\)\\?;=&#]+/;\nfunction matchUrlSegment(str: string): string {\n  SEGMENT_RE.lastIndex = 0;\n  var match = SEGMENT_RE.exec(str);\n  return match ? match[0] : '';\n}\n\nconst QUERY_PARAM_VALUE_RE = /^[^\\(\\)\\?;&#]+/;\nfunction matchUrlQueryParamValue(str: string): string {\n  QUERY_PARAM_VALUE_RE.lastIndex = 0;\n  const match = QUERY_PARAM_VALUE_RE.exec(str);\n  return match ? match[0] : '';\n}\n\nclass UrlParser {\n  constructor(private remaining: string) {}\n\n  peekStartsWith(str: string): boolean { return this.remaining.startsWith(str); }\n\n  capture(str: string): void {\n    if (!this.remaining.startsWith(str)) {\n      throw new Error(`Expected \"${str}\".`);\n    }\n    this.remaining = this.remaining.substring(str.length);\n  }\n\n  parseRootSegment(): TreeNode<UrlSegment> {\n    if (this.remaining  == '' || this.remaining == '/') {\n      return new TreeNode<UrlSegment>(new UrlSegment('', {}, PRIMARY_OUTLET), []);\n    } else {\n      const segments = this.parseSegments(false);\n      return new TreeNode<UrlSegment>(new UrlSegment('', {}, PRIMARY_OUTLET), segments);\n    }\n  }\n\n  parseSegments(hasOutletName: boolean): TreeNode<UrlSegment>[] {\n    if (this.remaining.length == 0) {\n      return [];\n    }\n    if (this.peekStartsWith('/')) {\n      this.capture('/');\n    }\n    let path = matchUrlSegment(this.remaining);\n    this.capture(path);\n\n    let outletName;\n    if (hasOutletName) {\n      if (path.indexOf(\":\") === -1) {\n        throw new Error(\"Not outlet name is provided\");\n      }\n      if (path.indexOf(\":\") > -1 && hasOutletName) {\n        let parts = path.split(\":\");\n        outletName = parts[0];\n        path = parts[1];\n      }\n    } else {\n      if (path.indexOf(\":\") > -1) {\n        throw new Error(\"Not outlet name is allowed\");\n      }\n      outletName = PRIMARY_OUTLET;\n    }\n\n    let matrixParams: {[key: string]: any} = {};\n    if (this.peekStartsWith(';')) {\n      matrixParams = this.parseMatrixParams();\n    }\n\n    let secondary = [];\n    if (this.peekStartsWith('(')) {\n      secondary = this.parseSecondarySegments();\n    }\n\n    let children: TreeNode<UrlSegment>[] = [];\n    if (this.peekStartsWith('/') && !this.peekStartsWith('//')) {\n      this.capture('/');\n      children = this.parseSegments(false);\n    }\n\n    const segment = new UrlSegment(path, matrixParams, outletName);\n    const node = new TreeNode<UrlSegment>(segment, children);\n    return [node].concat(secondary);\n  }\n\n  parseQueryParams(): {[key: string]: any} {\n    var params: {[key: string]: any} = {};\n    if (this.peekStartsWith('?')) {\n      this.capture('?');\n      this.parseQueryParam(params);\n      while (this.remaining.length > 0 && this.peekStartsWith('&')) {\n        this.capture('&');\n        this.parseQueryParam(params);\n      }\n    }\n    return params;\n  }\n\n  parseFragment(): string | null {\n    if (this.peekStartsWith('#')) {\n      return this.remaining.substring(1);\n    } else {\n      return null;\n    }\n  }\n\n  parseMatrixParams(): {[key: string]: any} {\n    var params: {[key: string]: any} = {};\n    while (this.remaining.length > 0 && this.peekStartsWith(';')) {\n      this.capture(';');\n      this.parseParam(params);\n    }\n    return params;\n  }\n\n  parseParam(params: {[key: string]: any}): void {\n    var key = matchUrlSegment(this.remaining);\n    if (!key) {\n      return;\n    }\n    this.capture(key);\n    var value: any = \"true\";\n    if (this.peekStartsWith('=')) {\n      this.capture('=');\n      var valueMatch = matchUrlSegment(this.remaining);\n      if (valueMatch) {\n        value = valueMatch;\n        this.capture(value);\n      }\n    }\n\n    params[key] = value;\n  }\n\n  parseQueryParam(params: {[key: string]: any}): void {\n    var key = matchUrlSegment(this.remaining);\n    if (!key) {\n      return;\n    }\n    this.capture(key);\n    var value: any = \"true\";\n    if (this.peekStartsWith('=')) {\n      this.capture('=');\n      var valueMatch = matchUrlQueryParamValue(this.remaining);\n      if (valueMatch) {\n        value = valueMatch;\n        this.capture(value);\n      }\n    }\n    params[key] = value;\n  }\n\n  parseSecondarySegments(): TreeNode<UrlSegment>[] {\n    var segments = [];\n    this.capture('(');\n\n    while (!this.peekStartsWith(')') && this.remaining.length > 0) {\n      segments = segments.concat(this.parseSegments(true));\n      if (this.peekStartsWith('//')) {\n        this.capture('//');\n      }\n    }\n    this.capture(')');\n\n    return segments;\n  }\n}\n"]}
|