diff --git a/.github/workflows/node.js.yml b/.github/workflows/node.js.yml index 296da15..c076ade 100644 --- a/.github/workflows/node.js.yml +++ b/.github/workflows/node.js.yml @@ -5,14 +5,13 @@ name: Node.js CI on: push: - branches: [ "main" ] + branches: ["main"] pull_request: - branches: [ "main" ] + branches: ["main"] workflow_dispatch: jobs: build: - runs-on: ubuntu-latest strategy: @@ -21,12 +20,12 @@ jobs: # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ steps: - - uses: actions/checkout@v4 - - name: Use Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v4 - with: - node-version: ${{ matrix.node-version }} - cache: 'npm' - - run: npm ci - - run: npm run build - - run: npm run coverage + - uses: actions/checkout@v4 + - name: Use Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v4 + with: + node-version: ${{ matrix.node-version }} + cache: "npm" + - run: npm ci + - run: npm run build + - run: npm run coverage diff --git a/README.md b/README.md index 4cb5b21..5963529 100644 --- a/README.md +++ b/README.md @@ -1,35 +1,42 @@ -# h2tunnel +# h2tunnel - TCP over HTTP/2 [![NPM Version](https://img.shields.io/npm/v/h2tunnel)](https://www.npmjs.com/package/h2tunnel) [![GitHub Actions Workflow Status](https://img.shields.io/github/actions/workflow/status/boronine/h2tunnel/node.js.yml)](https://github.com/boronine/h2tunnel/actions/workflows/node.js.yml) -A CLI tool and Node.js library for a popular "tunneling" workflow, similar to the proprietary [ngrok](https://ngrok.com/) -or the openssh-based `ssh -R` solution. All in [less than 500 LOC](https://github.com/boronine/h2tunnel/blob/main/src/h2tunnel.ts) +A CLI tool and Node.js library for a popular "tunneling" workflow, like the proprietary [ngrok](https://ngrok.com/) +or the [`ssh -R` solution](https://www.ssh.com/academy/ssh/tunneling-example#remote-forwarding). + +The client (localhost) establishes a tunnel to the server (public IP), and the server forwards incoming connections to +your local machine through this tunnel. In effect, your local server becomes publically available. + +All in [less than 500 LOC](https://github.com/boronine/h2tunnel/blob/main/src/h2tunnel.ts) with no dependencies. ![Diagram](https://raw.githubusercontent.com/boronine/h2tunnel/main/diagram.drawio.svg) -## The "tunneling" workflow +## How does h2tunnel work? -This workflow allows exposing your localhost development server to the internet. This requires a server component -hosted on a public IP address, and a client component running on your local machine. The client establishes a tunnel -to the server, and the server proxies requests through this tunnel to your local machine. +h2tunnel is unique among [its many alternatives](https://github.com/anderspitman/awesome-tunneling) for the way it +leverages existing protocols: -## How does h2tunnel work? +1. The client initiates a TLS connection to the server and starts listening for HTTP/2 sessions +2. The server receives a TLS connection and initiates an HTTP/2 session through it +3. The server takes incoming TCP connections, converts them into HTTP/2 streams, and forwards them to the client +4. The client receives these HTTP/2 streams, converts them back into TCP connections for the local server -1. The client initiates a TLS connection to the server and starts listening for HTTP2 sessions -2. The server takes the newly created TLS socket and initiates an HTTP2 session through it -3. The server starts accepting TCP connections, converting them into HTTP2 streams, and fowarding them to the client -4. The client receives these HTTP2 streams and converts them back into TCP connections to feed them into the local server +The purpose of using HTTP/2 is to take advantage of its multiplexing capability. This feature of the protocol allows +simultaneous requests to be processed on a single TCP connection (the "tunnel"). -The purpose of using HTTP2 is to take advantage of its multiplexing capabilities. This feature of the protocol allows -simultaneous requests to be processed on a single TCP connection. +For authentication we use a self-signed TLS certificate + private key pair. This pair is used by both the client and +the server, and both are configured to reject anything else. The pair is effectively a shared password. -For authentication we use a self-signed TLS certificate + private key pair. This pair is used by both the client and -the server, and both are configured to reject anything else. This way, the pair effectively becomes a shared password. +## Installation -## Usage +```bash +npm install -g h2tunnel +``` +## Usage ``` usage: h2tunnel [options] @@ -37,7 +44,7 @@ usage: h2tunnel [options] commands: client server - + client options: --crt Path to certificate file (.crt) --key Path to private key file (.key) @@ -53,7 +60,7 @@ server options: --tunnel-listen-port Port for the tunnel server to listen on --proxy-listen-ip IP for the remote TCP proxy server to bind on (default: 0.0.0.0) --proxy-listen-port Port for the remote TCP proxy server to listen on - + The tunnel and proxy servers will bind to 0.0.0.0 by default which will make them publically available. This requires superuser permissions on Linux. You can change this setting to bind to a specific network interface, e.g. a VPN, but this is advanced usage. @@ -67,38 +74,38 @@ openssl req -x509 -newkey ec -pkeyopt ec_paramgen_curve:secp384r1 -days 3650 -no ### Forward localhost:8000 to http://mysite.example.com -On your server (example.com), we will be listening for tunnel connections on port 15001, and providing an HTTP proxy +On your server (example.com), we will be listening for tunnel connections on port 15001, and providing an HTTP proxy on port 80. Make sure these are open in your firewall. ```bash -# sudo is required to bind to 0.0.0.0, which makes the tunnel server and proxy server publically available +# sudo is required to bind to 0.0.0.0, which is necessary for public access sudo h2tunnel server \ --crt h2tunnel.crt \ --key h2tunnel.key \ --tunnel-listen-port 15001 \ --proxy-listen-port 80 -```` +``` -On your local machine, we will connect to the tunnel and forward a local HTTP server on port 8000. +On your local machine, we will connect to the tunnel and forward a local HTTP server on port 8000. ```bash -python3 -m http.server # runs on port 8000 by default h2tunnel client \ --crt h2tunnel.crt \ --key h2tunnel.key \ --tunnel-host=mysite.example.com \ --tunnel-port=15001 \ --local-http-port=8000 + +# If you have python3 installed, you can test using this built-in HTTP server +python3 -m http.server ``` ### Forward localhost:8000 to https://mysite.example.com This is the same as the previous example, but with an extra layer: a [Caddy](https://caddyserver.com/) reverse proxy -that will auto-provision TLS certificates for your domain. This is useful if you want to expose a local HTTP server +that will auto-provision TLS certificates for your domain. This is useful if you want to expose a local HTTP server as HTTPS. -The client command line is the same as before, but for the server we will use a docker compose setup. - Specify your domain in the `.env` file: ``` @@ -108,30 +115,33 @@ TUNNEL_DOMAIN=mysite.example.com Push the necessary files to the server: ```bash -scp .env Caddyfile Dockerfile docker-compose.yml h2tunnel.crt h2tunnel.key example.com:/home/myuser +scp .env Caddyfile Dockerfile docker-compose.yml h2tunnel.crt h2tunnel.key mysite.example.com:/home/myuser ``` Start the server: ```bash -docker compose up +ssh myuser@mysite.example.com +docker compose up ``` +To connect to your tunnel, run the same client command as in the above recipe. + ### Use as a library You can integrate h2tunnel into your own Node.js application by importing the `TunnelServer` and `TunnelClient` classes. ```typescript -import {TunnelClient} from "h2tunnel"; +import { TunnelClient } from "h2tunnel"; const client = new TunnelClient({ - logger: (line) => console.log(line), // optional - key: `-----BEGIN PRIVATE KEY----- ...`, - cert: `-----BEGIN CERTIFICATE----- ...`, - originHost: "localhost", // optional - originPort: 8000, - tunnelHost: `mysite.example.com`, - tunnelPort: 15001, + logger: (line) => console.log(line), // optional + key: `-----BEGIN PRIVATE KEY----- ...`, + cert: `-----BEGIN CERTIFICATE----- ...`, + originHost: "localhost", // optional + originPort: 8000, + tunnelHost: `mysite.example.com`, + tunnelPort: 15001, }); // Start the client @@ -145,16 +155,16 @@ await client.stop(); ``` ```typescript -import {TunnelServer} from "h2tunnel"; +import { TunnelServer } from "h2tunnel"; const server = new TunnelServer({ - logger: (line) => console.log(line), // optional - key: `-----BEGIN PRIVATE KEY----- ...`, - cert: `-----BEGIN CERTIFICATE----- ...`, - tunnelListenIp: "0.0.0.0", // optional - tunnelListenPort: 15001, - proxyListenIp: "0.0.0.0", // optional - proxyListenPort: 80, + logger: (line) => console.log(line), // optional + key: `-----BEGIN PRIVATE KEY----- ...`, + cert: `-----BEGIN CERTIFICATE----- ...`, + tunnelListenIp: "0.0.0.0", // optional + tunnelListenPort: 15001, + proxyListenIp: "0.0.0.0", // optional + proxyListenPort: 80, }); // Start the server @@ -179,10 +189,16 @@ npm run coverage ## Changelog +### 0.3.0 + +- Support tunneling half-closed TCP connections, these are sometimes killed by middleboxes but they will be safe in h2tunnel +- Remove mux/demux port configuration, instead take a random port assigned by the OS +- Allow specifying the origin host for advanced use cases, default is localhost + ### 0.2.0 - Tunnel TCP instead of HTTP1, supporting a wide range of protocols -- Prevent double TLS encryption by using Node.js unencrypted HTTP2 connection +- Prevent double TLS encryption by using Node.js unencrypted HTTP/2 connection - Lots of testing improvements - Reduce code size to <500 LOC @@ -193,7 +209,7 @@ npm run coverage ### 0.1.0 - Proof of concept -- Supports tunneling HTTP1 over HTTP2 + TLS +- Supports tunneling HTTP1 over HTTP/2 + TLS ## License diff --git a/package.json b/package.json index 979f7ef..8c1ef74 100644 --- a/package.json +++ b/package.json @@ -1,14 +1,26 @@ { "name": "h2tunnel", - "description": "localhost tunnel using HTTP2 multiplexing", + "description": "Zero-dependency remote port forwarding (TCP over HTTP/2)", "version": "0.2.0", "type": "module", "license": "MIT", + "homepage": "https://github.com/boronine/h2tunnel#readme", + "keywords": [ + "tls", + "http2", + "tunnel", + "localhost", + "multiplexing", + "ngrok" + ], "author": { "name": "Alexei Boronine", "email": "alexei@boronine.com", "url": "https://www.boronine.com" }, + "bugs": { + "url": "https://github.com/boronine/h2tunnel/issues" + }, "devDependencies": { "@types/node": "^22.7.5", "typescript": "^5.6.3", diff --git a/tsconfig.json b/tsconfig.json index 05020a1..66555e2 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -18,7 +18,5 @@ "outDir": "build", "strictNullChecks": true }, - "include": [ - "src/*" - ] + "include": ["src/*"] }