require '../common' Campaign = require '../../../server/models/Campaign' Classroom = require '../../../server/models/Classroom' Course = require '../../../server/models/Course' CourseInstance = require '../../../server/models/CourseInstance' Level = require '../../../server/models/Level' User = require '../../../server/models/User' request = require '../request' utils = require '../utils' moment = require 'moment' mongoose = require 'mongoose' describe 'Level', -> level = name: 'King\'s Peak 3' description: 'Climb a mountain.' permissions: simplePermissions scripts: [] thangs: [] documentation: {specificArticles: [], generalArticles: []} urlLevel = '/db/level' it 'clears things first', (done) -> clearModels [Level, User], (err) -> expect(err).toBeNull() done() it 'can make a Level.', (done) -> loginJoe -> request.post {uri: getURL(urlLevel), json: level}, (err, res, body) -> expect(res.statusCode).toBe(200) done() it 'get schema', (done) -> request.get {uri: getURL(urlLevel+'/schema')}, (err, res, body) -> expect(res.statusCode).toBe(200) body = JSON.parse(body) expect(body.type).toBeDefined() done() describe 'POST /db/level/:handle', -> it 'creates a new version', utils.wrap (done) -> yield utils.clearModels([Campaign, Course, CourseInstance, Level, User]) admin = yield utils.initAdmin() yield utils.loginUser(admin) @level = yield utils.makeLevel() levelJSON = @level.toObject() levelJSON.name = 'New name' url = getURL("/db/level/#{@level.id}") [res, body] = yield request.postAsync({url: url, json: levelJSON}) expect(res.statusCode).toBe(201) done() it 'does not break the target level if a name change would conflict with another level', utils.wrap (done) -> yield utils.clearModels([Level, User]) user = yield utils.initUser() yield utils.loginUser(user) yield utils.makeLevel({name: 'Taken Name'}) level = yield utils.makeLevel({name: 'Another Level'}) json = _.extend({}, level.toObject(), {name: 'Taken Name'}) [res, body] = yield request.postAsync({url: utils.getURL("/db/level/#{level.id}"), json}) expect(res.statusCode).toBe(409) level = yield Level.findById(level.id) # should be unchanged expect(level.get('slug')).toBe('another-level') expect(level.get('version').isLatestMinor).toBe(true) expect(level.get('version').isLatestMajor).toBe(true) expect(level.get('index')).toBeDefined() done() it 'enforces permissions', -> yield utils.clearModels([Level, User]) user = yield utils.initUser() yield utils.loginUser(user) level = yield utils.makeLevel({description:'Original desc'}) otherUser = yield utils.initUser() yield utils.loginUser(otherUser) json = _.extend({}, level.toObject(), {description: 'Trollin'}) [res, body] = yield request.postAsync({url: utils.getURL("/db/level/#{level.id}"), json}) expect(res.statusCode).toBe(403) level = yield Level.findById(level.id) expect(level.get('description')).toBe('Original desc') done() describe 'GET /db/level/:handle/session', -> describe 'when level IS a course level', -> beforeEach utils.wrap (done) -> yield utils.clearModels([Campaign, Course, CourseInstance, Level, User]) admin = yield utils.initAdmin() yield utils.loginUser(admin) @level = yield utils.makeLevel({type: 'course'}) # To ensure test compares original, not id, make them different. TODO: Make factories do this normally? @level.set('original', new mongoose.Types.ObjectId()) @level.save() @campaign = yield utils.makeCampaign({}, {levels: [@level]}) @course = yield utils.makeCourse({free: true, releasePhase: 'released'}, {campaign: @campaign}) @student = yield utils.initUser({role: 'student'}) members = [@student] teacher = yield utils.initUser({role: 'teacher'}) yield utils.loginUser(teacher) @classroom = yield utils.makeClassroom({aceConfig: { language: 'javascript' }}, { members }) @courseInstance = yield utils.makeCourseInstance({}, { @course, @classroom, members }) @url = getURL("/db/level/#{@level.id}/session") yield utils.loginUser(@student) done() it 'creates a new session if the user is in a course with that level', utils.wrap (done) -> [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(200) expect(body.codeLanguage).toBe('javascript') done() it 'works if the classroom has no aceConfig', utils.wrap (done) -> @classroom.set('aceConfig', undefined) yield @classroom.save() [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(200) expect(body.codeLanguage).toBe('python') done() it 'does not break if the user has a courseInstance without an associated classroom', utils.wrap (done) -> yield @courseInstance.update({$unset: {classroomID: ''}}) [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(402) done() it 'returns 402 if the user is not in a course with that level', utils.wrap (done) -> otherStudent = yield utils.initUser({role: 'student'}) yield utils.loginUser(otherStudent) [res, body] = yield request.getAsync({ uri: @url, json: true }) expect(res.statusCode).toBe(402) expect(res.body.message).toBe('You must be in a course which includes this level to play it') done() describe 'when the course is not free', -> beforeEach utils.wrap (done) -> @course.set({free: false}) yield @course.save() done() it 'returns 402 if the user is not enrolled', utils.wrap (done) -> [res, body] = yield request.getAsync({ uri: @url, json: true }) expect(res.statusCode).toBe(402) expect(res.body.message).toBe('You must be enrolled to access this content') done() it 'creates the session if the user is enrolled', utils.wrap (done) -> @student.set({ coursePrepaid: { _id: {} startDate: moment().subtract(1, 'month').toISOString() endDate: moment().add(1, 'month').toISOString() } }) @student.save() [res, body] = yield request.getAsync({ uri: @url, json: true }) expect(res.statusCode).toBe(200) done() it 'returns 402 if the user\'s license is expired', utils.wrap (done) -> @student.set({ coursePrepaid: { _id: {} startDate: moment().subtract(2, 'month').toISOString() endDate: moment().subtract(1, 'month').toISOString() } }) @student.save() [res, body] = yield request.getAsync({ uri: @url, json: true }) expect(res.statusCode).toBe(402) expect(res.body.message).toBe('You must be enrolled to access this content') done() describe 'when the level is NOT a course level', -> beforeEach utils.wrap (done) -> yield utils.clearModels([Level, User]) admin = yield utils.initAdmin() yield utils.loginUser(admin) @level = yield utils.makeLevel() @player = yield utils.initUser() yield utils.loginUser(@player) @url = getURL("/db/level/#{@level.id}/session") done() it 'idempotently creates and returns a session for that level', utils.wrap (done) -> [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(200) sessionID = body._id [res, body] = yield request.getAsync { uri: @url, json: true } expect(body._id).toBe(sessionID) done() describe 'when the level is not free', -> beforeEach utils.wrap (done) -> yield @level.update({$set: {requiresSubscription: true}}) done() it 'returns 402 for normal users', utils.wrap (done) -> [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(402) done() it 'returns 200 for admins', utils.wrap (done) -> yield @player.update({$set: {permissions: ['admin']}}) [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(200) done() it 'returns 200 for adventurer levels', utils.wrap (done) -> yield @level.update({$set: {adventurer: true}}) [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(200) done() it 'returns 200 for subscribed users', utils.wrap (done) -> yield @player.update({$set: {stripe: {free: true}}}) [res, body] = yield request.getAsync { uri: @url, json: true } expect(res.statusCode).toBe(200) done()