Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with
or
.
Download ZIP
Browse files

fixed #2 support custom page not found handler; update to 0.2.1

  • Loading branch information...
commit 27867f100dc40ea6c98026b18a1cec31545c6758 1 parent 99b6d0b
@fengmk2 fengmk2 authored
View
1  .gitignore
@@ -1,4 +1,3 @@
node_modules
siege.log
lib-cov
-coverage.html
View
12 Makefile
@@ -1,13 +1,19 @@
TESTS = test/*.js
TESTTIMEOUT = 1000
-REPORTER = dot
-SUPPORT_VERSIONS := 1.9.0 1.8.0 1.8.5 1.8.6 1.8.7 2.2.0 2.2.1 2.2.2 2.3.0 2.3.1 2.3.2 2.3.3
+REPORTER = progress
+SUPPORT_VERSIONS := 1.9.0 1.8.0 1.8.5 1.8.6 1.8.7 \
+ 2.2.0 2.2.1 2.2.2 \
+ 2.3.0 2.3.1 2.3.2 2.3.3
test:
@NODE_ENV=test ./node_modules/.bin/mocha -R $(REPORTER) --timeout $(TESTTIMEOUT) $(TESTS)
test-cov: lib-cov
@URLROUTER_COV=1 $(MAKE) test REPORTER=html-cov > coverage.html
+ @$(MAKE) test-results
+
+test-results:
+ @$(MAKE) test REPORTER=markdown > test_results.md
lib-cov:
@rm -rf ./$@
@@ -23,4 +29,4 @@ test-version:
$(MAKE) test; \
done
-.PHONY: test test-cov clean lib-cov test-version
+.PHONY: test test-cov clean lib-cov test-version test-results
View
13 README.md
@@ -19,6 +19,8 @@ Support `connect` @1.8.x and @2.2.0+ .
$ make test-version
```
+test results: [test_results.md](https://github.com/fengmk2/urlrouter/blob/master/test_results.md)
+
## Install
```bash
@@ -49,6 +51,13 @@ connect(urlrouter(function (app) {
var http = require('http');
var urlrouter = require('urlrouter');
+var options = {
+ pageNotFound: function (req, res) {
+ res.statusCode = 404;
+ res.end('er... some page miss...');
+ }
+};
+
var router = urlrouter(function (app) {
app.get('/', function (req, res) {
res.end('GET home page' + req.url + ' , headers: ' + JSON.stringify(req.headers));
@@ -87,9 +96,9 @@ var router = urlrouter(function (app) {
});
app.options('/check', function (req, res) {
- res.end('PUT ' + req.url + ' , headers: ' + JSON.stringify(req.headers));
+ res.end('OPTIONS ' + req.url + ' , headers: ' + JSON.stringify(req.headers));
});
-});
+}, options);
http.createServer(router).listen(3000);
```
View
334 coverage.html
@@ -0,0 +1,334 @@
+<!DOCTYPE html><html><head><title>Coverage</title><script>
+
+headings = [];
+
+onload = function(){
+ headings = document.querySelectorAll('h2');
+};
+
+onscroll = function(e){
+ var heading = find(window.scrollY);
+ if (!heading) return;
+ var links = document.querySelectorAll('#menu a')
+ , link;
+
+ for (var i = 0, len = links.length; i < len; ++i) {
+ link = links[i];
+ link.className = link.getAttribute('href') == '#' + heading.id
+ ? 'active'
+ : '';
+ }
+};
+
+function find(y) {
+ var i = headings.length
+ , heading;
+
+ while (i--) {
+ heading = headings[i];
+ if (y > heading.offsetTop) {
+ return heading;
+ }
+ }
+}
+</script><style>
+
+body {
+ font: 14px/1.6 "Helvetica Neue", Helvetica, Arial, sans-serif;
+ margin: 0;
+ color: #2C2C2C;
+ border-top: 2px solid #ddd;
+}
+
+#coverage {
+ padding: 60px;
+}
+
+h1 a {
+ color: inherit;
+ font-weight: inherit;
+}
+
+h1 a:hover {
+ text-decoration: none;
+}
+
+.onload h1 {
+ opacity: 1;
+}
+
+h2 {
+ width: 80%;
+ margin-top: 80px;
+ margin-bottom: 0;
+ font-weight: 100;
+ letter-spacing: 1px;
+ border-bottom: 1px solid #eee;
+}
+
+a {
+ color: #8A6343;
+ font-weight: bold;
+ text-decoration: none;
+}
+
+a:hover {
+ text-decoration: underline;
+}
+
+ul {
+ margin-top: 20px;
+ padding: 0 15px;
+ width: 100%;
+}
+
+ul li {
+ float: left;
+ width: 40%;
+ margin-top: 5px;
+ margin-right: 60px;
+ list-style: none;
+ border-bottom: 1px solid #eee;
+ padding: 5px 0;
+ font-size: 12px;
+}
+
+ul::after {
+ content: '.';
+ height: 0;
+ display: block;
+ visibility: hidden;
+ clear: both;
+}
+
+code {
+ font: 12px monaco, monospace;
+}
+
+pre {
+ margin: 30px;
+ padding: 30px;
+ border: 1px solid #eee;
+ border-bottom-color: #ddd;
+ -webkit-border-radius: 2px;
+ -moz-border-radius: 2px;
+ -webkit-box-shadow: inset 0 0 10px #eee;
+ -moz-box-shadow: inset 0 0 10px #eee;
+ overflow-x: auto;
+}
+
+img {
+ margin: 30px;
+ padding: 1px;
+ -webkit-border-radius: 3px;
+ -moz-border-radius: 3px;
+ -webkit-box-shadow: 0 3px 10px #dedede, 0 1px 5px #888;
+ -moz-box-shadow: 0 3px 10px #dedede, 0 1px 5px #888;
+ max-width: 100%;
+}
+
+footer {
+ background: #eee;
+ width: 100%;
+ padding: 50px 0;
+ text-align: right;
+ border-top: 1px solid #ddd;
+}
+
+footer span {
+ display: block;
+ margin-right: 30px;
+ color: #888;
+ font-size: 12px;
+}
+
+#menu {
+ position: fixed;
+ font-size: 12px;
+ overflow-y: auto;
+ top: 0;
+ right: 0;
+ margin: 0;
+ height: 100%;
+ padding: 15px 0;
+ text-align: right;
+ border-left: 1px solid #eee;
+ -moz-box-shadow: 0 0 2px #888
+ , inset 5px 0 20px rgba(0,0,0,.5)
+ , inset 5px 0 3px rgba(0,0,0,.3);
+ -webkit-box-shadow: 0 0 2px #888
+ , inset 5px 0 20px rgba(0,0,0,.5)
+ , inset 5px 0 3px rgba(0,0,0,.3);
+ -webkit-font-smoothing: antialiased;
+ background: url("");
+}
+
+#logo {
+ position: fixed;
+ bottom: 10px;
+ right: 10px;
+ background: rgba(255,255,255,.1);
+ font-size: 11px;
+ display: block;
+ width: 20px;
+ height: 20px;
+ line-height: 20px;
+ text-align: center;
+ -webkit-border-radius: 20px;
+ -moz-border-radius: 20px;
+ -webkit-box-shadow: 0 0 3px rgba(0,0,0,.2);
+ -moz-box-shadow: 0 0 3px rgba(0,0,0,.2);
+ color: inherit;
+}
+
+#menu li a {
+ display: block;
+ color: white;
+ padding: 0 35px 0 25px;
+ -webkit-transition: background 300ms;
+ -moz-transition: background 300ms;
+}
+
+#menu li {
+ position: relative;
+ list-style: none;
+}
+
+#menu a:hover,
+#menu a.active {
+ text-decoration: none;
+ background: rgba(255,255,255,.1);
+}
+
+#menu li:hover .cov {
+ opacity: 1;
+}
+
+#menu li .dirname {
+ opacity: .60;
+ padding-right: 2px;
+}
+
+#menu li .basename {
+ opacity: 1;
+}
+
+#menu .cov {
+ background: rgba(0,0,0,.4);
+ position: absolute;
+ top: 0;
+ right: 8px;
+ font-size: 9px;
+ opacity: .6;
+ text-align: left;
+ width: 17px;
+ -webkit-border-radius: 10px;
+ -moz-border-radius: 10px;
+ padding: 2px 3px;
+ text-align: center;
+}
+
+#stats:nth-child(2n) {
+ display: inline-block;
+ margin-top: 15px;
+ border: 1px solid #eee;
+ padding: 10px;
+ -webkit-box-shadow: inset 0 0 2px #eee;
+ -moz-box-shadow: inset 0 0 2px #eee;
+ -webkit-border-radius: 5px;
+ -moz-border-radius: 5px;
+}
+
+#stats div {
+ float: left;
+ padding: 0 5px;
+}
+
+#stats::after {
+ display: block;
+ content: '';
+ clear: both;
+}
+
+#stats .sloc::after {
+ content: ' SLOC';
+ color: #b6b6b6;
+}
+
+#stats .percentage::after {
+ content: ' coverage';
+ color: #b6b6b6;
+}
+
+#stats .hits,
+#stats .misses {
+ display: none;
+}
+
+.high {
+ color: #00d4b4;
+}
+.medium {
+ color: #e87d0d;
+}
+.low {
+ color: #d4081a;
+}
+.terrible {
+ color: #d4081a;
+ font-weight: bold;
+}
+
+table {
+ width: 80%;
+ margin-top: 10px;
+ border-collapse: collapse;
+ border: 1px solid #cbcbcb;
+ color: #363636;
+ -webkit-border-radius: 3px;
+ -moz-border-radius: 3px;
+}
+
+table thead {
+ display: none;
+}
+
+table td.line,
+table td.hits {
+ width: 20px;
+ background: #eaeaea;
+ text-align: center;
+ font-size: 11px;
+ padding: 0 10px;
+ color: #949494;
+}
+
+table td.hits {
+ width: 10px;
+ padding: 2px 5px;
+ color: rgba(0,0,0,.2);
+ background: #f0f0f0;
+}
+
+tr.miss td.line,
+tr.miss td.hits {
+ background: #e6c3c7;
+}
+
+tr.miss td {
+ background: #f8d5d8;
+}
+
+td.source {
+ padding-left: 15px;
+ line-height: 15px;
+ white-space: pre;
+ font: 12px monaco, monospace;
+}
+
+code .comment { color: #ddd }
+code .init { color: #2F6FAD }
+code .string { color: #5890AD }
+code .keyword { color: #8A6343 }
+code .number { color: #2F6FAD }
+</style></head><body><div id="coverage"><h1 id="overview">Coverage</h1><div id="menu"><li><a href="#overview">overview</a></li><li><span class="cov high">100</span><a href="#urlrouter.js"><span class="basename">urlrouter.js</span></a></li><li><span class="cov high">100</span><a href="#utils.js"><span class="basename">utils.js</span></a></li><a id="logo" href="http://visionmedia.github.com/mocha/">m</a></div><div id="stats" class="high"><div class="percentage">100%</div><div class="sloc">68</div><div class="hits">68</div><div class="misses">0</div></div><div id="files"><div class="file"><h2 id="urlrouter.js">urlrouter.js</h2><div id="stats" class="high"><div class="percentage">100%</div><div class="sloc">35</div><div class="hits">35</div><div class="misses">0</div></div><table id="source"><thead><tr><th>Line</th><th>Hits</th><th>Source</th></tr></thead><tbody><tr><td class="line">1</td><td class="hits"></td><td class="source">/*!</td></tr><tr><td class="line">2</td><td class="hits"></td><td class="source"> * urlrouter.js</td></tr><tr><td class="line">3</td><td class="hits"></td><td class="source"> * Copyright(c) 2012 fengmk2 &lt;fengmk2@gmail.com&gt;</td></tr><tr><td class="line">4</td><td class="hits"></td><td class="source"> * MIT Licensed</td></tr><tr><td class="line">5</td><td class="hits"></td><td class="source"> */</td></tr><tr><td class="line">6</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">7</td><td class="hits">1</td><td class="source">&quot;use strict&quot;;</td></tr><tr><td class="line">8</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">9</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">10</td><td class="hits"></td><td class="source"> * Module dependencies.</td></tr><tr><td class="line">11</td><td class="hits"></td><td class="source"> */</td></tr><tr><td class="line">12</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">13</td><td class="hits">1</td><td class="source">var urlparse = require('url').parse;</td></tr><tr class="hit"><td class="line">14</td><td class="hits">1</td><td class="source">var utils = require('./utils');</td></tr><tr><td class="line">15</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">16</td><td class="hits">1</td><td class="source">var METHODS = ['get', 'post', 'put', 'delete', 'head', 'options'];</td></tr><tr><td class="line">17</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">18</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">19</td><td class="hits"></td><td class="source"> * Default page not found handler.</td></tr><tr><td class="line">20</td><td class="hits"></td><td class="source"> * </td></tr><tr><td class="line">21</td><td class="hits"></td><td class="source"> * @param {HttpRequest} req</td></tr><tr><td class="line">22</td><td class="hits"></td><td class="source"> * @param {HttpResponse} res</td></tr><tr><td class="line">23</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">24</td><td class="hits">1</td><td class="source">function pageNotFound(req, res) {</td></tr><tr class="hit"><td class="line">25</td><td class="hits">12</td><td class="source"> res.statusCode = 404;</td></tr><tr class="hit"><td class="line">26</td><td class="hits">12</td><td class="source"> res.end(req.method !== 'HEAD' &amp;&amp; req.method + ' ' + req.url + ' Not Found ');</td></tr><tr><td class="line">27</td><td class="hits"></td><td class="source">}</td></tr><tr><td class="line">28</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">29</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">30</td><td class="hits"></td><td class="source"> * Create a url router.</td></tr><tr><td class="line">31</td><td class="hits"></td><td class="source"> * </td></tr><tr><td class="line">32</td><td class="hits"></td><td class="source"> * @param {Function(app)} fn</td></tr><tr><td class="line">33</td><td class="hits"></td><td class="source"> * @param {Object} [options]</td></tr><tr><td class="line">34</td><td class="hits"></td><td class="source"> * - {String} paramsName, req[paramsName] for url router match `params`.</td></tr><tr><td class="line">35</td><td class="hits"></td><td class="source"> * - {Function(req, res)} pageNotFound, page not found handler.</td></tr><tr><td class="line">36</td><td class="hits"></td><td class="source"> * @return {Function(req, res[, next])}</td></tr><tr><td class="line">37</td><td class="hits"></td><td class="source"> * @public</td></tr><tr><td class="line">38</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">39</td><td class="hits">1</td><td class="source">function router(fn, options) {</td></tr><tr class="hit"><td class="line">40</td><td class="hits">2</td><td class="source"> var routes = [];</td></tr><tr class="hit"><td class="line">41</td><td class="hits">2</td><td class="source"> var methods = {};</td></tr><tr class="hit"><td class="line">42</td><td class="hits">2</td><td class="source"> options = options || {};</td></tr><tr class="hit"><td class="line">43</td><td class="hits">2</td><td class="source"> options.paramsName = options.paramsName || 'params';</td></tr><tr class="hit"><td class="line">44</td><td class="hits">2</td><td class="source"> options.pageNotFound = options.pageNotFound || pageNotFound;</td></tr><tr><td class="line">45</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">46</td><td class="hits">2</td><td class="source"> function createMethod(name) {</td></tr><tr class="hit"><td class="line">47</td><td class="hits">12</td><td class="source"> var localRoutes = routes[name.toUpperCase()] = [];</td></tr><tr class="hit"><td class="line">48</td><td class="hits">12</td><td class="source"> return function (urlpattern, handle) {</td></tr><tr class="hit"><td class="line">49</td><td class="hits">8</td><td class="source"> localRoutes.push([utils.createRouter(urlpattern), handle]);</td></tr><tr><td class="line">50</td><td class="hits"></td><td class="source"> };</td></tr><tr><td class="line">51</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">52</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">53</td><td class="hits">2</td><td class="source"> METHODS.forEach(function (method) {</td></tr><tr class="hit"><td class="line">54</td><td class="hits">12</td><td class="source"> methods[method] = createMethod(method);</td></tr><tr><td class="line">55</td><td class="hits"></td><td class="source"> });</td></tr><tr><td class="line">56</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">57</td><td class="hits">2</td><td class="source"> fn(methods);</td></tr><tr><td class="line">58</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">59</td><td class="hits">2</td><td class="source"> return function lookup(req, res, next) {</td></tr><tr class="hit"><td class="line">60</td><td class="hits">39</td><td class="source"> var method = req.method.toUpperCase();</td></tr><tr class="hit"><td class="line">61</td><td class="hits">39</td><td class="source"> var localRoutes = routes[method];</td></tr><tr class="hit"><td class="line">62</td><td class="hits">39</td><td class="source"> if (localRoutes &amp;&amp; localRoutes.length &gt; 0) {</td></tr><tr class="hit"><td class="line">63</td><td class="hits">36</td><td class="source"> var pathname = urlparse(req.url).pathname;</td></tr><tr class="hit"><td class="line">64</td><td class="hits">36</td><td class="source"> for (var i = 0, l = localRoutes.length; i &lt; l; i++) {</td></tr><tr class="hit"><td class="line">65</td><td class="hits">64</td><td class="source"> var route = localRoutes[i];</td></tr><tr class="hit"><td class="line">66</td><td class="hits">64</td><td class="source"> var urlroute = route[0];</td></tr><tr class="hit"><td class="line">67</td><td class="hits">64</td><td class="source"> var fn = route[1];</td></tr><tr class="hit"><td class="line">68</td><td class="hits">64</td><td class="source"> var match = urlroute.match(pathname);</td></tr><tr class="hit"><td class="line">69</td><td class="hits">64</td><td class="source"> if (match) {</td></tr><tr class="hit"><td class="line">70</td><td class="hits">26</td><td class="source"> req[options.paramsName] = match;</td></tr><tr class="hit"><td class="line">71</td><td class="hits">26</td><td class="source"> return fn(req, res, next);</td></tr><tr><td class="line">72</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">73</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">74</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">75</td><td class="hits"></td><td class="source"> // not found</td></tr><tr class="hit"><td class="line">76</td><td class="hits">13</td><td class="source"> next ? next() : options.pageNotFound(req, res);</td></tr><tr><td class="line">77</td><td class="hits"></td><td class="source"> };</td></tr><tr><td class="line">78</td><td class="hits"></td><td class="source">}</td></tr><tr><td class="line">79</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">80</td><td class="hits">1</td><td class="source">module.exports = router;</td></tr></tbody></table></div><div class="file"><h2 id="utils.js">utils.js</h2><div id="stats" class="high"><div class="percentage">100%</div><div class="sloc">33</div><div class="hits">33</div><div class="misses">0</div></div><table id="source"><thead><tr><th>Line</th><th>Hits</th><th>Source</th></tr></thead><tbody><tr><td class="line">1</td><td class="hits"></td><td class="source">/*!</td></tr><tr><td class="line">2</td><td class="hits"></td><td class="source"> * urlrouter - lib/utils.js</td></tr><tr><td class="line">3</td><td class="hits"></td><td class="source"> * Copyright(c) 2012 fengmk2 &lt;fengmk2@gmail.com&gt;</td></tr><tr><td class="line">4</td><td class="hits"></td><td class="source"> * MIT Licensed</td></tr><tr><td class="line">5</td><td class="hits"></td><td class="source"> */</td></tr><tr><td class="line">6</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">7</td><td class="hits">1</td><td class="source">&quot;use strict&quot;;</td></tr><tr><td class="line">8</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">9</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">10</td><td class="hits"></td><td class="source"> * Module dependencies.</td></tr><tr><td class="line">11</td><td class="hits"></td><td class="source"> */</td></tr><tr><td class="line">12</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">13</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">14</td><td class="hits"></td><td class="source"> * URL Router</td></tr><tr><td class="line">15</td><td class="hits"></td><td class="source"> * @param {String} url, routing url.</td></tr><tr><td class="line">16</td><td class="hits"></td><td class="source"> * e.g.: /user/:id, /user/:id([0-9]+), /user/:id.:format?</td></tr><tr><td class="line">17</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">18</td><td class="hits">1</td><td class="source">function Router(url) {</td></tr><tr class="hit"><td class="line">19</td><td class="hits">22</td><td class="source"> this.keys = null;</td></tr><tr class="hit"><td class="line">20</td><td class="hits">22</td><td class="source"> if (url instanceof RegExp) {</td></tr><tr class="hit"><td class="line">21</td><td class="hits">2</td><td class="source"> this.rex = url;</td></tr><tr class="hit"><td class="line">22</td><td class="hits">2</td><td class="source"> this.source = this.rex.source;</td></tr><tr class="hit"><td class="line">23</td><td class="hits">2</td><td class="source"> return;</td></tr><tr><td class="line">24</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">25</td><td class="hits"></td><td class="source"> </td></tr><tr class="hit"><td class="line">26</td><td class="hits">20</td><td class="source"> var keys = [];</td></tr><tr class="hit"><td class="line">27</td><td class="hits">20</td><td class="source"> this.source = url;</td></tr><tr class="hit"><td class="line">28</td><td class="hits">20</td><td class="source"> url = url.replace(/\//g, '\\/') // '/' =&gt; '\/'</td></tr><tr><td class="line">29</td><td class="hits"></td><td class="source"> .replace(/\./g, '\\.?') // '.' =&gt; '\.?'</td></tr><tr><td class="line">30</td><td class="hits"></td><td class="source"> .replace(/\*/g, '.+'); // '*' =&gt; '.+'</td></tr><tr><td class="line">31</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">32</td><td class="hits"></td><td class="source"> // ':id' =&gt; ([^\/]+), </td></tr><tr><td class="line">33</td><td class="hits"></td><td class="source"> // ':id?' =&gt; ([^\/]*), </td></tr><tr><td class="line">34</td><td class="hits"></td><td class="source"> // ':id([0-9]+)' =&gt; ([0-9]+)+, </td></tr><tr><td class="line">35</td><td class="hits"></td><td class="source"> // ':id([0-9]+)?' =&gt; ([0-9]+)* </td></tr><tr class="hit"><td class="line">36</td><td class="hits">20</td><td class="source"> url = url.replace(/:(\w+)(?:\(([^\)]+)\))?(\?)?/g, function (all, name, rex, atLeastOne) {</td></tr><tr class="hit"><td class="line">37</td><td class="hits">13</td><td class="source"> keys.push(name);</td></tr><tr class="hit"><td class="line">38</td><td class="hits">13</td><td class="source"> if (!rex) {</td></tr><tr class="hit"><td class="line">39</td><td class="hits">10</td><td class="source"> rex = '[^\\/]' + (atLeastOne === '?' ? '*' : '+');</td></tr><tr><td class="line">40</td><td class="hits"></td><td class="source"> }</td></tr><tr class="hit"><td class="line">41</td><td class="hits">13</td><td class="source"> return '(' + rex + ')';</td></tr><tr><td class="line">42</td><td class="hits"></td><td class="source"> });</td></tr><tr><td class="line">43</td><td class="hits"></td><td class="source"> // /user/:id =&gt; /user, /user/123</td></tr><tr class="hit"><td class="line">44</td><td class="hits">20</td><td class="source"> url = url.replace(/\\\/\(\[\^\\\/\]\*\)/g, '(?:\\/(\\w*))?');</td></tr><tr class="hit"><td class="line">45</td><td class="hits">20</td><td class="source"> this.keys = keys;</td></tr><tr class="hit"><td class="line">46</td><td class="hits">20</td><td class="source"> this.rex = new RegExp('^' + url + '\\/?$');</td></tr><tr><td class="line">47</td><td class="hits"></td><td class="source">}</td></tr><tr><td class="line">48</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">49</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">50</td><td class="hits"></td><td class="source"> * Try to match given pathname, if match, return the match `params`.</td></tr><tr><td class="line">51</td><td class="hits"></td><td class="source"> * </td></tr><tr><td class="line">52</td><td class="hits"></td><td class="source"> * @param {String} pathname</td></tr><tr><td class="line">53</td><td class="hits"></td><td class="source"> * @return {Object|null} match `params` or null.</td></tr><tr><td class="line">54</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">55</td><td class="hits">1</td><td class="source">Router.prototype.match = function (pathname) {</td></tr><tr class="hit"><td class="line">56</td><td class="hits">131</td><td class="source"> var m = this.rex.exec(pathname);</td></tr><tr><td class="line">57</td><td class="hits"></td><td class="source"> // console.log(this.rex, pathname, this.keys, m, this.source)</td></tr><tr class="hit"><td class="line">58</td><td class="hits">131</td><td class="source"> var match = null;</td></tr><tr class="hit"><td class="line">59</td><td class="hits">131</td><td class="source"> if (m) {</td></tr><tr class="hit"><td class="line">60</td><td class="hits">60</td><td class="source"> if (!this.keys) {</td></tr><tr class="hit"><td class="line">61</td><td class="hits">16</td><td class="source"> return m.slice(1);</td></tr><tr><td class="line">62</td><td class="hits"></td><td class="source"> }</td></tr><tr class="hit"><td class="line">63</td><td class="hits">44</td><td class="source"> match = {};</td></tr><tr class="hit"><td class="line">64</td><td class="hits">44</td><td class="source"> var keys = this.keys;</td></tr><tr class="hit"><td class="line">65</td><td class="hits">44</td><td class="source"> for (var i = 0, l = keys.length; i &lt; l; i++) {</td></tr><tr class="hit"><td class="line">66</td><td class="hits">32</td><td class="source"> var value = m[i + 1];</td></tr><tr class="hit"><td class="line">67</td><td class="hits">32</td><td class="source"> if (value) {</td></tr><tr class="hit"><td class="line">68</td><td class="hits">26</td><td class="source"> match[keys[i]] = value;</td></tr><tr><td class="line">69</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">70</td><td class="hits"></td><td class="source"> }</td></tr><tr><td class="line">71</td><td class="hits"></td><td class="source"> }</td></tr><tr class="hit"><td class="line">72</td><td class="hits">115</td><td class="source"> return match;</td></tr><tr><td class="line">73</td><td class="hits"></td><td class="source">};</td></tr><tr><td class="line">74</td><td class="hits"></td><td class="source"> </td></tr><tr><td class="line">75</td><td class="hits"></td><td class="source">/**</td></tr><tr><td class="line">76</td><td class="hits"></td><td class="source"> * Create a `Router` instance.</td></tr><tr><td class="line">77</td><td class="hits"></td><td class="source"> *</td></tr><tr><td class="line">78</td><td class="hits"></td><td class="source"> * @param {String|RegExp} urlpattern</td></tr><tr><td class="line">79</td><td class="hits"></td><td class="source"> * @return {Router}</td></tr><tr><td class="line">80</td><td class="hits"></td><td class="source"> */</td></tr><tr class="hit"><td class="line">81</td><td class="hits">1</td><td class="source">exports.createRouter = function (urlpattern) {</td></tr><tr class="hit"><td class="line">82</td><td class="hits">22</td><td class="source"> return new Router(urlpattern);</td></tr><tr><td class="line">83</td><td class="hits"></td><td class="source">};</td></tr></tbody></table></div></div></div></body></html>
View
2  example/http-hello.js
@@ -39,7 +39,7 @@ var router = urlrouter(function (app) {
});
app.options('/check', function (req, res) {
- res.end('PUT ' + req.url + ' , headers: ' + JSON.stringify(req.headers));
+ res.end('OPTIONS ' + req.url + ' , headers: ' + JSON.stringify(req.headers));
});
});
View
27 lib/urlrouter.js
@@ -4,6 +4,8 @@
* MIT Licensed
*/
+"use strict";
+
/**
* Module dependencies.
*/
@@ -13,21 +15,38 @@ var utils = require('./utils');
var METHODS = ['get', 'post', 'put', 'delete', 'head', 'options'];
-function notFound(req, res) {
+/**
+ * Default page not found handler.
+ *
+ * @param {HttpRequest} req
+ * @param {HttpResponse} res
+ */
+function pageNotFound(req, res) {
res.statusCode = 404;
res.end(req.method !== 'HEAD' && req.method + ' ' + req.url + ' Not Found ');
}
+/**
+ * Create a url router.
+ *
+ * @param {Function(app)} fn
+ * @param {Object} [options]
+ * - {String} paramsName, req[paramsName] for url router match `params`.
+ * - {Function(req, res)} pageNotFound, page not found handler.
+ * @return {Function(req, res[, next])}
+ * @public
+ */
function router(fn, options) {
var routes = [];
var methods = {};
options = options || {};
options.paramsName = options.paramsName || 'params';
+ options.pageNotFound = options.pageNotFound || pageNotFound;
function createMethod(name) {
var localRoutes = routes[name.toUpperCase()] = [];
- return function (urlroute, fn) {
- localRoutes.push([utils.createRouter(urlroute), fn]);
+ return function (urlpattern, handle) {
+ localRoutes.push([utils.createRouter(urlpattern), handle]);
};
}
@@ -54,7 +73,7 @@ function router(fn, options) {
}
}
// not found
- next ? next() : notFound(req, res);
+ next ? next() : options.pageNotFound(req, res);
};
}
View
18 lib/utils.js
@@ -4,6 +4,8 @@
* MIT Licensed
*/
+"use strict";
+
/**
* Module dependencies.
*/
@@ -44,6 +46,12 @@ function Router(url) {
this.rex = new RegExp('^' + url + '\\/?$');
}
+/**
+ * Try to match given pathname, if match, return the match `params`.
+ *
+ * @param {String} pathname
+ * @return {Object|null} match `params` or null.
+ */
Router.prototype.match = function (pathname) {
var m = this.rex.exec(pathname);
// console.log(this.rex, pathname, this.keys, m, this.source)
@@ -64,6 +72,12 @@ Router.prototype.match = function (pathname) {
return match;
};
-exports.createRouter = function (router) {
- return new Router(router);
+/**
+ * Create a `Router` instance.
+ *
+ * @param {String|RegExp} urlpattern
+ * @return {Router}
+ */
+exports.createRouter = function (urlpattern) {
+ return new Router(urlpattern);
};
View
2  package.json
@@ -2,7 +2,7 @@
"name": "urlrouter",
"description": "connect missing router middleware.",
"keywords": ["router", "url", "connect", "middleware", "express"],
- "version": "0.2.0",
+ "version": "0.2.1",
"homepage": "http://github.com/fengmk2/connect-router",
"author": "fengmk2 <fengmk2@gmail.com> (http://fengmk2.github.com)",
"repository": {
View
28 test/urlrouter.js
@@ -61,7 +61,7 @@ var router = urlrouter(function (app) {
before(function (done) {
app = m.createServer(router);
if (moduleName === 'connect') {
- app.use(urlrouter.__get__('notFound'));
+ app.use(urlrouter.__get__('pageNotFound'));
}
app = app.listen(0, done);
});
@@ -213,3 +213,29 @@ var router = urlrouter(function (app) {
});
+var routerWith404Handler = urlrouter(function (app) {
+
+}, {
+ pageNotFound: function (req, res) {
+ res.statusCode = 404;
+ res.end('oh no, page ' + req.url + ' missing...');
+ }
+});
+
+describe('options.pageNotFound()', function () {
+ var app;
+ before(function (done) {
+ app = http.createServer(routerWith404Handler);
+ app.listen(0, done);
+ });
+ after(function () {
+ app.close();
+ });
+ it('should using custom page not found handler', function (done) {
+ app.request().get('/404').end(function (res) {
+ res.should.status(404);
+ res.body.toString().should.equal('oh no, page /404 missing...');
+ done();
+ });
+ });
+});
View
599 test_results.md
@@ -0,0 +1,599 @@
+# TOC
+ - [http.createServer()](#httpcreateserver)
+ - [support RegExp()](#httpcreateserver-support-regexp)
+ - [get()](#httpcreateserver-get)
+ - [post()](#httpcreateserver-post)
+ - [put()](#httpcreateserver-put)
+ - [head()](#httpcreateserver-head)
+ - [delete()](#httpcreateserver-delete)
+ - [404 Page Not Found](#httpcreateserver-404-page-not-found)
+ - [connect.createServer()](#connectcreateserver)
+ - [support RegExp()](#connectcreateserver-support-regexp)
+ - [get()](#connectcreateserver-get)
+ - [post()](#connectcreateserver-post)
+ - [put()](#connectcreateserver-put)
+ - [head()](#connectcreateserver-head)
+ - [delete()](#connectcreateserver-delete)
+ - [404 Page Not Found](#connectcreateserver-404-page-not-found)
+ - [options.pageNotFound()](#optionspagenotfound)
+ - [utils.js](#utilsjs)
+ - [createRouter()](#utilsjs-createrouter)
+<a name="" />
+
+<a name="httpcreateserver" />
+# http.createServer()
+<a name="httpcreateserver-support-regexp" />
+## support RegExp()
+should /user 200.
+
+```js
+app.request().get('/user').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql([null, null]);
+ done();
+});
+```
+
+should /users 200.
+
+```js
+app.request().get('/users').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql([null, null]);
+ done();
+});
+```
+
+should /users/123 200.
+
+```js
+app.request().get('/users/123').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql(['123', null]);
+ done();
+});
+```
+
+should /users/mk2 200 return [null, null].
+
+```js
+app.request().get('/users/mk2').end(function (res) {
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql([null, null]);
+ done();
+});
+```
+
+should /user/123 200.
+
+```js
+app.request().get('/user/123').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql(['123', null]);
+ done();
+});
+```
+
+should /users/1..100 200.
+
+```js
+app.request().get('/users/1..100').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql(['1', '100']);
+ done();
+});
+```
+
+should /topic/9999 200.
+
+```js
+app.request().get('/topic/9999').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('topic 9999');
+ done();
+});
+```
+
+<a name="httpcreateserver-get" />
+## get()
+should return / home page.
+
+```js
+app.request().get('/').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('home page');
+ done();
+});
+```
+
+should return /foo.
+
+```js
+app.request().get('/foo').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('GET /foo');
+ done();
+});
+```
+
+<a name="httpcreateserver-post" />
+## post()
+should /post 200.
+
+```js
+app.request().post('/post').write(' helloworld').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('POST /post helloworld');
+ done();
+});
+```
+
+<a name="httpcreateserver-put" />
+## put()
+should /put 200.
+
+```js
+app.request().put('/put').write(' helloworld').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('PUT /put helloworld');
+ done();
+});
+```
+
+<a name="httpcreateserver-head" />
+## head()
+should /status 200.
+
+```js
+app.request().head('/status').end(function (res) {
+ res.should.status(200);
+ done();
+});
+```
+
+<a name="httpcreateserver-delete" />
+## delete()
+should /remove 200.
+
+```js
+app.request().delete('/remove').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('DELETE /remove');
+ done();
+});
+```
+
+<a name="httpcreateserver-404-page-not-found" />
+## 404 Page Not Found
+should get /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should post /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should put /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should delete /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should head /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should options /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+<a name="connectcreateserver" />
+# connect.createServer()
+<a name="connectcreateserver-support-regexp" />
+## support RegExp()
+should /user 200.
+
+```js
+app.request().get('/user').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql([null, null]);
+ done();
+});
+```
+
+should /users 200.
+
+```js
+app.request().get('/users').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql([null, null]);
+ done();
+});
+```
+
+should /users/123 200.
+
+```js
+app.request().get('/users/123').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql(['123', null]);
+ done();
+});
+```
+
+should /users/mk2 200 return [null, null].
+
+```js
+app.request().get('/users/mk2').end(function (res) {
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql([null, null]);
+ done();
+});
+```
+
+should /user/123 200.
+
+```js
+app.request().get('/user/123').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql(['123', null]);
+ done();
+});
+```
+
+should /users/1..100 200.
+
+```js
+app.request().get('/users/1..100').end(function (res) {
+ res.should.status(200);
+ var params = JSON.parse(res.body);
+ params.should.length(2);
+ params.should.eql(['1', '100']);
+ done();
+});
+```
+
+should /topic/9999 200.
+
+```js
+app.request().get('/topic/9999').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('topic 9999');
+ done();
+});
+```
+
+<a name="connectcreateserver-get" />
+## get()
+should return / home page.
+
+```js
+app.request().get('/').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('home page');
+ done();
+});
+```
+
+should return /foo.
+
+```js
+app.request().get('/foo').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('GET /foo');
+ done();
+});
+```
+
+<a name="connectcreateserver-post" />
+## post()
+should /post 200.
+
+```js
+app.request().post('/post').write(' helloworld').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('POST /post helloworld');
+ done();
+});
+```
+
+<a name="connectcreateserver-put" />
+## put()
+should /put 200.
+
+```js
+app.request().put('/put').write(' helloworld').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('PUT /put helloworld');
+ done();
+});
+```
+
+<a name="connectcreateserver-head" />
+## head()
+should /status 200.
+
+```js
+app.request().head('/status').end(function (res) {
+ res.should.status(200);
+ done();
+});
+```
+
+<a name="connectcreateserver-delete" />
+## delete()
+should /remove 200.
+
+```js
+app.request().delete('/remove').end(function (res) {
+ res.should.status(200);
+ res.body.toString().should.equal('DELETE /remove');
+ done();
+});
+```
+
+<a name="connectcreateserver-404-page-not-found" />
+## 404 Page Not Found
+should get /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should post /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should put /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should delete /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should head /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+should options /404 not found.
+
+```js
+app.request()[method]('/404').end(function (res) {
+ res.should.status(404);
+ if (method !== 'head') {
+ res.body.toString().should.equal(method.toUpperCase() + ' /404 Not Found ');
+ }
+ done();
+});
+```
+
+<a name="optionspagenotfound" />
+# options.pageNotFound()
+should using custom page not found handler.
+
+```js
+app.request().get('/404').end(function (res) {
+ res.should.status(404);
+ res.body.toString().should.equal('oh no, page /404 missing...');
+ done();
+});
+```
+
+<a name="utilsjs" />
+# utils.js
+<a name="utilsjs-createrouter" />
+## createRouter()
+should match regexp.
+
+```js
+var router = utils.createRouter(/^\/users?(?:\/(\d+)(?:\.\.(\d+))?)?/);
+var cases = [
+ ['/user', [undefined, undefined]],
+ ['/users', [undefined, undefined]],
+ ['/users/1..100', ['1', '100']],
+ ['/user/123', ['123', undefined]]
+];
+cases.forEach(function (item) {
+ var url = item[0];
+ var m = router.match(url);
+ should.ok(m);
+ m.should.eql(item[1]);
+});
+```
+
+should match all string.
+
+```js
+// http://expressjs.com/guide.html#routing
+ var cases = [
+ // [urlrouter, [matchs], [dont matchs]]
+ ["/user/:id",
+ [
+ ["/user/12", {id: '12'}],
+ ["/user/mk2", {id: 'mk2'}],
+ ["/user/mk2.123@$qwe,.xml-_hdhd", {id: 'mk2.123@$qwe,.xml-_hdhd'}],
+ ["/user/中文", {id: '中文'}],
+ ['/user/%E4%B8%AD%E5%8D%88', {id: '%E4%B8%AD%E5%8D%88'}]
+ ],
+ ["/user", "/", "/use/12", "/user12"]
+ ],
+ ["/users/:name?",
+ [
+ ["/users/5", {name: '5'}], ["/users", {}], ["/users/", {}]
+ ], ["/user", "/", "/user/12", "/users12"]],
+ ["/index/:i([0-9])",
+ [
+ ['/index/1', {i: '1'}, '/index/0', {i: '0'}]
+ ],
+ ['/index/', '/index', '/index/a', '/index/123', '/index/12', '/index/:i']
+ ],
+ ["/user/:name/status/:id([0-9]+)",
+ [
+ ["/user/123/status/456", {name: '123', id: '456'}],
+ ["/user/mk2/status/783972", {name: 'mk2', id: '783972'}],
+ ["/user/mk-2005_bac/status/783972", {name: 'mk-2005_bac', id: '783972'}]
+ ], ["/user/foo", "/user", "/user/", "/user/mk2/status/foo"]],
+ ["/files/*", ["/files/jquery.js", "/files/javascripts/jquery.js"], ['/files/', '/files']],
+ ["/files/*.*", ["/files/jquery.js", "/files/javascripts/jquery.js"], ['/files/', '/files']],
+ ["/user/:id/:operation?",
+ [["/user/1", {id: '1'}], ["/user/1/edit", {id: '1', operation: 'edit'}]]
+ ],
+ ["/products.:format",
+ [["/products.json", {format: 'json'}], ["/products.xml", {format: 'xml'}]],
+ [
+ "/products", "/products/"
+ // "/products."
+ ]
+ ],
+ ["/products.:format(json|xml)",
+ [["/products.json", {format: 'json'}], ["/products.xml", {format: 'xml'}]],
+ [
+ "/products", "/products/",
+ "/products.txt", "/products.html", "/products.js", "/products.xml2", "/products.rss"
+ ]
+ ],
+ ["/products.:format?",
+ ["/products.json", "/products.xml", "/products"]
+ ],
+ ["/user/:id.:format?", ["/user/12", "/user/12.json"]],
+ ["/users", ["/users", "/users/"]],
+ ["/users/", ["/users/"], ["/users", "/user"]]
+ ];
+
+ cases.forEach(function (item) {
+ var router = utils.createRouter(item[0]);
+ var matchs = item[1];
+ var unmatchs = item[2] || [];
+ matchs.forEach(function (match) {
+ var url;
+ var params = null;
+ if (Array.isArray(match)) {
+ url = match[0];
+ params = match[1];
+ } else {
+ url = match;
+ }
+ var m = router.match(url);
+ // console.log('match', m, url, router.rex, item[0]);
+ should.ok(m);
+ if (params) {
+ m.should.eql(params);
+ }
+ });
+ unmatchs.forEach(function (unmatch) {
+ var m = router.match(unmatch);
+ // console.log(unmatch, m, router.rex, item[0]);
+ should.not.exist(m);
+ });
+ });
+```
+
Please sign in to comment.
Something went wrong with that request. Please try again.