braid-http
v1.3.106
Published
An implementation of Braid-HTTP for Node.js and Browsers
Readme
Braid-HTTP
This ponyfill library extends the HTTP implementations of Browsers and Nodejs with Braid-HTTP; transforming them from state transfer to state synchronization systems.
These features are provided in an elegant, backwards-compatible way:
- Browsers: get a drop-in replacement for
fetch() - Nodejs: get a route handler that adds abilities to the
http,https, andhttp2modules
It conforms to the Braid-HTTP v04 specification, with the additional HTTP Multiresponse and Multiplexing v1.0 extensions.
Developed in braid.org.
Installing
Browsers:
<script src="https://unpkg.com/braid-http/braid-http-client.js"></script>
<script>
// To live on the cutting edge, you can now replace the browser's fetch() if desired:
// window.fetch = braid_fetch
</script>Node.js:
npm install braid-http// Import with require()
require('braid-http').fetch // A polyfill for fetch
require('braid-http').http_client // A polyfill for require('http') clients
require('braid-http').http_server // A polyfill for require('http') servers
// Or as es6 module
import {fetch, http_client, http_server} from 'braid-http'Using it in Browsers
This library adds a {subscribe: true} option to fetch(), and lets you
access the result of a subscription with these new fields on the fetch response:
response.subscribe( update => ... )response.subscription: an iterator that can be used withfor awaitresponse.version: the parsed version from the response headers (if present)
Example Subscription with Promises
Here is an example of subscribing to a Braid resource using promises:
fetch('https://braid.org/chat', {subscribe: true}).then(
res => res.subscribe(
(update) => {
console.log('We got a new update!', update)
// {
// version: ["me"],
// parents: ["mom", "dad"],
// patches: [{
//. unit: "json",
// range: ".foo",
// content: new Uint8Array([51]),
// content_text: "3" <-- getter
//. }],
// body: new Uint8Array([51]),
// body_text: "3" <-- getter
// }
//
// Note that `update` will contain either patches *or* body
}
)
)If you want automatic reconnections, this library add a {retry: true} option to fetch().
fetch('https://braid.org/chat', {subscribe: true, retry: true}).then(
res => res.subscribe(
(update) => {
console.log('We got a new update!', update)
// Do something with the update
}
)
)For use in conjunction with {retry: true}, it's possible to make the parents param equal to a function, which will be called to get the current parents each time the fetch establishes a new connection.
fetch('https://braid.org/chat', {subscribe: true, retry: true, parents: () => {
return current_parents
}}).then(
res => res.subscribe(
(update) => {
console.log('We got a new update!', update)
// Do something with the update
}
)
)You can monitor the subscription's connection status with onSubscriptionStatus:
fetch('https://braid.org/chat', {
subscribe: true,
retry: true,
onSubscriptionStatus: ({online, error, status, statusText}) => {
if (online)
console.log('Connected!')
else
console.log('Disconnected:', error)
}
}).then(
res => res.subscribe(
(update) => { console.log('Got update!', update) }
)
)The callback receives an object with only the fields relevant to the event:
{online: true}— the subscription is connected{online: false, error}— the subscription went offline, with the error/reason for disconnection
Example Subscription with Async/Await
(await fetch('/chat', {subscribe: true, retry: true})).subscribe(
(update) => {
// We got a new update!
})Example Subscription with for await
var subscription_iterator = (await fetch('/chat',
{subscribe: true, retry: true})).subscription
for await (var update of subscription_iterator) {
// Updates might come in the form of patches:
if (update.patches)
chat = apply_patches(update.patches, chat)
// Or complete snapshots:
else
// Beware the server doesn't send these yet.
chat = JSON.parse(update.body_text)
render_stuff()
}Using it in Nodejs
You can braidify your nodejs server with:
var braidify = require('braid-http').http_serverBraidify adds these new abilities to requests and responses:
req.subscribereq.startSubscription({onClose: cb})await req.parseUpdate()res.sendUpdate()
You can call it in two ways:
braidify((req, res) => ...)wraps your HTTP request handler, and gives it perfectly braidified requests and responses.braidify(req, res, next)will add arguments to your existing requests and responses. You can use this as express middleware.
Example Nodejs server with the built-in HTTP module
var braidify = require('braid-http').http_server
// or:
import {http_server as braidify} from 'braid-http'
require('http').createServer(
braidify((req, res) => {
// Now braid stuff is available on req and res
// So you can easily handle subscriptions
if (req.subscribe)
res.startSubscription({ onClose: _=> null })
// startSubscription automatically sets statusCode = 209
else
res.statusCode = 200
// And send updates over a subscription
res.sendUpdate({
version: ['greg'],
body: JSON.stringify({greg: 'greg'})
})
})
).listen(9935)If you are working from a library, or from code that does not have access to
the root of the HTTP handler or next in (req, res, next), you can also
call braidify inline:
require('http').createServer(
(req, res) => {
braidify(req, res); if (req.is_multiplexer) return
// Now braid stuff is available on req and res
// ...
})
).listen(9935)This works, but the inline form leaks the multiplexing abstraction in three minor ways.
Example Nodejs server with Express
Or if you're using express, you can just call app.use(braidify) to get
braid features added to every request and response.
var braidify = require('braid-http').http_server
// or:
import {http_server as braidify} from 'braid-http'
var app = require('express')()
app.use(braidify) // Add braid stuff to req and res
app.get('/', (req, res) => {
// Now use it
if (req.subscribe)
res.startSubscription({ onClose: _=> null })
// startSubscription automatically sets statusCode = 209
else
res.statusCode = 200
// Send the current version
res.sendUpdate({
version: ['greg'],
parents: ['gr','eg'],
body: JSON.stringify({greg: 'greg'})
})
// Or you can send patches like this:
// res.sendUpdate({
// version: ['greg'],
// parents: ['gr','eg'],
// patches: [{range: '.greg', unit: 'json', content: '"greg"'}]
// })
})
require('http').createServer(app).listen(8583)Example Nodejs client with require('http')
// Use this line if necessary for self-signed certs
// process.env["NODE_TLS_REJECT_UNAUTHORIZED"] = 0
var https = require('braid-http').http_client(require('https'))
// or:
// import braid_http from 'braid-http'
// https = braid_http.http_client(require('https'))
https.get(
'https://braid.org/chat',
{subscribe: true},
(res) => {
res.on('update', (update) => {
console.log('well we got one', update)
})
}
)To get auto-reconnections use:
function connect () {
https.get(
'https://braid.org/chat',
{subscribe: true},
(res) => {
res.on('update', (update) => {
// {
// version: ["me"],
// parents: ["mom", "dad"],
// patches: [{
//. unit: "json",
// range: ".foo",
// content: new Uint8Array([51]),
// content_text: "3" <-- getter
//. }],
// body: new Uint8Array([51]),
// body_text: "3" <-- getter
// }
// Update will contain either patches *or* body, but not both
console.log('We got a new update!', update)
})
res.on('end', e => setTimeout(connect, 1000))
res.on('error', e => setTimeout(connect, 1000))
})
}
connect()Example Nodejs client with fetch()
var fetch = require('braid-http').fetch
// or:
import {fetch} from 'braid-http'
// process.env["NODE_TLS_REJECT_UNAUTHORIZED"] = 0
fetch('https://localhost:3009/chat',
{subscribe: true}).andThen(
x => console.log('Got ', x)
)Testing
Run all tests from the command line:
npm testRun tests in a browser (auto-opens):
npm run test:browserYou can also filter tests by name:
node test/test.js --filter="version"Multiplexing
This library automatically
multiplexes subscriptions behind
the scenes to overcome web browsers' 6-connection limit (with HTTP/1) and
100-connection limit (with HTTP/2). When you setup a server's braidify in
the recommended ways, you don't need to know it's happening — the abstraction
is completely transparent.
// Recommendation #1: Wrapping the entire request handler
require('http').createServer(
braidify((req, res) => {
...
})
)// Recommendation #2: As middleware
var app = require('express')()
app.use(braidify)// Recommendation #3: With braidify(req, res, next)
// (Equivalent to the middleware form.)
app.use(
(req, res, next) => {
...
braidify(req, res, next)
...
}
)Inline braidify(req, res) leaks the abstraction
If you are using braidify from within a library, or in another context without
access to the entire request handler, or a next() method, then you can use
the inline braidify(req, res) form:
require('http').createServer(
(req, res) => {
...
braidify(req, res); if (req.is_multiplexer) return
...
}
)Just know that there are three abstraction leaks when using this form:
- You must add
if (req.is_multiplexer) returnafter thebraidify(req, res)call. - The library will disable the buffering
optimization
on optimistic multiplexer creation. This buffering prevents two minor
inconveniences that occur on about ~15% of page loads:
- One round trip of additional latency on the first subscription to a host
- A harmless
424error in the javascript console, which can be safely ignored:
The buffering works like this: when the client connects to a new host, it
sends a POST to create the multiplexer and GETs to subscribe — all in
parallel. Sometimes a GET arrives before the POST. With the recommended
forms, the server briefly buffers the GET (70ms, event-driven) until the
POST lands, then processes it normally. Without next, the server can't
re-run the handler, so it returns 424 immediately and the client retries.
Configuring multiplexing
You can tune multiplexing on the client, per-request or globally:
braid_fetch('/a', {multiplex: true}) // force on for this request
braid_fetch('/a', {multiplex: false}) // force off for this request
braid_fetch.enable_multiplex = true // on for all GETs
braid_fetch.enable_multiplex = false // off globally
braid_fetch.enable_multiplex = {after: 1} // on after N connections (default)And on the server:
braidify.enable_multiplex = true // default; set false to disable
braidify.multiplex_wait = 10 // ms; timeout for the buffering optimization (default 10)
// set to 0 to disable buffering