backend/app/controllers/api/v1/Page.controller.js

331 lines
10 KiB
JavaScript
Raw Normal View History

const Controller = require('libflitter/controller/Controller')
const PageModel = require('../../../models/api/Page.model')
const Node = require('../../../models/api/Node.model')
/*
* Page Controller
* -------------------------------------------------------------
* Put some description here!
*/
class Page extends Controller {
async get_page(req, res) {
const PageId = req.params.PageId
const user = req.user
const page = await PageModel.findOne({UUID: PageId})
if ( !page ) return res.status(404).message('Page not found with that ID.').api({})
if ( !(await page.is_accessible_by(user)) ) return req.security.deny()
page.level = await page.access_level_for(req.user)
return res.api(page)
}
async save_page(req, res) {
const PageId = req.params.PageId
let page;
if ( PageId ) {
page = await PageModel.findOne({UUID: PageId})
if ( !page ) return res.status(404).message('Page not found with that ID.').api({})
2020-10-29 03:55:04 +00:00
if ( !(await page.is_accessible_by(req.user, 'update')) ) return res.security.deny()
} else {
page = new PageModel
page.CreatedUserId = req.user.id
page.OrgUserId = req.user._id
}
if ( !req.body.Name ) return res.status(400).message('Missing required: Name').api({})
page.Name = req.body.Name
if ( 'IsPublic' in req.body ) {
page.IsPublic = !!req.body.IsPublic
}
if ( 'IsVisibleInMenu' in req.body ) {
page.IsVisibleInMenu = !!req.body.IsVisibleInMenu
}
/*let parent;
if ( !req.body.ParentId ) return res.status(400).message('Missing required: ParentId').api({})
else {
parent = await PageModel.findOne({UUID: req.body.ParentId})
if ( !parent ) return res.status(404).message('Parent page not found with that ID.').api({})
if ( !(await parent.is_accessible_by(req.user, 'update')) ) return req.security.kickout()
}*/ // TODO re-implement this to account for sharing
page.UpdatedAt = new Date
page.UpdateUserId = req.user._id
await page.save()
return res.api(page)
}
async get_nodes(req, res) {
const PageId = req.params.PageId
let page;
if ( PageId ) {
page = await PageModel.findOne({UUID: PageId})
if ( !page ) return res.status(404).message('Page not found with that ID.').api({})
if ( !(await page.is_accessible_by(req.user)) ) return req.security.deny()
}
const nodes = await Node.find({PageId: page.UUID});
const assoc_nodes = {}
nodes.forEach(node => {
assoc_nodes[node.UUID] = node
})
const return_nodes = [];
for ( const uuid of page.NodeIds ) {
return_nodes.push(assoc_nodes[uuid])
}
return res.api(return_nodes);
}
async save_node_to_page(req, res) {
const PageId = req.params.PageId
const page = await PageModel.findOne({UUID: PageId})
if ( !page ) return res.status(404).message('Page not found with that ID.').api({})
2020-10-29 03:55:04 +00:00
if ( !(await page.is_accessible_by(req.user, 'update')) ) return res.security.deny()
const nodes = await Node.find({PageId: page.UUID})
const assoc_nodes = {}
nodes.forEach(node => {
assoc_nodes[node.UUID] = node
})
if ( !req.body.nodeData ) {
return res.status(400).message('Missing nodeData body').api()
}
const node = req.body.nodeData;
if ( node.UUID && assoc_nodes[node.UUID] ) {
assoc_nodes[node.UUID].update_from_raw(node)
assoc_nodes[node.UUID].UpdatedAt = new Date
assoc_nodes[node.UUID].UpdateUserId = req.user._id
assoc_nodes[node.UUID].save()
return res.api(assoc_nodes[node.UUID])
} else {
const node_obj = new Node({
Type: node.Type,
Value: node.Value,
PageId,
CreatedUserId: req.user._id,
UpdateUserId: req.user._id,
})
2020-10-29 03:55:04 +00:00
if ( node.UUID ) {
const existingUUID = await Node.findOne({ UUID: node.UUID })
if ( !existingUUID ) {
node_obj.UUID = node.UUID
}
}
await node_obj.save()
page.NodeIds.push(node_obj.UUID);
await page.save();
return res.api(node_obj)
}
}
async save_nodes(req, res) {
const PageId = req.params.PageId
let page;
if ( PageId ) {
page = await PageModel.findOne({UUID: PageId})
if ( !page ) return res.status(404).message('Page not found with that ID.').api({})
if ( !(await page.is_accessible_by(req.user, 'update')) ) return req.security.deny()
}
const nodes = await Node.find({PageId: page.UUID})
const assoc_nodes = {}
nodes.forEach(node => {
assoc_nodes[node.UUID] = node
})
if ( !Array.isArray(req.body) ) return res.status(400).message('Invalid request body. Should be array of nodes.').api({})
const updated_node_ids = []
for ( let node of req.body ) {
if ( node.UUID && Object.keys(assoc_nodes).includes(node.UUID) ) updated_node_ids.push(node.UUID)
}
const updated_nodes = []
for ( let node of req.body ) {
if ( node.UUID && assoc_nodes[node.UUID] ) {
assoc_nodes[node.UUID].update_from_raw(node)
assoc_nodes[node.UUID].UpdatedAt = new Date
assoc_nodes[node.UUID].UpdateUserId = req.user._id
updated_nodes.push(assoc_nodes[node.UUID])
} else {
const node_obj = new Node({
Type: node.Type,
Value: node.Value,
PageId,
CreatedUserId: req.user._id,
UpdateUserId: req.user._id,
})
await node_obj.save()
updated_nodes.push(node_obj)
}
}
// console.log('updated nodes', updated_nodes)
let delete_nodes = nodes
for ( const node of updated_nodes ) {
await node.save()
delete_nodes = delete_nodes.filter(n => {
return n.UUID !== node.UUID
})
}
for ( const node of delete_nodes ) {
await node.delete()
}
page.NodeIds = updated_nodes.map(x => x.UUID)
await page.save()
res.api(updated_nodes)
// Step 1: make sure the updated nodes are updated
// Make sure the deleted nodes are deleted
// Make sure the new nodes are created
// Make sure the order is correct
}
async create_top_level(req, res) {
if ( !req.body.name ) {
return res.status(400).message('Missing required field: name').api({})
}
const root_page = await req.user.get_root_page()
const new_page = new PageModel({
Name: req.body.name,
OrgUserId: req.user._id,
ParentId: root_page.UUID,
CreatedUserId: req.user.id,
UpdateUserId: req.user.id
})
2020-10-29 03:55:04 +00:00
if ( req.body.UUID ) {
const existingUUID = await PageModel.findOne({UUID: req.body.UUID})
if ( !existingUUID ) {
new_page.UUID = req.body.UUID
}
}
await new_page.save()
root_page.ChildPageIds.push(new_page.UUID)
await root_page.save()
req.user.allow(`page:${new_page.UUID}`)
await req.user.save()
2020-10-29 03:55:04 +00:00
if ( !req.body.omit_starter ) {
const starter_node = new Node({
Type: 'paragraph',
Value: {
Value: 'Double-click to edit...',
},
PageId: new_page.UUID,
CreatedUserId: req.user.id,
UpdateUserId: req.user.id
})
2020-10-29 03:55:04 +00:00
await starter_node.save()
new_page.NodeIds.push(starter_node.UUID)
await new_page.save()
}
return res.api(new_page)
}
async create_child(req, res) {
if ( !req.body.name ) {
return res.status(400).message('Missing required field: name').api({})
}
if ( !req.body.parentId ) {
return res.status(400).message('Missing required field: parentId').api({})
}
// Get the parent
const parent = await PageModel.findOne({UUID: req.body.parentId})
if ( !parent ) {
return res.status(404).message('Unable to find parent with that ID.').api({})
} else if ( !(await parent.is_accessible_by(req.user, 'manage')) ) {
return res.security.deny()
}
const new_page = new PageModel({
Name: req.body.name,
OrgUserId: req.user._id,
ParentId: parent.UUID,
CreatedUserId: req.user.id,
UpdateUserId: req.user.id,
})
2020-10-29 03:55:04 +00:00
if ( req.body.UUID ) {
const existingUUID = await PageModel.findOne({UUID: req.body.UUID})
if ( !existingUUID ) {
new_page.UUID = req.body.UUID
}
}
await new_page.save()
parent.ChildPageIds.push(new_page.UUID)
await parent.save()
req.user.allow(`page:${new_page.UUID}`)
await req.user.save()
2020-10-29 03:55:04 +00:00
if ( !req.body.omit_starter ) {
const starter_node = new Node({
Type: 'paragraph',
Value: {
Value: 'Click to edit...',
},
PageId: new_page.UUID,
CreatedUserId: req.user.id,
UpdateUserId: req.user.id
})
2020-10-29 03:55:04 +00:00
await starter_node.save()
new_page.NodeIds.push(starter_node.UUID)
await new_page.save()
}
return res.api(new_page)
}
async delete_page(req, res) {
const PageId = req.params.PageId
let page;
if ( PageId ) {
page = await PageModel.findOne({UUID: PageId})
if ( !page ) return res.status(404).message('Page not found with that ID.').api({})
if ( !(await page.is_accessible_by(req.user, 'manage')) ) return req.security.deny()
if ( page.ParentId === '0' ) return req.security.kickout()
}
page.Active = false
page.DeletedAt = new Date
await page.save()
return res.api({})
}
}
module.exports = exports = Page