overleaf/services/web/app/coffee/Features/Project/ProjectController.coffee

295 lines
8.9 KiB
CoffeeScript
Raw Normal View History

2014-04-08 10:28:18 -04:00
async = require("async")
logger = require("logger-sharelatex")
projectDeleter = require("./ProjectDeleter")
projectDuplicator = require("./ProjectDuplicator")
projectCreationHandler = require("./ProjectCreationHandler")
2014-04-28 12:47:47 -04:00
editorController = require("../Editor/EditorController")
metrics = require('../../infrastructure/Metrics')
sanitize = require('sanitizer')
Project = require('../../models/Project').Project
User = require('../../models/User').User
TagsHandler = require("../Tags/TagsHandler")
SubscriptionLocator = require("../Subscription/SubscriptionLocator")
LimitationsManager = require("../Subscription/LimitationsManager")
_ = require("underscore")
Settings = require("settings-sharelatex")
SecurityManager = require("../../managers/SecurityManager")
2014-06-24 15:28:53 -04:00
fs = require "fs"
2014-06-16 08:34:38 -04:00
module.exports = ProjectController =
2014-06-25 08:51:02 -04:00
updateProjectSettings: (req, res, next) ->
project_id = req.params.Project_id
jobs = []
if req.body.compiler?
jobs.push (callback) ->
editorController.setCompiler project_id, req.body.compiler, callback
if req.body.name?
jobs.push (callback) ->
editorController.renameProject project_id, req.body.name, callback
if req.body.spellCheckLanguage?
jobs.push (callback) ->
editorController.setSpellCheckLanguage project_id, req.body.spellCheckLanguage, callback
if req.body.rootDocId?
jobs.push (callback) ->
editorController.setRootDoc project_id, req.body.rootDocId, callback
if req.body.publicAccessLevel?
jobs.push (callback) ->
editorController.setPublicAccessLevel project_id, req.body.publicAccessLevel, callback
async.series jobs, (error) ->
return next(error) if error?
res.send(204)
deleteProject: (req, res) ->
project_id = req.params.Project_id
2014-06-04 07:45:04 -04:00
forever = req.query?.forever?
logger.log project_id: project_id, forever: forever, "received request to delete project"
if forever
doDelete = projectDeleter.deleteProject
else
doDelete = projectDeleter.archiveProject
doDelete project_id, (err)->
if err?
res.send 500
else
res.send 200
restoreProject: (req, res) ->
project_id = req.params.Project_id
logger.log project_id:project_id, "received request to restore project"
projectDeleter.restoreProject project_id, (err)->
if err?
res.send 500
else
res.send 200
cloneProject: (req, res)->
metrics.inc "cloned-project"
project_id = req.params.Project_id
projectName = req.body.projectName
logger.log project_id:project_id, projectName:projectName, "cloning project"
if !req.session.user?
return res.send redir:"/register"
projectDuplicator.duplicate req.session.user, project_id, projectName, (err, project)->
if err?
logger.error err:err, project_id: project_id, user_id: req.session.user._id, "error cloning project"
return next(err)
res.send(project_id:project._id)
newProject: (req, res)->
user = req.session.user
2014-07-09 06:05:00 -04:00
projectName = req.body.projectName?.trim()
template = req.body.template
logger.log user: user, type: template, name: projectName, "creating project"
2014-04-08 10:28:18 -04:00
async.waterfall [
(cb)->
if template == 'example'
projectCreationHandler.createExampleProject user._id, projectName, cb
else
2014-04-08 10:28:18 -04:00
projectCreationHandler.createBasicProject user._id, projectName, cb
], (err, project)->
if err?
logger.error err: err, project: project, user: user, name: projectName, type: template, "error creating project"
res.send 500
else
logger.log project: project, user: user, name: projectName, type: template, "created project"
res.send {project_id:project._id}
2014-04-28 12:47:47 -04:00
renameProject: (req, res)->
project_id = req.params.Project_id
newName = req.body.newProjectName
editorController.renameProject project_id, newName, (err)->
if err?
logger.err err:err, project_id:project_id, newName:newName, "problem renaming project"
res.send 500
else
res.send 200
projectListPage: (req, res, next)->
timer = new metrics.Timer("project-list")
user_id = req.session.user._id
async.parallel {
tags: (cb)->
TagsHandler.getAllTags user_id, cb
projects: (cb)->
2014-06-16 08:34:38 -04:00
Project.findAllUsersProjects user_id, 'name lastUpdated publicAccesLevel archived owner_ref', cb
hasSubscription: (cb)->
LimitationsManager.userHasSubscriptionOrIsGroupMember req.session.user, cb
}, (err, results)->
2014-04-08 16:06:01 -04:00
if err?
logger.err err:err, "error getting data for project list page"
2014-06-16 08:34:38 -04:00
return next(err)
logger.log results:results, user_id:user_id, "rendering project list"
2014-06-16 08:34:38 -04:00
tags = results.tags[0]
projects = ProjectController._buildProjectList results.projects[0], results.projects[1], results.projects[2]
ProjectController._injectProjectOwners projects, (error, projects) ->
return next(error) if error?
2014-06-17 11:19:40 -04:00
viewModel = {
2014-08-21 08:02:28 -04:00
title:'your_projects'
2014-06-16 08:34:38 -04:00
priority_title: true
projects: projects
tags: tags
hasSubscription: results.hasSubscription
2014-06-16 08:34:38 -04:00
}
2014-06-17 11:19:40 -04:00
if Settings?.algolia?.app_id? and Settings?.algolia?.read_only_api_key?
2014-06-17 11:19:40 -04:00
viewModel.showUserDetailsArea = true
viewModel.algolia_api_key = Settings.algolia.read_only_api_key
viewModel.algolia_app_id = Settings.algolia.app_id
2014-06-17 11:19:40 -04:00
else
viewModel.showUserDetailsArea = false
res.render 'project/list', viewModel
2014-06-16 08:34:38 -04:00
timer.done()
loadEditor: (req, res, next)->
timer = new metrics.Timer("load-editor")
if !Settings.editorIsOpen
2014-08-01 08:47:14 -04:00
return res.render("general/closed", {title:"updating_site"})
if req.session.user?
user_id = req.session.user._id
anonymous = false
else
anonymous = true
user_id = 'openUser'
2014-04-08 12:59:29 -04:00
project_id = req.params.Project_id
2014-04-08 12:59:29 -04:00
async.parallel {
project: (cb)->
Project.findPopulatedById project_id, cb
user: (cb)->
if user_id == 'openUser'
cb null, defaultSettingsForAnonymousUser(user_id)
else
User.findById user_id, cb
2014-04-08 12:59:29 -04:00
subscription: (cb)->
if user_id == 'openUser'
return cb()
2014-04-08 12:59:29 -04:00
SubscriptionLocator.getUsersSubscription user_id, cb
}, (err, results)->
if err?
logger.err err:err, "error getting details for project page"
return next err
2014-04-08 12:59:29 -04:00
project = results.project
user = results.user
subscription = results.subscription
2014-04-08 16:06:01 -04:00
2014-04-08 12:59:29 -04:00
SecurityManager.userCanAccessProject user, project, (canAccess, privilegeLevel)->
if !canAccess
return res.send 401
if subscription? and subscription.freeTrial? and subscription.freeTrial.expiresAt?
allowedFreeTrial = !!subscription.freeTrial.allowed || true
res.render 'project/editor',
title: project.name
priority_title: true
bodyClasses: ["editor"]
project : project
2014-06-21 17:20:37 -04:00
project_id : project._id
2014-07-09 06:05:00 -04:00
user : {
2014-04-08 12:59:29 -04:00
id : user.id
email : user.email
first_name : user.first_name
last_name : user.last_name
referal_id : user.referal_id
subscription :
freeTrial: {allowed: allowedFreeTrial}
2014-07-09 06:05:00 -04:00
}
userSettings: {
2014-04-08 12:59:29 -04:00
mode : user.ace.mode
theme : user.ace.theme
fontSize : user.ace.fontSize
autoComplete: user.ace.autoComplete
pdfViewer : user.ace.pdfViewer
2014-07-09 06:05:00 -04:00
}
2014-04-08 12:59:29 -04:00
privilegeLevel: privilegeLevel
chatUrl: Settings.apis.chat.url
anonymous: anonymous
languages: Settings.languages
2014-06-24 15:28:53 -04:00
themes: THEME_LIST
2014-04-08 12:59:29 -04:00
timer.done()
2014-06-16 08:34:38 -04:00
_buildProjectList: (ownedProjects, sharedProjects, readOnlyProjects)->
projects = []
for project in ownedProjects
projects.push ProjectController._buildProjectViewModel(project, "owner")
for project in sharedProjects
projects.push ProjectController._buildProjectViewModel(project, "readWrite")
for project in readOnlyProjects
projects.push ProjectController._buildProjectViewModel(project, "readOnly")
return projects
_buildProjectViewModel: (project, accessLevel) ->
{
id: project._id
name: project.name
lastUpdated: project.lastUpdated
publicAccessLevel: project.publicAccesLevel
accessLevel: accessLevel
archived: !!project.archived
owner_ref: project.owner_ref
}
_injectProjectOwners: (projects, callback = (error, projects) ->) ->
users = {}
for project in projects
if project.owner_ref?
users[project.owner_ref.toString()] = true
jobs = []
for user_id, _ of users
do (user_id) ->
jobs.push (callback) ->
User.findById user_id, "first_name last_name", (error, user) ->
return callback(error) if error?
users[user_id] = user
callback()
async.series jobs, (error) ->
for project in projects
if project.owner_ref?
project.owner = users[project.owner_ref.toString()]
callback null, projects
defaultSettingsForAnonymousUser = (user_id)->
2014-04-08 12:59:29 -04:00
id : user_id
ace:
mode:'none'
theme:'textmate'
fontSize: '12'
autoComplete: true
spellCheckLanguage: ""
pdfViewer: ""
subscription:
freeTrial:
allowed: true
featureSwitches:
dropbox: false
trackChanges: false
2014-06-24 15:28:53 -04:00
THEME_LIST = []
do generateThemeList = () ->
2014-07-09 12:59:04 -04:00
files = fs.readdirSync __dirname + '/../../../../public/js/ace'
2014-06-24 15:28:53 -04:00
for file in files
2014-07-09 12:59:04 -04:00
if file.slice(-2) == "js" and file.match(/^theme-/)
cleanName = file.slice(0,-3).slice(6)
2014-06-24 15:28:53 -04:00
THEME_LIST.push cleanName