.gitignore000066400000000000000000000000371516067375100130610ustar00rootroot00000000000000node_modules npm-debug.log tmp .jshintrc000066400000000000000000000003161516067375100127160ustar00rootroot00000000000000{ "curly": true, "eqeqeq": true, "immed": true, "latedef": true, "newcap": true, "noarg": true, "sub": true, "undef": true, "boss": true, "eqnull": true, "node": true, "es5": true } .travis.yml000066400000000000000000000000431516067375100131770ustar00rootroot00000000000000language: node_js node_js: - 0.8 Gruntfile.js000066400000000000000000000036021516067375100133670ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; module.exports = function(grunt) { // Project configuration. grunt.initConfig({ jshint: { all: [ 'Gruntfile.js', 'tasks/*.js', 'tests/*.js', ], options: { jshintrc: '.jshintrc', }, }, // Before generating any new files, remove any previously-created files. clean: { tests: ['tmp'], }, // Unit tests. nodeunit: { defaults: { src: 'test/defaults_test.js' }, custom_args: { src: 'test/custom_args_test.js' }, custom_port: { src: 'test/custom_port_test.js' }, stoppable: { src: 'test/stoppable_test.js' } }, express: { options: { script: './test/server.js', port: 3000 }, defaults: {}, custom_args: { options: { args: [ 1, 2] } }, custom_port: { options: { port: 8080 } }, stoppable: {} } }); // Actually load this plugin's task(s). grunt.loadTasks('tasks'); // These plugins provide necessary tasks. grunt.loadNpmTasks('grunt-contrib-jshint'); grunt.loadNpmTasks('grunt-contrib-clean'); grunt.loadNpmTasks('grunt-contrib-nodeunit'); // Whenever the "test" task is run, first clean the "tmp" dir, then run this // plugin's task(s), then test the result. grunt.registerTask('test', [ 'clean', 'express:defaults', 'nodeunit:defaults', 'express:custom_args', 'nodeunit:custom_args', 'express:custom_port', 'nodeunit:custom_port', 'express:stoppable', 'express:stoppable:stop', 'nodeunit:stoppable' ]); // By default, lint and run all tests. grunt.registerTask('default', ['jshint', 'test']); }; LICENSE-MIT000066400000000000000000000020411516067375100125220ustar00rootroot00000000000000Copyright (c) 2013 Eric Clemmons Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. README.md000066400000000000000000000071711516067375100123560ustar00rootroot00000000000000# grunt-express-server [![Build Status](https://travis-ci.org/ericclemmons/grunt-express-server.png?branch=master)](https://travis-ci.org/ericclemmons/grunt-express-server) > Simple grunt task for running an Express server that works great with LiveReload + Watch/Regarde ## Getting Started This plugin requires Grunt `~0.4.0` If you haven't used [Grunt](http://gruntjs.com/) before, be sure to check out the [Getting Started](http://gruntjs.com/getting-started) guide, as it explains how to create a [Gruntfile](http://gruntjs.com/sample-gruntfile) as well as install and use Grunt plugins. Once you're familiar with that process, you may install this plugin with this command: ```shell npm install grunt-express-server --save-dev ``` One the plugin has been installed, it may be enabled inside your Gruntfile with this line of JavaScript: ```js grunt.loadNpmTasks('grunt-express-server'); ``` ## The `express` task ### Setup In your project's Gruntfile, you can create one or multiple servers: ```js grunt.initConfig({ express: { options: { // Override defaults here }, dev: { options: { script: 'path/to/dev/server.js' } }, prod: { options: { script: 'path/to/prod/server.js' } } test: { options: { script: 'path/to/test/server.js' } } } }); ``` You can override the default `options` either in the root of the `express` config or within each individual server task. ### Default `options` ```js express: { options: { // Will turn into: `node path/to/server.js ARG1 ARG2 ... ARGN` args: [ ], // Setting to `false` will effectively just run `node path/to/server.js` background: true, // Called if spawning the server fails error: function(err, result, code) {}, // Called when the spawned server throws errors fallback: function() {}, // Override node env's PORT port: 3000 } } ``` ### Usage #### Starting the server If you have a server defined named `dev`, you can start the server by running `expess:dev`. #### Stopping the server Similarly, if you start the `dev` server with `express:dev`, you can stop the server with `express:dev:stop`. #### With [grunt-contrib-livereload](https://github.com/gruntjs/grunt-contrib-livereload) ```js grunt.initConfig({ watch: { express: { files: [ 'path/to/files/to/watch/**.js' ], tasks: [ 'express:dev', 'livereload' ] } } }); grunt.registerTask('server', [ 'express:dev', 'livereload', 'watch' ]) ``` This will let you override `grunt server` with a LiveReload-able Express Server. Finally, you can make changes to your API and watch the JSON change in your browser! #### With [grunt-contrib-watch](https://github.com/gruntjs/grunt-contrib-watch) ```js grunt.initConfig({ watch: { express: { files: [ '**/*.js' ], tasks: [ 'express:dev' ], options: { nospawn: true //Without this option specified express won't be reloaded } } } }); grunt.registerTask('server', [ 'express:dev', 'watch' ]) ``` ## Contributing In lieu of a formal styleguide, take care to maintain the existing coding style. Add unit tests for any new or changed functionality. Lint and test your code using [Grunt](http://gruntjs.com/). ## Release History ### v0.3.0 - `express` is now a multitask with customizable options, better error handling and `:stop` task ### v0.2.0 - Change `express-server` task to `express` - Config is set via `express: '...'` instead of `server: { script: '...' } ` ### v0.1.0 - Initial import from [Genesis Skeleton](https://github.com/ericclemmons/genesis-skeleton) & release package.json000066400000000000000000000021561516067375100133630ustar00rootroot00000000000000{ "name": "grunt-express-server", "description": "Grunt task for running an Express Server that works great with LiveReload + Watch/Regarde", "version": "0.3.0", "homepage": "https://github.com/ericclemmons/grunt-express-server", "author": { "name": "Eric Clemmons", "email": "eric@smarterspam.com" }, "repository": { "type": "git", "url": "git://github.com/ericclemmons/grunt-express-server.git" }, "bugs": { "url": "https://github.com/ericclemmons/grunt-express-server/issues" }, "licenses": [ { "type": "MIT", "url": "https://github.com/ericclemmons/grunt-express-server/blob/master/LICENSE-MIT" } ], "main": "Gruntfile.js", "engines": { "node": ">= 0.8.0" }, "scripts": { "test": "./node_modules/.bin/grunt" }, "devDependencies": { "express": "~3.2.4", "grunt": "~0.4.0", "grunt-cli": "~0.1.6", "grunt-contrib-jshint": "~0.1.1", "grunt-contrib-clean": "~0.4.0", "grunt-contrib-nodeunit": "~0.1.2" }, "peerDependencies": { "grunt": "~0.4.0" }, "keywords": [ "gruntplugin", "express", "server" ] } tasks/000077500000000000000000000000001516067375100122165ustar00rootroot00000000000000tasks/express.js000066400000000000000000000016301516067375100142450ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; var path = require('path'); module.exports = function(grunt) { var server = require('./lib/server')(grunt); grunt.registerMultiTask('express', 'Start an express web server', function() { var options = this.options({ args: [ ], background: true, error: function(err, result, code) { /* Callback has to exist */ }, fallback: function() { /* Prevent EADDRINUSE from breaking Grunt */ }, port: 3000 }); options.script = path.resolve(options.script); options.args.unshift(options.script); if (!grunt.file.exists(options.script)) { grunt.log.error('Could not find server script: ' + options.script); return false; } server.start(options); }); }; tasks/lib/000077500000000000000000000000001516067375100127645ustar00rootroot00000000000000tasks/lib/server.js000066400000000000000000000027021516067375100146310ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; module.exports = function(grunt) { var server = null; // Store server between live reloads to close/restart express return { start: function(options) { if (server) { this.stop(); if (grunt.task.current.flags.stop) { return; } }; grunt.log.writeln('Starting '.cyan + (options.background ? 'background' : 'foreground') + ' Express server'); // Set PORT for new processes process.env.PORT = options.port; if (options.background) { var done = grunt.task.current.async(); server = grunt.util.spawn({ cmd: process.argv[0], args: options.args, fallback: options.fallback }, options.error); server.stdout.on('data', function() { if (done) { done(); } done = null; }); server.stdout.pipe(process.stdout); server.stderr.pipe(process.stderr); } else { // Server is ran in current process server = require(options.script); } process.on('exit', this.stop); }, stop: function() { if (server) { grunt.log.writeln('Stopping'.red + ' Express server'); server.kill('SIGTERM'); server = null; }; } }; }; test/000077500000000000000000000000001516067375100120505ustar00rootroot00000000000000test/app.js000066400000000000000000000006431516067375100131710ustar00rootroot00000000000000var express = require('express'); var app = module.exports = express(); var args = process.argv; app.set('port', process.env.PORT || 3000); app.use(express.static(__dirname + '/../test/fixtures')); // Setup simple echo for each additional argument passed for testing args.slice(2).forEach(function(arg) { console.log(arg); app.get('/' + arg, function(req, res) { res.send('Howdy ' + arg + '!'); }); }); test/custom_args_test.js000066400000000000000000000013701516067375100157740ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; var get = require('./lib/get'); module.exports.custom_args = { 1: function(test) { test.expect(2); get('http://localhost:3000/1', function(res, body) { test.equal(res.statusCode, 200, 'should return 200'); test.equal(body, 'Howdy 1!', 'should return dynamic page'); test.done(); }); }, 2: function(test) { test.expect(2); get('http://localhost:3000/2', function(res, body) { test.equal(res.statusCode, 200, 'should return 200'); test.equal(body, 'Howdy 2!', 'should return dynamic page'); test.done(); }); } }; test/custom_port_test.js000066400000000000000000000010001516067375100160120ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; var get = require('./lib/get'); module.exports.custom_port = { hello: function(test) { test.expect(2); get('http://localhost:8080/hello.txt', function(res, body) { test.equal(res.statusCode, 200, 'should return 200'); test.equal(body, 'Howdy!\n', 'should return static page'); test.done(); }); } }; test/defaults_test.js000066400000000000000000000007751516067375100152650ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; var get = require('./lib/get'); module.exports.defaults = { hello: function(test) { test.expect(2); get('http://localhost:3000/hello.txt', function(res, body) { test.equal(res.statusCode, 200, 'should return 200'); test.equal(body, 'Howdy!\n', 'should return static page'); test.done(); }); } }; test/fixtures/000077500000000000000000000000001516067375100137215ustar00rootroot00000000000000test/fixtures/hello.txt000066400000000000000000000000071516067375100155620ustar00rootroot00000000000000Howdy! test/lib/000077500000000000000000000000001516067375100126165ustar00rootroot00000000000000test/lib/get.js000066400000000000000000000010031516067375100137250ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; var http = require('http'); module.exports = function get(url, callback, error) { var req = http.get(url, function(res) { var body = ''; res.on('data', function(chunk) { body += chunk; }).on('end', function() { callback(res, body); }); }); if (error) { req.on('error', error) }; req.end(); } test/server.js000066400000000000000000000002711516067375100137140ustar00rootroot00000000000000/** * Test Server */ var app = require('./app'); module.exports = app.listen(app.get('port'), function() { console.log("Express server listening on port " + app.get('port')); }); test/stoppable_test.js000066400000000000000000000007651516067375100154460ustar00rootroot00000000000000/* * grunt-express-server * https://github.com/ericclemmons/grunt-express-server * * Copyright (c) 2013 Eric Clemmons * Licensed under the MIT license. */ 'use strict'; var get = require('./lib/get'); module.exports.stoppable = { hello: function(test) { test.expect(1); get('http://localhost:3000/hello.txt', function(res, body) { test.done(); }, function(err) { test.equal('ECONNREFUSED', err.code, 'should return ECONNREFUSED'); test.done(); }); } };