_ = require 'underscore' request = require 'superagent' expect = require 'expect.js' require 'coffee-script' conf = require("nconf") conf.argv().env().file({file: __dirname + '../config.json'}).defaults # Override normal ENV values with nconf ENV values (ENV values are used the same way without nconf) #FIXME can't get nconf file above to load... process.env.BASE_URL = conf.get("BASE_URL") process.env.FACEBOOK_KEY = conf.get("FACEBOOK_KEY") process.env.FACEBOOK_SECRET = conf.get("FACEBOOK_SECRET") process.env.NODE_DB_URI = 'mongodb://localhost/habirpg' ## monkey-patch expect.js for better diffs on mocha ## see: https://github.com/LearnBoost/expect.js/pull/34 origBe = expect.Assertion::be expect.Assertion::be = expect.Assertion::equal = (obj) -> @_expected = obj origBe.call this, obj # Custom modules character = require '../src/app/character' ###### Helpers & Variables ###### model = null uuid = null taskPath = null baseURL = 'http://localhost:1337/api/v1' ###### Specs ###### describe 'API', -> server = null store = null model = null user = null uid = null before (done) -> server = require '../src/server' server.listen '1337', '0.0.0.0' server.on 'listening', (data) -> store = server.habitStore #store.flush() model = store.createModel() model.set '_userId', uid = model.id() user = character.newUserObject() user.apiToken = model.id() model.session = {userId:uid} model.set "users.#{uid}", user delete model.session # Crappy hack to let server start before tests run setTimeout done, 2000 describe 'Without token or user id', -> it '/api/v1/status', (done) -> request.get("#{baseURL}/status") .set('Accept', 'application/json') .end (res) -> expect(res.statusCode).to.be 200 expect(res.body.status).to.be 'up' done() it '/api/v1/user', (done) -> request.get("#{baseURL}/user") .set('Accept', 'application/json') .end (res) -> expect(res.statusCode).to.be 401 expect(res.body.err).to.be 'You must include a token and uid (user id) in your request' done() describe 'With token and user id', -> params = null currentUser = null before -> user = model.at("users.#{uid}") currentUser = user.get() params = title: 'Title' text: 'Text' type: 'habit' beforeEach -> currentUser = user.get() it 'GET /api/v1/user', (done) -> request.get("#{baseURL}/user") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .end (res) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 200 expect(res.body.id).not.to.be.empty() self = _.clone(currentUser) delete self.apiToken self.stats.toNextLevel = 150 self.stats.maxHealth = 50 expect(res.body).to.eql self done() it 'GET /api/v1/user/task/:id', (done) -> tid = _.pluck(currentUser.tasks, 'id')[0] request.get("#{baseURL}/user/task/#{tid}") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .end (res) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 200 expect(res.body).to.eql currentUser.tasks[tid] done() it 'POST /api/v1/user/task', (done) -> request.post("#{baseURL}/user/task") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .send(params) .end (res) -> query = model.query('users').withIdAndToken(currentUser.id, currentUser.apiToken) query.fetch (err, user) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 201 expect(res.body.id).not.to.be.empty() # Ensure that user owns the newly created object expect(user.get().tasks[res.body.id]).to.be.an('object') done() it 'POST /api/v1/user/task (without type)', (done) -> request.post("#{baseURL}/user/task") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .send({}) .end (res) -> expect(res.body.err).to.be 'type must be habit, todo, daily, or reward' expect(res.statusCode).to.be 400 done() it 'POST /api/v1/user/task (only type)', (done) -> request.post("#{baseURL}/user/task") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .send(type: 'habit') .end (res) -> query = model.query('users').withIdAndToken(currentUser.id, currentUser.apiToken) query.fetch (err, user) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 201 expect(res.body.id).not.to.be.empty() # Ensure that user owns the newly created object expect(user.get().tasks[res.body.id]).to.be.an('object') done() it 'PUT /api/v1/user/task/:id', (done) -> tid = _.pluck(currentUser.tasks, 'id')[0] request.put("#{baseURL}/user/task/#{tid}") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .send(text: 'bye') .end (res) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 200 currentUser.tasks[tid].text = 'bye' expect(res.body).to.eql currentUser.tasks[tid] done() it 'PUT /api/v1/user/task/:id (shouldnt update type)', (done) -> tid = _.pluck(currentUser.tasks, 'id')[1] type = if currentUser.tasks[tid].type is 'habit' then 'daily' else 'habit' request.put("#{baseURL}/user/task/#{tid}") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .send(type: type, text: 'fishman') .end (res) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 200 currentUser.tasks[tid].text = 'fishman' expect(res.body).to.eql currentUser.tasks[tid] done() it 'PUT /api/v1/user/task/:id (update notes)', (done) -> tid = _.pluck(currentUser.tasks, 'id')[2] request.put("#{baseURL}/user/task/#{tid}") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .send(text: 'hi',notes:'foobar matey') .end (res) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 200 currentUser.tasks[tid].text = 'hi' currentUser.tasks[tid].notes = 'foobar matey' expect(res.body).to.eql currentUser.tasks[tid] done() it 'GET /api/v1/user/tasks', (done) -> request.get("#{baseURL}/user/tasks") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .end (res) -> query = model.query('users').withIdAndToken(currentUser.id, currentUser.apiToken) query.fetch (err, user) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 200 model.ref '_user', user tasks = [] for type in ['habit','todo','daily','reward'] model.refList "_#{type}List", "_user.tasks", "_user.#{type}Ids" tasks = tasks.concat model.get("_#{type}List") # Ensure that user owns the tasks expect(res.body.length).to.equal tasks.length # Ensure that the two sets are equal expect(_.difference(_.pluck(res.body,'id'), _.pluck(tasks,'id')).length).to.equal 0 done() it 'GET /api/v1/user/tasks (todos)', (done) -> request.get("#{baseURL}/user/tasks") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .query(type:'todo') .end (res) -> query = model.query('users').withIdAndToken(currentUser.id, currentUser.apiToken) query.fetch (err, user) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 200 model.ref '_user', user model.refList "_todoList", "_user.tasks", "_user.todoIds" tasks = model.get("_todoList") # Ensure that user owns the tasks expect(res.body.length).to.equal tasks.length # Ensure that the two sets are equal expect(_.difference(_.pluck(res.body,'id'), _.pluck(tasks,'id')).length).to.equal 0 done() it 'DELETE /api/v1/user/task/:id', (done) -> tid = currentUser.habitIds[2] request.del("#{baseURL}/user/task/#{tid}") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .end (res) -> expect(res.body.err).to.be undefined expect(res.statusCode).to.be 204 query = model.query('users').withIdAndToken(currentUser.id, currentUser.apiToken) query.fetch (err, user) -> expect(user.get('habitIds').indexOf(tid)).to.be -1 expect(user.get("tasks.#{tid}")).to.be undefined done() it 'DELETE /api/v1/user/task/:id (no task found)', (done) -> tid = "adsfasdfjunkshouldntbeatask" request.del("#{baseURL}/user/task/#{tid}") .set('Accept', 'application/json') .set('X-API-User', currentUser.id) .set('X-API-Key', currentUser.apiToken) .end (res) -> expect(res.statusCode).to.be 400 expect(res.body.err).to.be 'No task found.' done()