codecombat/server/middleware/campaigns.coffee
2016-05-05 13:11:39 -07:00

82 lines
No EOL
3.5 KiB
CoffeeScript

utils = require '../lib/utils'
errors = require '../commons/errors'
wrap = require 'co-express'
Promise = require 'bluebird'
database = require '../commons/database'
mongoose = require 'mongoose'
Campaign = require '../models/Campaign'
parse = require '../commons/parse'
LevelSession = require '../models/LevelSession'
Achievement = require '../models/Achievement'
Level = require '../models/Level'
slack = require '../slack'
module.exports =
fetchByType: wrap (req, res, next) ->
type = req.query.type
return next() unless type
unless _.contains(Campaign.jsonSchema.properties.type.enum, type)
throw new errors.UnprocessableEntity('Bad campaign type')
dbq = Campaign.find { type: type }
dbq.select(parse.getProjectFromReq(req))
campaigns = yield dbq.exec()
campaigns = (campaign.toObject({req: req}) for campaign in campaigns)
res.status(200).send(campaigns)
put: wrap (req, res) ->
campaign = yield database.getDocFromHandle(req, Campaign)
if not campaign
throw new errors.NotFound('Campaign not found.')
levelsBefore = _.keys(campaign.get('levels'))
hasPermission = req.user.isAdmin()
unless hasPermission or database.isJustFillingTranslations(req, campaign)
throw new errors.Forbidden('Must be an admin or submitting translations to edit a campaign')
database.assignBody(req, campaign)
database.validateDoc(campaign)
levelsAfter = _.keys(campaign.get('levels'))
if not _.isEqual(levelsBefore, levelsAfter)
campaign.set('levelsUpdated', new Date())
campaign = yield campaign.save()
res.status(200).send(campaign.toObject())
docLink = "http://codecombat.com#{req.headers['x-current-path']}"
slack.sendChangedSlackMessage creator: req.user, target: campaign, docLink: docLink
fetchRelatedAchievements: wrap (req, res) ->
campaign = yield database.getDocFromHandle(req, Campaign)
if not campaign
throw new errors.NotFound('Campaign not found.')
levels = campaign.get('levels') or []
project = parse.getProjectFromReq(req)
fetches = []
for level in _.values(levels)
fetches.push Achievement.find({ related: level.original }).select(project)
achievementses = yield fetches
achievements = _.flatten(achievementses)
res.status(200).send((achievement.toObject({req: req}) for achievement in achievements))
fetchRelatedLevels: wrap (req, res) ->
campaign = yield database.getDocFromHandle(req, Campaign)
if not campaign
throw new errors.NotFound('Campaign not found.')
levels = campaign.get('levels') or []
project = parse.getProjectFromReq(req)
sort = { 'version.major': -1, 'version.minor': -1 }
fetches = []
for level in _.values(levels)
query = { original: mongoose.Types.ObjectId(level.original) }
fetches.push Level.findOne(query).sort(sort).select(project)
levels = yield fetches
res.status(200).send((level.toObject({req: req}) for level in levels))
fetchOverworld: wrap (req, res) ->
project = parse.getProjectFromReq(req)
campaigns = yield Campaign.find({type: 'hero'}).select(project)
formatCampaign = (campaign) ->
campaign.toObject({req: req})
campaign.adjacentCampaigns = _.mapValues(campaign.adjacentCampaigns, (a) ->
_.pick(a, ['showIfUnlocked', 'color', 'name', 'description' ]))
for original, level of campaign.levels
campaign.levels[original] = _.pick level, ['locked', 'disabled', 'original', 'rewards', 'slug']
return campaign
res.status(200).send((formatCampaign(campaign) for campaign in campaigns))