diff --git a/config-angular/testacular-e2e.conf.js b/config-angular/testacular-e2e.conf.js new file mode 100644 index 0000000000..51f51d2ee3 --- /dev/null +++ b/config-angular/testacular-e2e.conf.js @@ -0,0 +1,22 @@ +basePath = '../'; + +files = [ + ANGULAR_SCENARIO, + ANGULAR_SCENARIO_ADAPTER, + 'test/e2e/**/*.js' +]; + +autoWatch = false; + +browsers = ['Chrome']; + +singleRun = true; + +proxies = { + '/': 'http://localhost:8000/' +}; + +junitReporter = { + outputFile: 'test_out/e2e.xml', + suite: 'e2e' +}; diff --git a/config-angular/testacular.conf.js b/config-angular/testacular.conf.js new file mode 100644 index 0000000000..92b8ef0edb --- /dev/null +++ b/config-angular/testacular.conf.js @@ -0,0 +1,24 @@ +basePath = '../'; + +files = [ + JASMINE, + JASMINE_ADAPTER, + 'app/assets/javascripts/shared/angular.js', + 'app/assets/javascripts/shared/angular-*.js', + //'test/lib/angular/angular-mocks.js', + + 'app/assets/javascripts/admin/order_cycle.js.erb', + + 'spec/javascripts/unit/**/*.js*' +]; + +exclude = ['**/.#*'] + +autoWatch = true; + +browsers = ['Chrome']; + +junitReporter = { + outputFile: 'log/testacular-unit.xml', + suite: 'unit' +}; diff --git a/script-angular/e2e-test.bat b/script-angular/e2e-test.bat new file mode 100644 index 0000000000..c89a708a14 --- /dev/null +++ b/script-angular/e2e-test.bat @@ -0,0 +1,11 @@ +@echo off + +REM Windows script for running e2e tests +REM You have to run server and capture some browser first +REM +REM Requirements: +REM - NodeJS (http://nodejs.org/) +REM - Testacular (npm install -g testacular) + +set BASE_DIR=%~dp0 +testacular start "%BASE_DIR%\..\config\testacular-e2e.conf.js" %* diff --git a/script-angular/e2e-test.sh b/script-angular/e2e-test.sh new file mode 100755 index 0000000000..e887c34f41 --- /dev/null +++ b/script-angular/e2e-test.sh @@ -0,0 +1,9 @@ +#!/bin/bash + +BASE_DIR=`dirname $0` + +echo "" +echo "Starting Testacular Server (http://vojtajina.github.com/testacular)" +echo "-------------------------------------------------------------------" + +testacular start $BASE_DIR/../config/testacular-e2e.conf.js $* diff --git a/script-angular/test-server.bat b/script-angular/test-server.bat new file mode 100644 index 0000000000..0b7fa66ebf --- /dev/null +++ b/script-angular/test-server.bat @@ -0,0 +1,19 @@ +@echo off + +REM Windows script for starting JSTD server +REM +REM Requirements: +REM - Java (http://www.java.com) + +set BASE_DIR=%~dp0 +set PORT=9876 + +echo Starting JsTestDriver Server (http://code.google.com/p/js-test-driver/) +echo Please open the following url and capture one or more browsers: +echo http://localhost:%PORT%/ + +java -jar "%BASE_DIR%\..\test\lib\jstestdriver\JsTestDriver.jar" ^ + --port %PORT% ^ + --browserTimeout 20000 ^ + --config "%BASE_DIR%\..\config\jsTestDriver.conf" ^ + --basePath "%BASE_DIR%\.." diff --git a/script-angular/test-server.sh b/script-angular/test-server.sh new file mode 100755 index 0000000000..ff38f5ee79 --- /dev/null +++ b/script-angular/test-server.sh @@ -0,0 +1,14 @@ +#!/bin/bash + +BASE_DIR=`dirname $0` +PORT=9876 + +echo "Starting JsTestDriver Server (http://code.google.com/p/js-test-driver/)" +echo "Please open the following url and capture one or more browsers:" +echo "http://localhost:$PORT" + +java -jar "$BASE_DIR/../test/lib/jstestdriver/JsTestDriver.jar" \ + --port $PORT \ + --browserTimeout 20000 \ + --config "$BASE_DIR/../config/jsTestDriver.conf" \ + --basePath "$BASE_DIR/.." diff --git a/script-angular/test.bat b/script-angular/test.bat new file mode 100644 index 0000000000..000242f53e --- /dev/null +++ b/script-angular/test.bat @@ -0,0 +1,11 @@ +@echo off + +REM Windows script for running unit tests +REM You have to run server and capture some browser first +REM +REM Requirements: +REM - NodeJS (http://nodejs.org/) +REM - Testacular (npm install -g testacular) + +set BASE_DIR=%~dp0 +testacular start "%BASE_DIR%\..\config\testacular.conf.js" %* diff --git a/script-angular/test.sh b/script-angular/test.sh new file mode 100755 index 0000000000..09b1072791 --- /dev/null +++ b/script-angular/test.sh @@ -0,0 +1,9 @@ +#!/bin/bash + +BASE_DIR=`dirname $0` + +echo "" +echo "Starting Testacular Server (http://vojtajina.github.com/testacular)" +echo "-------------------------------------------------------------------" + +testacular start $BASE_DIR/../config-angular/testacular.conf.js $* diff --git a/script-angular/update-repo.sh b/script-angular/update-repo.sh new file mode 100755 index 0000000000..5a85bfd116 --- /dev/null +++ b/script-angular/update-repo.sh @@ -0,0 +1,13 @@ +#!/bin/bash + +### +# This scripts updates the local repo with the latest changes from github. +# +# The master branch will be REPLACED with what's in github and all local changes +# will be LOST. +### + +git checkout master +git fetch -f origin +git fetch --tags origin +git reset --hard origin/master diff --git a/script-angular/watchr.rb b/script-angular/watchr.rb new file mode 100755 index 0000000000..89ef656d27 --- /dev/null +++ b/script-angular/watchr.rb @@ -0,0 +1,19 @@ +#!/usr/bin/env watchr + +# config file for watchr http://github.com/mynyml/watchr +# install: gem install watchr +# run: watch watchr.rb +# note: make sure that you have jstd server running (server.sh) and a browser captured + +log_file = File.expand_path(File.dirname(__FILE__) + '/../logs/jstd.log') + +`cd ..` +`touch #{log_file}` + +puts "String watchr... log file: #{log_file}" + +watch( '(app/js|test/unit)' ) do + `echo "\n\ntest run started @ \`date\`" > #{log_file}` + `scripts/test.sh &> #{log_file}` +end + diff --git a/script-angular/web-server.js b/script-angular/web-server.js new file mode 100755 index 0000000000..3f74441e31 --- /dev/null +++ b/script-angular/web-server.js @@ -0,0 +1,244 @@ +#!/usr/bin/env node + +var util = require('util'), + http = require('http'), + fs = require('fs'), + url = require('url'), + events = require('events'); + +var DEFAULT_PORT = 8000; + +function main(argv) { + new HttpServer({ + 'GET': createServlet(StaticServlet), + 'HEAD': createServlet(StaticServlet) + }).start(Number(argv[2]) || DEFAULT_PORT); +} + +function escapeHtml(value) { + return value.toString(). + replace('<', '<'). + replace('>', '>'). + replace('"', '"'); +} + +function createServlet(Class) { + var servlet = new Class(); + return servlet.handleRequest.bind(servlet); +} + +/** + * An Http server implementation that uses a map of methods to decide + * action routing. + * + * @param {Object} Map of method => Handler function + */ +function HttpServer(handlers) { + this.handlers = handlers; + this.server = http.createServer(this.handleRequest_.bind(this)); +} + +HttpServer.prototype.start = function(port) { + this.port = port; + this.server.listen(port); + util.puts('Http Server running at http://localhost:' + port + '/'); +}; + +HttpServer.prototype.parseUrl_ = function(urlString) { + var parsed = url.parse(urlString); + parsed.pathname = url.resolve('/', parsed.pathname); + return url.parse(url.format(parsed), true); +}; + +HttpServer.prototype.handleRequest_ = function(req, res) { + var logEntry = req.method + ' ' + req.url; + if (req.headers['user-agent']) { + logEntry += ' ' + req.headers['user-agent']; + } + util.puts(logEntry); + req.url = this.parseUrl_(req.url); + var handler = this.handlers[req.method]; + if (!handler) { + res.writeHead(501); + res.end(); + } else { + handler.call(this, req, res); + } +}; + +/** + * Handles static content. + */ +function StaticServlet() {} + +StaticServlet.MimeMap = { + 'txt': 'text/plain', + 'html': 'text/html', + 'css': 'text/css', + 'xml': 'application/xml', + 'json': 'application/json', + 'js': 'application/javascript', + 'jpg': 'image/jpeg', + 'jpeg': 'image/jpeg', + 'gif': 'image/gif', + 'png': 'image/png', +  'svg': 'image/svg+xml' +}; + +StaticServlet.prototype.handleRequest = function(req, res) { + var self = this; + var path = ('./' + req.url.pathname).replace('//','/').replace(/%(..)/g, function(match, hex){ + return String.fromCharCode(parseInt(hex, 16)); + }); + var parts = path.split('/'); + if (parts[parts.length-1].charAt(0) === '.') + return self.sendForbidden_(req, res, path); + fs.stat(path, function(err, stat) { + if (err) + return self.sendMissing_(req, res, path); + if (stat.isDirectory()) + return self.sendDirectory_(req, res, path); + return self.sendFile_(req, res, path); + }); +} + +StaticServlet.prototype.sendError_ = function(req, res, error) { + res.writeHead(500, { + 'Content-Type': 'text/html' + }); + res.write('\n'); + res.write('Internal Server Error\n'); + res.write('

Internal Server Error

'); + res.write('
' + escapeHtml(util.inspect(error)) + '
'); + util.puts('500 Internal Server Error'); + util.puts(util.inspect(error)); +}; + +StaticServlet.prototype.sendMissing_ = function(req, res, path) { + path = path.substring(1); + res.writeHead(404, { + 'Content-Type': 'text/html' + }); + res.write('\n'); + res.write('404 Not Found\n'); + res.write('

Not Found

'); + res.write( + '

The requested URL ' + + escapeHtml(path) + + ' was not found on this server.

' + ); + res.end(); + util.puts('404 Not Found: ' + path); +}; + +StaticServlet.prototype.sendForbidden_ = function(req, res, path) { + path = path.substring(1); + res.writeHead(403, { + 'Content-Type': 'text/html' + }); + res.write('\n'); + res.write('403 Forbidden\n'); + res.write('

Forbidden

'); + res.write( + '

You do not have permission to access ' + + escapeHtml(path) + ' on this server.

' + ); + res.end(); + util.puts('403 Forbidden: ' + path); +}; + +StaticServlet.prototype.sendRedirect_ = function(req, res, redirectUrl) { + res.writeHead(301, { + 'Content-Type': 'text/html', + 'Location': redirectUrl + }); + res.write('\n'); + res.write('301 Moved Permanently\n'); + res.write('

Moved Permanently

'); + res.write( + '

The document has moved here.

' + ); + res.end(); + util.puts('301 Moved Permanently: ' + redirectUrl); +}; + +StaticServlet.prototype.sendFile_ = function(req, res, path) { + var self = this; + var file = fs.createReadStream(path); + res.writeHead(200, { + 'Content-Type': StaticServlet. + MimeMap[path.split('.').pop()] || 'text/plain' + }); + if (req.method === 'HEAD') { + res.end(); + } else { + file.on('data', res.write.bind(res)); + file.on('close', function() { + res.end(); + }); + file.on('error', function(error) { + self.sendError_(req, res, error); + }); + } +}; + +StaticServlet.prototype.sendDirectory_ = function(req, res, path) { + var self = this; + if (path.match(/[^\/]$/)) { + req.url.pathname += '/'; + var redirectUrl = url.format(url.parse(url.format(req.url))); + return self.sendRedirect_(req, res, redirectUrl); + } + fs.readdir(path, function(err, files) { + if (err) + return self.sendError_(req, res, error); + + if (!files.length) + return self.writeDirectoryIndex_(req, res, path, []); + + var remaining = files.length; + files.forEach(function(fileName, index) { + fs.stat(path + '/' + fileName, function(err, stat) { + if (err) + return self.sendError_(req, res, err); + if (stat.isDirectory()) { + files[index] = fileName + '/'; + } + if (!(--remaining)) + return self.writeDirectoryIndex_(req, res, path, files); + }); + }); + }); +}; + +StaticServlet.prototype.writeDirectoryIndex_ = function(req, res, path, files) { + path = path.substring(1); + res.writeHead(200, { + 'Content-Type': 'text/html' + }); + if (req.method === 'HEAD') { + res.end(); + return; + } + res.write('\n'); + res.write('' + escapeHtml(path) + '\n'); + res.write('\n'); + res.write('

Directory: ' + escapeHtml(path) + '

'); + res.write('
    '); + files.forEach(function(fileName) { + if (fileName.charAt(0) !== '.') { + res.write('
  1. ' + + escapeHtml(fileName) + '
  2. '); + } + }); + res.write('
'); + res.end(); +}; + +// Must be last, +main(process.argv); diff --git a/spec/javascripts/unit/order_cycle_spec.js.coffee b/spec/javascripts/unit/order_cycle_spec.js.coffee new file mode 100644 index 0000000000..45f9156f47 --- /dev/null +++ b/spec/javascripts/unit/order_cycle_spec.js.coffee @@ -0,0 +1,45 @@ +describe 'OrderCycle controllers', -> + + describe 'AdminCreateOrderCycleCtrl', -> + ctrl = null + scope = null + OrderCycle = null + Enterprise = null + + beforeEach -> + scope = {} + OrderCycle = + order_cycle: 'my order cycle' + toggleProducts: jasmine.createSpy('toggleProducts') + addSupplier: jasmine.createSpy('addSupplier') + create: jasmine.createSpy('create') + Enterprise = + index: jasmine.createSpy('index') + ctrl = new AdminCreateOrderCycleCtrl(scope, OrderCycle, Enterprise) + + it 'Loads enterprises', -> + expect(Enterprise.index).toHaveBeenCalled() + + it 'Loads order cycles', -> + expect(scope.order_cycle).toEqual('my order cycle') + + it 'Delegates toggleProducts to OrderCycle', -> + scope.toggleProducts('event', 'exchange') + expect(OrderCycle.toggleProducts).toHaveBeenCalledWith('event', 'exchange') + + it 'Adds order cycle suppliers', -> + scope.new_supplier_id = 'new supplier id' + scope.addSupplier('event') + expect(OrderCycle.addSupplier).toHaveBeenCalledWith('event', 'new supplier id') + + it 'Submits the order cycle via OrderCycle create', -> + scope.submit() + expect(OrderCycle.create).toHaveBeenCalled() + + describe 'AdminEditOrderCycleCtrl', -> + + +describe 'OrderCycle services', -> + + +describe 'OrderCycle directives', ->