* Split permissions for reading documents from updating collection * fix: Admins should have collection read permission, tests * tsc * Add admin option to permission selector * Combine publish and create permissions, update -> createDocuments where appropriate * Plural -> singular * wip * Quick version of collection structure loading, will revisit * Remove documentIds method * stash * fixing tests to account for admin creation * Add self-hosted migration * fix: Allow groups to have admin permission * Prefetch collection documents * fix: Document explorer (move/publish) not working with async documents * fix: Cannot re-parent document to collection by drag and drop * fix: Cannot drag to import into collection item without admin permission * Remove unused isEditor getter
325 lines
11 KiB
TypeScript
325 lines
11 KiB
TypeScript
import { CollectionPermission } from "@shared/types";
|
|
import { CollectionUser, Collection } from "@server/models";
|
|
import {
|
|
buildUser,
|
|
buildTeam,
|
|
buildCollection,
|
|
buildAdmin,
|
|
} from "@server/test/factories";
|
|
import { setupTestDatabase } from "@server/test/support";
|
|
import { serialize } from "./index";
|
|
|
|
setupTestDatabase();
|
|
|
|
describe("admin", () => {
|
|
it("should allow updating collection but not reading documents", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildAdmin({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: null,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.readDocument).toEqual(false);
|
|
expect(abilities.createDocument).toEqual(false);
|
|
expect(abilities.share).toEqual(false);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.update).toEqual(true);
|
|
});
|
|
});
|
|
|
|
describe("member", () => {
|
|
describe("admin permission", () => {
|
|
it("should allow updating collection", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
await CollectionUser.create({
|
|
createdById: user.id,
|
|
collectionId: collection.id,
|
|
userId: user.id,
|
|
permission: CollectionPermission.Admin,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.createDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(true);
|
|
});
|
|
});
|
|
|
|
describe("read_write permission", () => {
|
|
it("should allow read write documents for team member", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
const abilities = serialize(user, collection);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
|
|
it("should override read membership permission", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
await CollectionUser.create({
|
|
createdById: user.id,
|
|
collectionId: collection.id,
|
|
userId: user.id,
|
|
permission: CollectionPermission.Read,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
});
|
|
|
|
describe("read permission", () => {
|
|
it("should allow read permissions for team member", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.Read,
|
|
});
|
|
const abilities = serialize(user, collection);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
expect(abilities.share).toEqual(false);
|
|
});
|
|
|
|
it("should allow override with read_write membership permission", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.Read,
|
|
});
|
|
await CollectionUser.create({
|
|
createdById: user.id,
|
|
collectionId: collection.id,
|
|
userId: user.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
});
|
|
|
|
describe("no permission", () => {
|
|
it("should allow no permissions for team member", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: null,
|
|
});
|
|
const abilities = serialize(user, collection);
|
|
expect(abilities.read).toEqual(false);
|
|
expect(abilities.readDocument).toEqual(false);
|
|
expect(abilities.createDocument).toEqual(false);
|
|
expect(abilities.share).toEqual(false);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
|
|
it("should allow override with team member membership permission", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: null,
|
|
});
|
|
await CollectionUser.create({
|
|
createdById: user.id,
|
|
collectionId: collection.id,
|
|
userId: user.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.createDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("viewer", () => {
|
|
describe("read_write permission", () => {
|
|
it("should allow read permissions for viewer", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
isViewer: true,
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
const abilities = serialize(user, collection);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.createDocument).toEqual(false);
|
|
expect(abilities.update).toEqual(false);
|
|
expect(abilities.share).toEqual(false);
|
|
});
|
|
|
|
it("should override read membership permission", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
isViewer: true,
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
await CollectionUser.create({
|
|
createdById: user.id,
|
|
collectionId: collection.id,
|
|
userId: user.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
});
|
|
|
|
describe("read permission", () => {
|
|
it("should allow override with read_write membership permission", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
isViewer: true,
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: CollectionPermission.Read,
|
|
});
|
|
await CollectionUser.create({
|
|
createdById: user.id,
|
|
collectionId: collection.id,
|
|
userId: user.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.createDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
});
|
|
|
|
describe("no permission", () => {
|
|
it("should allow no permissions for viewer", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
isViewer: true,
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: null,
|
|
});
|
|
const abilities = serialize(user, collection);
|
|
expect(abilities.read).toEqual(false);
|
|
expect(abilities.update).toEqual(false);
|
|
expect(abilities.share).toEqual(false);
|
|
});
|
|
|
|
it("should allow override with team member membership permission", async () => {
|
|
const team = await buildTeam();
|
|
const user = await buildUser({
|
|
isViewer: true,
|
|
teamId: team.id,
|
|
});
|
|
const collection = await buildCollection({
|
|
teamId: team.id,
|
|
permission: null,
|
|
});
|
|
await CollectionUser.create({
|
|
createdById: user.id,
|
|
collectionId: collection.id,
|
|
userId: user.id,
|
|
permission: CollectionPermission.ReadWrite,
|
|
});
|
|
// reload to get membership
|
|
const reloaded = await Collection.scope({
|
|
method: ["withMembership", user.id],
|
|
}).findByPk(collection.id);
|
|
const abilities = serialize(user, reloaded);
|
|
expect(abilities.read).toEqual(true);
|
|
expect(abilities.readDocument).toEqual(true);
|
|
expect(abilities.createDocument).toEqual(true);
|
|
expect(abilities.share).toEqual(true);
|
|
expect(abilities.update).toEqual(false);
|
|
});
|
|
});
|
|
});
|