UNPKG

@inrupt/solid-client

Version:

Make your web apps work with Solid Pods.

576 lines (552 loc) • 16.1 kB
// Copyright Inrupt Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal in // the Software without restriction, including without limitation the rights to use, // copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the // Software, and to permit persons to whom the Software is furnished to do so, // subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, // INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A // PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE // SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // import { describe, it, expect } from "@jest/globals"; import { ACL, ACP } from "../constants"; import { DEFAULT_ACCESS_CONTROL_RESOURCE_URL, DEFAULT_DOMAIN, } from "../mock/constants"; import { createDatasetFromSubjects } from "../mock/dataset"; import { mockAccessControlledResource } from "../mock/mockAccessControlledResource"; import { getAgentAccessAll } from "./getAgentAccessAll"; describe("getAgentAccessAll", () => { it("returns an empty object for an empty ACR", async () => { const resource = mockAccessControlledResource(); expect(resource.internal_acp.acr.graphs).toStrictEqual({ default: {} }); expect(await getAgentAccessAll(resource)).toStrictEqual({}); }); it("returns one allowed access modes an ACR with 1 policy", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [[ACP.apply, [DEFAULT_DOMAIN.concat("p1")]]], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("bob")]]], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("bob")]: { read: true, append: false, write: false, controlRead: false, controlWrite: false, }, }); }); it("returns all access modes defined in multiple policies", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [ [ACP.apply, [DEFAULT_DOMAIN.concat("p1")]], [ACP.access, [DEFAULT_DOMAIN.concat("p2")]], ], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read, ACL.Append, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("p2"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("bob")]]], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("bob")]: { read: true, append: true, write: true, controlRead: true, controlWrite: true, }, }); }); it("returns access modes when the same policy is used for ACR and Resource", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [ [ACP.apply, [DEFAULT_DOMAIN.concat("p1")]], [ACP.access, [DEFAULT_DOMAIN.concat("p1")]], ], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read, ACL.Append, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("bob")]]], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("bob")]: { read: true, append: true, write: true, controlRead: true, controlWrite: true, }, }); }); it("doesn't return access modes when the agent is not matched", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [ [ACP.apply, [DEFAULT_DOMAIN.concat("p1")]], [ACP.access, [DEFAULT_DOMAIN.concat("p2")]], ], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read, ACL.Append, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("p2"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m2")]], [ACP.allow, [ACL.Read, ACL.Append, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [ [ ACP.agent, [DEFAULT_DOMAIN.concat("alice"), DEFAULT_DOMAIN.concat("bob")], ], ], ], [ DEFAULT_DOMAIN.concat("m2"), [[ACP.agent, [DEFAULT_DOMAIN.concat("alice")]]], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("alice")]: { read: true, append: true, write: true, controlRead: true, controlWrite: true, }, [DEFAULT_DOMAIN.concat("bob")]: { read: true, append: true, write: true, controlRead: false, controlWrite: false, }, }); }); it("returns access modes without the ones denied", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [ [ ACP.apply, [DEFAULT_DOMAIN.concat("p1"), DEFAULT_DOMAIN.concat("p2")], ], [ ACP.access, [DEFAULT_DOMAIN.concat("p3"), DEFAULT_DOMAIN.concat("p4")], ], ], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read, ACL.Append, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("p2"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.deny, [ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("p3"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("p4"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.deny, [ACL.Read]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("bob")]]], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("bob")]: { read: true, append: true, write: false, controlRead: false, controlWrite: true, }, }); }); it("returns access modes only if all of matchers match", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [ [ ACP.apply, [DEFAULT_DOMAIN.concat("p1"), DEFAULT_DOMAIN.concat("p2")], ], ], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ ACP.allOf, [DEFAULT_DOMAIN.concat("m1"), DEFAULT_DOMAIN.concat("m2")], ], [ACP.allow, [ACL.Read, ACL.Append, ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("p2"), [ [ ACP.allOf, [DEFAULT_DOMAIN.concat("m1"), DEFAULT_DOMAIN.concat("m3")], ], [ACP.allow, [ACL.Write]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("bob")]]], ], [ DEFAULT_DOMAIN.concat("m2"), [ [ ACP.agent, [DEFAULT_DOMAIN.concat("alice"), DEFAULT_DOMAIN.concat("latifa")], ], ], ], [ DEFAULT_DOMAIN.concat("m3"), [ [ ACP.agent, [DEFAULT_DOMAIN.concat("malik"), DEFAULT_DOMAIN.concat("bob")], ], ], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("bob")]: { read: false, append: false, write: true, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("malik")]: { read: false, append: false, write: false, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("alice")]: { read: false, append: false, write: false, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("latifa")]: { read: false, append: false, write: false, controlRead: false, controlWrite: false, }, }); }); it("returns access modes only if any of matchers match", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [[ACP.apply, [DEFAULT_DOMAIN.concat("p1")]]], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ ACP.anyOf, [DEFAULT_DOMAIN.concat("m1"), DEFAULT_DOMAIN.concat("m2")], ], [ACP.allow, [ACL.Read]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("jorge")]]], ], [ DEFAULT_DOMAIN.concat("m2"), [ [ ACP.agent, [DEFAULT_DOMAIN.concat("latifa"), DEFAULT_DOMAIN.concat("bob")], ], ], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("bob")]: { read: true, append: false, write: false, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("latifa")]: { read: true, append: false, write: false, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("jorge")]: { read: true, append: false, write: false, controlRead: false, controlWrite: false, }, }); }); it("returns access modes only if none of matchers don't match", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [[ACP.apply, [DEFAULT_DOMAIN.concat("p1")]]], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ACP.anyOf, [DEFAULT_DOMAIN.concat("m2")]], [ ACP.noneOf, [DEFAULT_DOMAIN.concat("m1"), DEFAULT_DOMAIN.concat("m3")], ], [ACP.allow, [ACL.Read]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("jorge")]]], ], [ DEFAULT_DOMAIN.concat("m2"), [ [ ACP.agent, [DEFAULT_DOMAIN.concat("latifa"), DEFAULT_DOMAIN.concat("bob")], ], ], ], [ DEFAULT_DOMAIN.concat("m3"), [ [ ACP.agent, [DEFAULT_DOMAIN.concat("malik"), DEFAULT_DOMAIN.concat("bob")], ], ], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("jorge")]: { read: false, append: false, write: false, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("latifa")]: { read: true, append: false, write: false, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("bob")]: { read: false, append: false, write: false, controlRead: false, controlWrite: false, }, [DEFAULT_DOMAIN.concat("malik")]: { read: false, append: false, write: false, controlRead: false, controlWrite: false, }, }); }); it("doesn't match if there is only a noneOf matcher matching", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [[ACP.apply, [DEFAULT_DOMAIN.concat("p1")]]], ], [ DEFAULT_DOMAIN.concat("p1"), [ [ACP.noneOf, [DEFAULT_DOMAIN.concat("m1")]], [ACP.allow, [ACL.Read]], ], ], [ DEFAULT_DOMAIN.concat("m1"), [[ACP.agent, [DEFAULT_DOMAIN.concat("bob")]]], ], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({ [DEFAULT_DOMAIN.concat("bob")]: { read: false, append: false, write: false, controlRead: false, controlWrite: false, }, }); }); it("doesn't match if there is no matchers", async () => { const resource = mockAccessControlledResource( createDatasetFromSubjects([ [ DEFAULT_ACCESS_CONTROL_RESOURCE_URL, [[ACP.accessControl, [DEFAULT_DOMAIN.concat("ac1")]]], ], [ DEFAULT_DOMAIN.concat("ac1"), [[ACP.apply, [DEFAULT_DOMAIN.concat("p1")]]], ], [DEFAULT_DOMAIN.concat("p1"), [[ACP.allow, [ACL.Read]]]], ]), ); expect(await getAgentAccessAll(resource)).toStrictEqual({}); }); });