From 14e4480271df6bd7b1021573097a6db4a159c19a Mon Sep 17 00:00:00 2001 From: dig Date: Sat, 22 Dec 2018 02:31:33 +0100 Subject: [PATCH] first commit --- package-lock.json | 415 ++++++++++++++++++++++++++++++++++++++ package.json | 19 ++ src/domains-file-parse.js | 67 ++++++ src/index.js | 5 + src/main.js | 42 ++++ tests/bouncy-mock.js | 4 + tests/domains-test | 35 ++++ tests/domains-test.js | 16 ++ tests/proxy-test.js | 119 +++++++++++ tests/proxy-testcases.js | 23 +++ 10 files changed, 745 insertions(+) create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 src/domains-file-parse.js create mode 100644 src/index.js create mode 100644 src/main.js create mode 100644 tests/bouncy-mock.js create mode 100644 tests/domains-test create mode 100644 tests/domains-test.js create mode 100644 tests/proxy-test.js create mode 100644 tests/proxy-testcases.js diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..9c2df9f --- /dev/null +++ b/package-lock.json @@ -0,0 +1,415 @@ +{ + "name": "yxorp", + "version": "1.0.0", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "accepts": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.5.tgz", + "integrity": "sha1-63d99gEXI6OxTopywIBcjoZ0a9I=", + "requires": { + "mime-types": "~2.1.18", + "negotiator": "0.6.1" + } + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" + }, + "body-parser": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.2.tgz", + "integrity": "sha1-h2eKGdhLR9hZuDGZvVm84iKxBFQ=", + "requires": { + "bytes": "3.0.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.1", + "http-errors": "~1.6.2", + "iconv-lite": "0.4.19", + "on-finished": "~2.3.0", + "qs": "6.5.1", + "raw-body": "2.3.2", + "type-is": "~1.6.15" + } + }, + "bouncy": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/bouncy/-/bouncy-3.2.2.tgz", + "integrity": "sha1-gqtK176uBYkO7VS5rzxFOUsYXcc=", + "requires": { + "optimist": "~0.3.5", + "through": "~2.3.4" + } + }, + "bytes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", + "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" + }, + "colors": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.2.1.tgz", + "integrity": "sha512-s8+wktIuDSLffCywiwSxQOMqtPxML11a/dtHE17tMn4B1MSWw/C22EKf7M2KGUBcDaVFEGT+S8N02geDXeuNKg==" + }, + "content-disposition": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", + "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" + }, + "content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" + }, + "cookie": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", + "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" + }, + "cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" + }, + "destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" + }, + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" + }, + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" + }, + "esm": { + "version": "3.0.28", + "resolved": "https://registry.npmjs.org/esm/-/esm-3.0.28.tgz", + "integrity": "sha512-UHNx/zKesgkZYQ12PiXtC0ZbIP0GVM3znNNBsxyBlZyqllwdpPeTrHFjjfkgdl/ybwUwAS2rdDTDkOH7GOJMWg==" + }, + "etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" + }, + "express": { + "version": "4.16.3", + "resolved": "https://registry.npmjs.org/express/-/express-4.16.3.tgz", + "integrity": "sha1-avilAjUNsyRuzEvs9rWjTSL37VM=", + "requires": { + "accepts": "~1.3.5", + "array-flatten": "1.1.1", + "body-parser": "1.18.2", + "content-disposition": "0.5.2", + "content-type": "~1.0.4", + "cookie": "0.3.1", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "1.1.1", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.2", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.3", + "qs": "6.5.1", + "range-parser": "~1.2.0", + "safe-buffer": "5.1.1", + "send": "0.16.2", + "serve-static": "1.13.2", + "setprototypeof": "1.1.0", + "statuses": "~1.4.0", + "type-is": "~1.6.16", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + } + }, + "finalhandler": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz", + "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==", + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.2", + "statuses": "~1.4.0", + "unpipe": "~1.0.0" + } + }, + "forwarded": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", + "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" + }, + "http-errors": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + } + }, + "iconv-lite": { + "version": "0.4.19", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.19.tgz", + "integrity": "sha512-oTZqweIP51xaGPI4uPa56/Pri/480R+mo7SeU+YETByQNhDG55ycFyNLIgta9vXhILrxXDmF7ZGhqZIcuN0gJQ==" + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "ipaddr.js": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.6.0.tgz", + "integrity": "sha1-4/o1e3c9phnybpXwSdBVxyeW+Gs=" + }, + "media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" + }, + "merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" + }, + "methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" + }, + "mime": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz", + "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ==" + }, + "mime-db": { + "version": "1.33.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.33.0.tgz", + "integrity": "sha512-BHJ/EKruNIqJf/QahvxwQZXKygOQ256myeN/Ew+THcAa5q+PjyTTMMeNQC4DZw5AwfvelsUrA6B67NKMqXDbzQ==" + }, + "mime-types": { + "version": "2.1.18", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.18.tgz", + "integrity": "sha512-lc/aahn+t4/SWV/qcmumYjymLsWfN3ELhpmVuUFjgsORruuZPVSwAQryq+HHGvO/SI2KVX26bx+En+zhM8g8hQ==", + "requires": { + "mime-db": "~1.33.0" + } + }, + "minimist": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", + "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=" + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "negotiator": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz", + "integrity": "sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk=" + }, + "on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "requires": { + "ee-first": "1.1.1" + } + }, + "optimist": { + "version": "0.3.7", + "resolved": "https://registry.npmjs.org/optimist/-/optimist-0.3.7.tgz", + "integrity": "sha1-yQlBrVnkJzMokjB00s8ufLxuwNk=", + "requires": { + "wordwrap": "~0.0.2" + } + }, + "parseurl": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz", + "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M=" + }, + "path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" + }, + "proxy-addr": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.3.tgz", + "integrity": "sha512-jQTChiCJteusULxjBp8+jftSQE5Obdl3k4cnmLA6WXtK6XFuWRnvVL7aCiBqaLPM8c4ph0S4tKna8XvmIwEnXQ==", + "requires": { + "forwarded": "~0.1.2", + "ipaddr.js": "1.6.0" + } + }, + "qs": { + "version": "6.5.1", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", + "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==" + }, + "range-parser": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", + "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=" + }, + "raw-body": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.2.tgz", + "integrity": "sha1-vNYMd9Prk83gBQKVw/N5OJvIj4k=", + "requires": { + "bytes": "3.0.0", + "http-errors": "1.6.2", + "iconv-lite": "0.4.19", + "unpipe": "1.0.0" + }, + "dependencies": { + "depd": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.1.tgz", + "integrity": "sha1-V4O04cRZ8G+lyif5kfPQbnoxA1k=" + }, + "http-errors": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.2.tgz", + "integrity": "sha1-CgAsyFcHGSp+eUbO7cERVfYOxzY=", + "requires": { + "depd": "1.1.1", + "inherits": "2.0.3", + "setprototypeof": "1.0.3", + "statuses": ">= 1.3.1 < 2" + } + }, + "setprototypeof": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.0.3.tgz", + "integrity": "sha1-ZlZ+NwQ+608E2RvWWMDL77VbjgQ=" + } + } + }, + "safe-buffer": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", + "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==" + }, + "send": { + "version": "0.16.2", + "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", + "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.6.2", + "mime": "1.4.1", + "ms": "2.0.0", + "on-finished": "~2.3.0", + "range-parser": "~1.2.0", + "statuses": "~1.4.0" + } + }, + "serve-static": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.2.tgz", + "integrity": "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw==", + "requires": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.2", + "send": "0.16.2" + } + }, + "setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==" + }, + "statuses": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz", + "integrity": "sha512-zhSCtt8v2NDrRlPQpCNtw/heZLtfUDqxBM1udqikb/Hbk52LK4nQSwr10u77iopCW5LsyHpuXS0GnEc48mLeew==" + }, + "through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=" + }, + "type-is": { + "version": "1.6.16", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.16.tgz", + "integrity": "sha512-HRkVv/5qY2G6I8iab9cI7v1bOIdhm94dVjQCPFElW9W+3GeDOSHmy2EBYe4VTApuzolPcmgFTN3ftVJRKR2J9Q==", + "requires": { + "media-typer": "0.3.0", + "mime-types": "~2.1.18" + } + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" + }, + "utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" + }, + "vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" + }, + "wordwrap": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.3.tgz", + "integrity": "sha1-o9XabNXAvAAI03I0u68b7WMFkQc=" + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..df86f7c --- /dev/null +++ b/package.json @@ -0,0 +1,19 @@ +{ + "name": "yxorp", + "version": "1.0.0", + "description": "simple reverse proxy", + "main": "src/index.js", + "module": "src/main.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "author": "Thomas Di Gregorio", + "license": "MIT", + "dependencies": { + "bouncy": "^3.2.2", + "colors": "^1.2.1", + "esm": "^3.0.28", + "express": "^4.16.3", + "minimist": "^1.2.0" + } +} diff --git a/src/domains-file-parse.js b/src/domains-file-parse.js new file mode 100644 index 0000000..847c95a --- /dev/null +++ b/src/domains-file-parse.js @@ -0,0 +1,67 @@ +import 'colors' +import 'log' +import fs from 'fs' + +const empty = o=> typeof o == 'string' ? o.trim() != '' : o.length > 0 +export const domains = [] + +export default function( file, watch = true ) +{ + load( file ) + + watch && fs.watchFile( file, (curr, prev) => { + log`The domain list file changed at: ${curr.mtime}s` + domains.splice(0) + load( file ) + }) + + return domains +} + +function load( path ) +{ + log`Loading the domains file ${path.yellow}s...` + + parse( fs.readFileSync(path, 'utf-8') ) + .map( o=> domains.push(o) ) +} + +var currentDomain +function transform( line ) +{ + let res + if( line.length == 1 ) + { + currentDomain = line.pop() + return '' + } + if( line.length && line[0][line[0].length-1] == '.' ) + { + line[0] = line[0] == '.' + ? currentDomain + : line[0] + currentDomain + // line[0] = line[0].replace(/\.$/g, '') + currentDomain + // line[0] = line[0].substring(0,line[0].length-1) + // line.splice(1,0,currentDomain) + } + //line.test = line[0][0] == line[0][line[0].length-1] == '/' + line.test = ( res = /^\/(.*)\/$/.exec(line[0]) ) + ? new RegExp( res[1] ) + : new RegExp( '^' + line[0].replace(/\*/g, '(.*?)') + '$' ) + line.resolve = ( subDomain, domain, tld, match, path='PPP' )=> eval('`'+line[1]+'`') + return line +} + +function parse( text ) +{ + return text +// .replace(/\/\/.*?$/mg,'') // cut out comments + .replace(/(^|\s)#.*?$/mg,'$1') // cut out comments + .split('\n') + .filter( empty ) + .map( line=> line.split(/\s+/).filter(empty) ) + .map( transform ) + .filter( empty ) + +} + diff --git a/src/index.js b/src/index.js new file mode 100644 index 0000000..620a538 --- /dev/null +++ b/src/index.js @@ -0,0 +1,5 @@ +"use strict" + +// Set options as a parameter, environment variable, or rc file. +require = require("esm")(module/*, options*/) +module.exports = require("./main.js") diff --git a/src/main.js b/src/main.js new file mode 100644 index 0000000..ec180d7 --- /dev/null +++ b/src/main.js @@ -0,0 +1,42 @@ +import 'colors' +import 'log' +import minimist from 'minimist' +import bouncy from 'bouncy' +import domainParse from './domains-file-parse' + +const args = minimist( process.argv.slice(2) ) + +const config = { +// https: args.secure || args.s, + port: args.port || args.p || 80, + domains: args._[0] || 'domains-exemple' + } +const domains = domainParse( config.domains ) +//console.log( args, config ) + + +bouncy( (req, res, bounce) => { + + let domain, match, debug = [], + [ ,,subDomain, mainDomain, tld ] = /^((.*)\.)*(.*?)\.(.*)$/.exec( req.headers.host ) + log`[${new Date().toUTCString().grey}s] +Received a ${req.method}s request for ${`http://${req.headers.host}${req.url}`.grey}s` + log`Detected ${subDomain ? subDomain.cyan : ''}s.${mainDomain.blue}s.${tld.grey}s` + + for( domain of domains ) + if( match = domain.test.exec(req.headers.host) ) + { + let dest = domain.resolve(subDomain, mainDomain, tld, match, req.url) + log`Found a match: ${domain[0].green}s redirecting to ${dest.yellow}s` + bounce( dest ) + break + } + + if( !match ) + { + log`${'No matching domain!'.red}s` + res.statusCode = 404 + res.end('no such host') + } + +}).listen( config.port ) diff --git a/tests/bouncy-mock.js b/tests/bouncy-mock.js new file mode 100644 index 0000000..bbe6efa --- /dev/null +++ b/tests/bouncy-mock.js @@ -0,0 +1,4 @@ +module.exports = function() +{ + +} diff --git a/tests/domains-test b/tests/domains-test new file mode 100644 index 0000000..f42cd73 --- /dev/null +++ b/tests/domains-test @@ -0,0 +1,35 @@ +// Text after '//' until the end of line are comments and ignored +// Each line is parsed and splitted by space|tab +// Each line have to be 2 columns, +// one for source [required] and one for destination [optional] +// A line with only a source element act as a master domain +// A line with the source ending with '.' is a subdomain for preceding master domain + + + exemple.com + . 8080 // http://exemple.com (no subDomain) + www. 8080 + money. 9888 + ipfs. 9169/ipfs/ // http://ipfs.exemple.com/QmDdCDgSD8JnsSDF8fsGhjggh78gyGfySD3JLK + ipns. 9169/ipns/ + ipdb. 9142 // IPDB > 9 16 4 2 + + nuitdebout. 192.168.7.62:3333 + + + mycooldomain.lol + . 6776 + www. 6776 + *. 6776/${subDomain}/ + tag-*. 6776/tags/${subDomain.split('-')[1]}/ + + + nuitdebook.io 31316 // 31/03/16 +// nuitdebook.city 31316/tag/${subDomain}/ + +// /(.*?)\.nuitdebook\.city/ 31316/tag/$1/ + + +// no spaces +money.p2p.legal 8080 + diff --git a/tests/domains-test.js b/tests/domains-test.js new file mode 100644 index 0000000..92c11ea --- /dev/null +++ b/tests/domains-test.js @@ -0,0 +1,16 @@ +require('colors') +require('log') +const domains = require('../src/domains.js')('domains-exemple') + + +const list = ()=> log`===================================================== +${domains.map( D=>` +${D[0].yellow} ${D.test.toString().green} +>>> ${D[1].yellow} ${D.resolve('sub','dom','tld',['A','B','C']).green} +--------------- +`).join('')}s +` + + +setInterval( list, 1000 ) + diff --git a/tests/proxy-test.js b/tests/proxy-test.js new file mode 100644 index 0000000..3b59014 --- /dev/null +++ b/tests/proxy-test.js @@ -0,0 +1,119 @@ +const args = require('minimist')( process.argv.splice(2) ) +require('colors') +require('log') +const domains = require('../src/domains.js')('domains-exemple') +const bouncy = require('./bouncy-mock.js') + +require('./proxy-testcases.js').map( test ) + + + +function test( host ) +{ + let domain, res, debug = [], + [ ,,subDomain, mainDomain, tld ] = /^((.*)\.)*(.*?)\.(.*)$/.exec( /*req.headers.*/host ) +// console.log( /^((.*)\.)*(.*?)\.(.*)$/.exec( /*req.headers.*/host ) ) +// console.log( 'subdomain: %s\ndomain: %s\ntld: %s', subDomain, domain, tld ) + + for( domain of domains ) + { + if( res = domain.test.exec(host) ) + { + debug.push([ + domain, + domain.test, res, + domain.resolve( subDomain, mainDomain, tld, res ) + ]) + } + }/* + let debug = domains.map( arr=> { + var res//, reg = new RegExp('^'+arr[0].replace(/\* /g, '(.*?)')+'$') + //, fun = ( subDomain, domain, tld, match )=> eval('`'+arr[1]+'`') + if( res = arr.test.exec(host) ) + { + return [ arr[0], arr.test, res, arr[1], arr.resolve( subDomain, domain, tld, res ) ] + } + })*/ + + debug.length ? + log`=============================================== +${'Testing'.grey}s "${host.cyan}s" +----------------- +Split: ${(subDomain+'').bgWhite.red}s.${mainDomain.bgWhite.blue}s.${tld.grey}s +Match: ${debug[0][0][0].magenta}s\t${debug[0][0][1].yellow}s +>>> 127.0.0.1:${debug[0][3].green}s +---------------- +${debug}o +` +: log`=============================================== +${'Testing'.grey}s "${host.cyan}s" +----------------- +Split: ${(subDomain+'').bgWhite.red}s.${mainDomain.bgWhite.blue}s.${tld.grey}s +Match: ${'none'.red}s +---------------- +${debug}o +` + + +} + + + +/* +require`bouncy`( (req,res,hop) => { + req.headers.host in config && hop( config[req.headers.host] ) + req.headers.host in config && hop( config[req.headers.host] ) +}).listen( 80 ) + + + + +require`bouncy`( (req,res,hop) => { + let [ subDomain, domain, tld ] = /^(.*?)\.(.*?)\.(.*)$/.exec( req.headers.host ); +//||||___________________________________|_______|______|_______||||____________|______||_|___________________________________ +case 'nuitdebook.io' : bounce( 31316 ); break; // 31/03/16 +case 'nuitdebook.city' : bounce( 31316, { path: '/tag/' + subDomain } ); break; +default : res.statusCode = 404; res.end( 'no such host: ' + req.headers.host ); break; + +}).listen( 80 ) + +*/ + + +/* +require('bouncy')(function (req, res, bounce) { + //console.log(req.headers, req.headers.host); + var host = req.headers.host.split('.') + , tld = host.pop() + , domain = host.pop() + '.' + tld + , subDomain = host.join('.') + ;/^(.*?)\.(.*?)\.(.*)$/.exec( req.headers.host )9169 9149 14 + //res.end("d:"+domain+" | t:"+tld+" s:"+subDomain); + switch( domain ) { + +// case'(subDomain).(domain).(tld)' : bounce( port ); break; // Memo ... +// ||||_____________________|_______|______|_______||||____________|_______||_|__________________________ + case 'devingfx.com' : bounce( 8080 ); break; + case 'www.devingfx.com' : bounce( 8080 ); break; + case 'money.devingfx.com' : bounce( 9888 ); break; + case 'ipfs.devingfx.com' : bounce( 9169 ); break; // IPNS > 9 16 6 19 + case 'ipns.devingfx.com' : bounce( 9149 ); break; // IPFS > 9 16 14 19 + case 'ipdb.devingfx.com' : bounce( 9149 ); break; // IPDB > + +// case 'nicedebout.devingfx.com' : bounce( 3333 ); break; + case 'nuitdeboutnice.fr' : bounce( 6776 ); break; + case 'nuideboutnice.fr' : bounce( 6776 ); break; +// case 'www.nuitdeboutnice.fr' : bounce( 3333 ); break; +// case 'www.nuideboutnice.fr' : bounce( 3333 ); break; +// case 'www.nuitdeboutnice.fr' : bounce( 3333 ); break; +// case 'www.nuitdeboutnice.fr' : bounce( 3103 ); break; +// case 'nuitdeboutnice.fr' : bounce( 3103 ); break; + case 'nuitdebook.io' : bounce( 31316 ); break; // 31/03/16 + + case 'nuitdebook.city' : bounce( 31316, { path: '/tag/' + subDomain } ); break; + + default: res.statusCode = 404; res.end( 'no such host: ' + req.headers.host ); break; + } + +}).listen( 80 ); +*/ diff --git a/tests/proxy-testcases.js b/tests/proxy-testcases.js new file mode 100644 index 0000000..30df31c --- /dev/null +++ b/tests/proxy-testcases.js @@ -0,0 +1,23 @@ +module.exports = [ +, "exemple.com" +, "www.exemple.com" +, "sub1.exemple.com" +, "sub2.exemple.com" +, "subdomain.exemple.com" +, "multi.sub.parts.domain.exemple.com" +, "tag-toto.exemple.com" +, "tag-titi.exemple.com" +, "tag-with.dot.parts.exemple.com" +, "tag-with-hyphen-parts.exemple.com" +, "blois.exemple.city" +, "mandelieu.exemple.city" +, "exemple.city" +, "foo---explorer.exemple.com" +, "qsd---foo.exemple.com" +, "admin.google.fr" +, "admin.ghost.peer.co" +, "www.exemple.fr" +, "www.exemple.nl" +, "www.exemple.chat" +] +