This commit is contained in:
Tom Moor
2020-12-16 21:39:37 -08:00
parent 051ecab0fc
commit 5012104a10
2 changed files with 155 additions and 68 deletions

View File

@ -2,6 +2,7 @@
import Router from "koa-router"; import Router from "koa-router";
import Sequelize from "sequelize"; import Sequelize from "sequelize";
import { subtractDate } from "../../shared/utils/date"; import { subtractDate } from "../../shared/utils/date";
import documentCreator from "../commands/documentCreator";
import documentImporter from "../commands/documentImporter"; import documentImporter from "../commands/documentImporter";
import documentMover from "../commands/documentMover"; import documentMover from "../commands/documentMover";
import { import {
@ -823,30 +824,6 @@ router.post("documents.unstar", auth(), async (ctx) => {
}; };
}); });
router.post("documents.create", auth(), createDocumentFromContext);
router.post("documents.import", auth(), async (ctx) => {
if (!ctx.is("multipart/form-data")) {
throw new InvalidRequestError("Request type must be multipart/form-data");
}
const file: any = Object.values(ctx.request.files)[0];
ctx.assertPresent(file, "file is required");
const user = ctx.state.user;
authorize(user, "create", Document);
const { text, title } = await documentImporter({
user,
file,
ip: ctx.request.ip,
});
ctx.body.text = text;
ctx.body.title = title;
await createDocumentFromContext(ctx);
});
router.post("documents.templatize", auth(), async (ctx) => { router.post("documents.templatize", auth(), async (ctx) => {
const { id } = ctx.body; const { id } = ctx.body;
ctx.assertPresent(id, "id is required"); ctx.assertPresent(id, "id is required");
@ -1128,8 +1105,76 @@ router.post("documents.unpublish", auth(), async (ctx) => {
}; };
}); });
// TODO: update to actual `ctx` type router.post("documents.import", auth(), async (ctx) => {
export async function createDocumentFromContext(ctx: any) { const {
publish,
collectionId,
parentDocumentId,
index,
} = ctx.body;
if (!ctx.is("multipart/form-data")) {
throw new InvalidRequestError("Request type must be multipart/form-data");
}
const file: any = Object.values(ctx.request.files)[0];
ctx.assertPresent(file, "file is required");
ctx.assertUuid(collectionId, "collectionId must be an uuid");
if (parentDocumentId) {
ctx.assertUuid(parentDocumentId, "parentDocumentId must be an uuid");
}
if (index) ctx.assertPositiveInteger(index, "index must be an integer (>=0)");
const user = ctx.state.user;
authorize(user, "create", Document);
const collection = await Collection.scope({
method: ["withMembership", user.id],
}).findOne({
where: {
id: collectionId,
teamId: user.teamId,
},
});
authorize(user, "publish", collection);
let parentDocument;
if (parentDocumentId) {
parentDocument = await Document.findOne({
where: {
id: parentDocumentId,
collectionId: collection.id,
},
});
authorize(user, "read", parentDocument, { collection });
}
const { text, title } = await documentImporter({
user,
file,
ip: ctx.request.ip,
});
const document = await documentCreator({
title,
text,
publish,
collectionId,
parentDocumentId,
index,
user,
ip: ctx.request.ip,
});
return (ctx.body = {
data: await presentDocument(document),
policies: presentPolicies(user, [document]),
});
});
router.post("documents.create", auth(), async (ctx) => {
const { const {
title = "", title = "",
text = "", text = "",
@ -1179,56 +1224,24 @@ export async function createDocumentFromContext(ctx: any) {
authorize(user, "read", templateDocument); authorize(user, "read", templateDocument);
} }
let document = await Document.create({ const document = await documentCreator({
title,
text,
publish,
collectionId,
parentDocumentId, parentDocumentId,
editorVersion, templateDocument,
collectionId: collection.id,
teamId: user.teamId,
userId: user.id,
lastModifiedById: user.id,
createdById: user.id,
template, template,
templateId: templateDocument ? templateDocument.id : undefined, index,
title: templateDocument ? templateDocument.title : title, user,
text: templateDocument ? templateDocument.text : text, editorVersion,
});
await Event.create({
name: "documents.create",
documentId: document.id,
collectionId: document.collectionId,
teamId: document.teamId,
actorId: user.id,
data: { title: document.title, templateId },
ip: ctx.request.ip, ip: ctx.request.ip,
}); });
if (publish) {
await document.publish();
await Event.create({
name: "documents.publish",
documentId: document.id,
collectionId: document.collectionId,
teamId: document.teamId,
actorId: user.id,
data: { title: document.title },
ip: ctx.request.ip,
});
}
// reload to get all of the data needed to present (user, collection etc)
// we need to specify publishedAt to bypass default scope that only returns
// published documents
document = await Document.findOne({
where: { id: document.id, publishedAt: document.publishedAt },
});
document.collection = collection;
return (ctx.body = { return (ctx.body = {
data: await presentDocument(document), data: await presentDocument(document),
policies: presentPolicies(user, [document]), policies: presentPolicies(user, [document]),
}); });
} });
export default router; export default router;

View File

@ -0,0 +1,74 @@
// @flow
import { Document, Event, User } from "../models";
export default async function documentCreator({
title = "",
text = "",
publish,
collectionId,
parentDocumentId,
templateDocument,
template,
index,
user,
editorVersion,
ip
}: {
title: string,
text: string,
publish?: boolean,
collectionId: string,
parentDocumentId?: string,
templateDocument?: Document,
template?: boolean,
index?: number,
user: User,
editorVersion?: string,
ip: string
}): Document {
const templateId = templateDocument ? templateDocument.id : undefined;
let document = await Document.create({
parentDocumentId,
editorVersion,
collectionId,
teamId: user.teamId,
userId: user.id,
lastModifiedById: user.id,
createdById: user.id,
template,
templateId,
title: templateDocument ? templateDocument.title : title,
text: templateDocument ? templateDocument.text : text,
});
await Event.create({
name: "documents.create",
documentId: document.id,
collectionId: document.collectionId,
teamId: document.teamId,
actorId: user.id,
data: { title: document.title, templateId },
ip,
});
if (publish) {
await document.publish();
await Event.create({
name: "documents.publish",
documentId: document.id,
collectionId: document.collectionId,
teamId: document.teamId,
actorId: user.id,
data: { title: document.title },
ip,
});
}
// reload to get all of the data needed to present (user, collection etc)
// we need to specify publishedAt to bypass default scope that only returns
// published documents
return Document.findOne({
where: { id: document.id, publishedAt: document.publishedAt },
});
}