Skip to content

bengl/whee

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

17 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

WHEE!!

Whee is a very simple framework/router/thing for node that emphasizes magic. Things just work, even though they probably shouldn't. Some batteries included.

WARNING: This is more of a whimsical little experiment than a production-ready web framework. Seriously, I made most of this on a plane on very little sleep. If you want something more battle-tested and realistic (and you should), check out express, restify or hapi.

Installation

# npm i --save whee

Usage

First of all, require it:

var w = require('whee')

And now, here's a Hello World:

w()
.get('/', function(){
  w.send('Hello, World!')
})
.listen(3000);

Go ahead and try it! It's that simple! Whee!

You can also do all your other favorite HTTP verbs, like post, put and delete.

The w object is also your this object, so you can call it that way, which is sometimes handy. Especially for CoffeeScript people.

// JavaScript

w()
.get('/', function() {
  this.send('Hello, World!')
})
.listen(3000)
# CoffeeScript

w()
.get '/', -> @send 'Hello, World!'
.listen 3000

You have access to send, sendJson, sendHtml and sendError on the w object, which are all pass-throughs to send-data.

w()
.get('/', function(){
  w.sendHtml('<b>Hello!</b>')
})
.listen(3000)

You have access to textBody', jsonBody, formBodyandanyBodyon thewobject, which are all pass-throughs to [body](https://www.npmjs.org/package/body). Instead of being passed in to the callback, the parsed body is then attached atw.body`.

w()
.post('/jsonEcho', function(){
  w.jsonBody(function(err){
    w.sendJson(w.body)
  })
})
.post('/jsonEchoAsync', async function(){
  await w.jsonBody()
  w.sendJson(w.body)
})

Heresy? Yup, probably. Too bad.

There's also some basic redirect and static file stuff built in, but I'm a lazy README author so you can find those in the tests.

Magic Revealed

This magic comes from hiding the necessary bits (like your req and res objects that you're used to) away in AsyncLocalStorage. Go read up on it. You'll love it! Also, it explains this stuff way better than I do.

Basically it lets you keep a context around for as long as your asynchronous causation chain lasts. For our purposes, that means the length of an HTTP request and response cycle. This is great news because we can hide stuff away in there!

Don't worry, req and res aren't gone forever. There's accessible at w.req and w.res, so long as you're inside a handler.

If you want to hide your own stuff in there, you've got a few options:

The Direct Way

Use Whee's AsyncLocalStorage instance directly:

var w = require('whee');

// somewhere in a request handler ...
w.storage.getStore().mysessionthing = {some: 'thing'});

// somewhere in some other code triggered by the handler

var session = w.storage.getStore().mysessionthing;

The Magic Way

This sets up accessors on the w object:

var w = require('whee');

// somewhere in a request handler ...
w.setMagicValue('mysessionthing', {some: 'thing'});

// ....

var session = w.mysessionthing;

This is how the builtins are added!

The Easy Way

A context object is provided for you on w so you can just put stuff there:

// somewhere in a request handler ...
w.context.mysessionthing = {some: 'thing'};

// ...

var session = w.context.mysessionthing;

With each of these, remember that this stuff carries over to your next continuation, and so on, until you're done dealing with that request.

Compatibility

What about your connect/express middleware? Well we wouldn't want to leave that giant mountain of code behind. Luckily there's a pretty cool function called wrap that we can use. It takes a function that would ordinarily take in a request and response object, and returns a function that has those arguments already prepended via whee's magic.

var w = require('whee');
var morgan = w.wrap(require('morgan'));
// morgan is a pretty good logging middleware

w()
.get('/', function(){
  morgan(function(){
    w.send('Hello, World!');
  });
});

Yes, for now, you'll have to call your middlewares directly. But being deliberate like this is half the fun right? Wheeeeeee!!!

LICENSE

See LICENSE file.

About

A silly little web framework thing, powered by CLS.

Resources

License

Stars

Watchers

Forks

Packages

No packages published