11

ASP.NET 或 Nancy 等框架提供了可用于指定路由的语法,例如:

MapRoute("/customers/{id}/invoices/{invoiceId}", ...)

在 ASP.NET 中,路由有两个方向。它们可以将请求 URI 匹配/customers/32/invoices/19到路由,并且可以将参数解析{ id: 37, invoiceId: 19 }为 URI。

RFC 6570:URI 模板也定义了一个类似的,但更丰富的 URI 规范,通常用于解析URI。例如:

UriTemplate("/customers/{id}/invoices{/invoiceId}{?sort}", { id: 37, invoiceId: 19, sort: 'asc' } )
// returns:  /customers/37/invoices/19?sort=asc

我的问题是,可以使用 RFC 6570 中指定的语法将请求 URI 与路由匹配吗?是否有一部分语法会使给定 URI 与给定 URI 模板的匹配变得模棱两可?是否有任何库支持将 URI 与 URI 模板匹配?

4

2 回答 2

1

我怀疑这将非常困难。当然,像前缀语法这样的东西会使重新生成原始参数变得不可能。

对于诸如路径段扩展之类的事情

 {/list*}           /red/green/blue

你怎么知道路径的哪些部分是文字,哪些部分是参数的一部分?URITemplate 规范中有很多相当怪异的行为,我怀疑即使可以匹配,它也会相当昂贵。

您是否有兴趣为路由目的执行此操作?

于 2013-11-23T18:16:36.600 回答
1

关于匹配很简单,但关于解析,您需要用 RFC 6570 替换 ASP.net 部分。

不幸的是,我在使用 express js 的节点中执行此操作,这可能没有帮助,但我确信像https://github.com/geraintluff/uri-templates(用于解析)之类的东西在 ASP 中也可用。

下面是一些 .js 代码来说明使用 RFC 6570 重写超模式 以与 express js 一起使用(在模式中使用的优点是您还可以为您的 uri 模板定义正则表达式):

  var deref = require('json-schema-deref');
  var tv4 = require('tv4');
  var url = require('url');
  var rql = require('rql/parser');
  var hyperschema = {
  "$schema": "http://json-schema.org/draft-04/hyper-schema",
  "links": [
    {
      "href": "{/id}{/ooo*}{#q}",
      "method": "GET",
      "rel": "self",
      "schema": {
        "type": "object",
        "properties": {
          "params": {
            "type": "object",
            "properties": {
              "id": {"$ref": "#/definitions/id"}
            },
            "additionalProperties": false
          }
        },
        "additionalProperties": true
      }
    }
  ],
  "definitions": {
    "id": {
      "type": "string",
      "pattern": "[a-z]{0,3}"
    }
  }
}
// DOJO lang AND _
function getDottedProperty(object, parts, create) {
    var key;
    var i = 0;

    while (object && (key = parts[i++])) {
        if (typeof object !== 'object') {
            return undefined;
        }
        object = key in object ? object[key] : (create ? object[key] = {} : undefined);
    }

    return object;
}
function getProperty(object, propertyName, create) {
    return getDottedProperty(object, propertyName.split('.'), create);
}
function _rEscape(str) {
  return str.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
}

function getPattern(k, ldo, customCat) {
  // ...* = explode = array
  // ...: = maxLength
  var key = ((k.slice(-1) === '*') ? k.slice(0,-1) : k).split(':')[0];
  var cat = (customCat) ? customCat : 'params'; // becomes default of customCat in TS
  var pattern = '';
  if (typeof ldo === 'object' && ldo.hasOwnProperty('schema')) {
    var res = getProperty(ldo.schema, ['properties',cat,'properties',key,'pattern'].join('.'));
    if (res) {
      console.log(['properties',cat,'properties',key,'pattern'].join('.'),res);
      return ['(',res,')'].join('');
    }
  }
  return pattern;
}
function ldoToRouter(ldo) {
  var expression = ldo.href.replace(/(\{\+)/g, '{') // encoding
    .replace(/(\{\?.*\})/g, '') // query
    .replace(/\{[#]([^}]*)\}/g, function(_, arg) {
      // crosshatch
      //console.log(arg);
      return ['(?:[/]*)?#:',arg,getPattern(arg,ldo,'anchor')].join('');
    })
    .replace(/\{([./])?([^}]*)\}/g, function(_, op, arg) {
      // path seperator
      //console.log(op, '::', arg, '::', ldo.schema);
      return [op,':',arg,getPattern(arg,ldo)].join('');
    });
    return {method: ldo.method.toLowerCase(), args:[expression]};
}

deref(hyperschema, function(err, fullSchema) {
  console.log('deref hyperschema:',JSON.stringify(fullSchema));
  var router = fullSchema.links.map(ldoToRouter);

  console.log('router:',JSON.stringify(router));
});
于 2015-10-30T18:45:48.377 回答