1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-12 17:09:00 -05:00

fix: url match logic of http server (denoland/deno_std#199)

Original: da188a7d30
This commit is contained in:
Yusuke Sakurai 2019-02-19 08:32:31 +09:00 committed by Ryan Dahl
parent 954fe83f62
commit bdeb6c43af
2 changed files with 84 additions and 20 deletions

View file

@ -152,25 +152,15 @@ class HttpServerImpl implements HttpServer {
async listen(addr: string, cancel: Deferred = defer()) { async listen(addr: string, cancel: Deferred = defer()) {
for await (const { req, res } of serve(addr, cancel)) { for await (const { req, res } of serve(addr, cancel)) {
let lastMatch: RegExpMatchArray; let { pathname } = new URL(req.url, addr);
let lastHandler: HttpHandler; const { index, match } = findLongestAndNearestMatch(
for (const { pattern, handler } of this.handlers) { pathname,
const match = req.url.match(pattern); this.handlers.map(v => v.pattern)
if (!match) { );
continue; req.match = match;
} if (index > -1) {
if (!lastMatch) { const { handler } = this.handlers[index];
lastMatch = match; await handler(req, res);
lastHandler = handler;
} else if (match[0].length > lastMatch[0].length) {
// use longest match
lastMatch = match;
lastHandler = handler;
}
}
req.match = lastMatch;
if (lastHandler) {
await lastHandler(req, res);
if (!res.isResponded) { if (!res.isResponded) {
await res.respond({ await res.respond({
status: 500, status: 500,
@ -187,6 +177,36 @@ class HttpServerImpl implements HttpServer {
} }
} }
/**
* Find the match that appeared in the nearest position to the beginning of word.
* If positions are same, the longest one will be picked.
* Return -1 and null if no match found.
* */
export function findLongestAndNearestMatch(
pathname: string,
patterns: (string | RegExp)[]
): { index: number; match: RegExpMatchArray } {
let lastMatchIndex = pathname.length;
let lastMatchLength = 0;
let match: RegExpMatchArray = null;
let index = -1;
for (let i = 0; i < patterns.length; i++) {
const pattern = patterns[i];
const m = pathname.match(pattern);
if (!m) continue;
if (
m.index < lastMatchIndex ||
(m.index === lastMatchIndex && m[0].length > lastMatchLength)
) {
index = i;
match = m;
lastMatchIndex = m.index;
lastMatchLength = m[0].length;
}
}
return { index, match };
}
class ServerResponderImpl implements ServerResponder { class ServerResponderImpl implements ServerResponder {
constructor(private w: Writer) {} constructor(private w: Writer) {}

View file

@ -6,10 +6,11 @@
// https://github.com/golang/go/blob/master/src/net/http/responsewrite_test.go // https://github.com/golang/go/blob/master/src/net/http/responsewrite_test.go
import { Buffer, copy, Reader } from "deno"; import { Buffer, copy, Reader } from "deno";
import { assert, assertEqual, test } from "../testing/mod.ts"; import { assert, assertEqual, runTests, test } from "../testing/mod.ts";
import { import {
createResponder, createResponder,
createServer, createServer,
findLongestAndNearestMatch,
readRequest, readRequest,
readResponse, readResponse,
ServerResponse, ServerResponse,
@ -102,6 +103,49 @@ test(async function httpReadRequestChunkedBody() {
assert.equal(dest.toString(), "deno.land"); assert.equal(dest.toString(), "deno.land");
}); });
test(function httpMatchNearest() {
assert.equal(
findLongestAndNearestMatch("/foo", ["/foo", "/bar", "/f"]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/foo", ["/foo", "/foo/bar"]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/foo/bar", [
"/",
"/foo",
"/hoo",
"/hoo/foo/bar",
"/foo/bar"
]).index,
4
);
assert.equal(
findLongestAndNearestMatch("/foo/bar/foo", ["/foo", "/foo/bar", "/bar/foo"])
.index,
1
);
assert.equal(
findLongestAndNearestMatch("/foo", ["/", "/hoo", "/hoo/foo"]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/deno/land", [/d(.+?)o/, /d(.+?)d/]).index,
1
);
assert.equal(findLongestAndNearestMatch("/foo", ["/", "/a/foo"]).index, 0);
assert.equal(
findLongestAndNearestMatch("/foo", [/\/foo/, /\/bar\/foo/]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/foo", [/\/a\/foo/, /\/foo/]).index,
1
);
});
test(async function httpServer() { test(async function httpServer() {
const server = createServer(); const server = createServer();
server.handle("/index", async (req, res) => { server.handle("/index", async (req, res) => {