diff --git a/packages/app-aco/package.json b/packages/app-aco/package.json index e3d6f0c97c2..4e70cff95d3 100644 --- a/packages/app-aco/package.json +++ b/packages/app-aco/package.json @@ -21,6 +21,7 @@ "@webiny/app-admin": "0.0.0", "@webiny/app-headless-cms-common": "0.0.0", "@webiny/app-security": "0.0.0", + "@webiny/app-utils": "0.0.0", "@webiny/app-wcp": "0.0.0", "@webiny/form": "0.0.0", "@webiny/plugins": "0.0.0", diff --git a/packages/app-aco/src/Folders.tsx b/packages/app-aco/src/Folders.tsx deleted file mode 100644 index 170e5c72ae2..00000000000 --- a/packages/app-aco/src/Folders.tsx +++ /dev/null @@ -1,21 +0,0 @@ -import React from "react"; -import { Plugin } from "@webiny/app-admin"; -import { FoldersApiProvider } from "~/contexts/FoldersApi"; - -interface FoldersApiProviderHOCProps { - children: React.ReactNode; -} - -const FoldersApiProviderHOC = (Component: React.ComponentType) => { - return function FoldersApiProviderHOC({ children }: FoldersApiProviderHOCProps) { - return ( - - {children} - - ); - }; -}; - -export const Folders = () => { - return ; -}; diff --git a/packages/app-aco/src/components/FolderTree/List/index.tsx b/packages/app-aco/src/components/FolderTree/List/index.tsx index 74eb13c089f..f6f0a24ac45 100644 --- a/packages/app-aco/src/components/FolderTree/List/index.tsx +++ b/packages/app-aco/src/components/FolderTree/List/index.tsx @@ -13,7 +13,7 @@ import { Node } from "../Node"; import { NodePreview } from "../NodePreview"; import { Placeholder } from "../Placeholder"; import { createInitialOpenList, createTreeData } from "./utils"; -import { useFolders } from "~/hooks"; +import { useGetFolderLevelPermission, useUpdateFolder } from "~/features"; import { ROOT_FOLDER } from "~/constants"; import { DndFolderItemData, FolderItem } from "~/types"; import { FolderProvider } from "~/contexts/folder"; @@ -33,7 +33,9 @@ export const List = ({ hiddenFolderIds, enableActions }: ListProps) => { - const { updateFolder, folderLevelPermissions: flp } = useFolders(); + const { updateFolder } = useUpdateFolder(); + const { getFolderLevelPermission: canManageStructure } = + useGetFolderLevelPermission("canManageStructure"); const { showSnackbar } = useSnackbar(); const [treeData, setTreeData] = useState[]>([]); const [initialOpenList, setInitialOpenList] = useState(); @@ -67,13 +69,10 @@ export const List = ({ setTreeData(newTree); - await updateFolder( - { - ...item, - parentId: dropTargetId !== ROOT_FOLDER ? (dropTargetId as string) : null - }, - { refetchFoldersList: true } - ); + await updateFolder({ + ...item, + parentId: dropTargetId !== ROOT_FOLDER ? (dropTargetId as string) : null + }); } catch (error) { // If an error occurred, revert the tree back to its previous state setTreeData(oldTree); @@ -98,9 +97,9 @@ export const List = ({ const canDrag = useCallback( (folderId: string) => { const isRootFolder = folderId === ROOT_FOLDER; - return !isRootFolder && flp.canManageStructure(folderId); + return !isRootFolder && canManageStructure(folderId); }, - [flp.canManageStructure] + [canManageStructure] ); return ( diff --git a/packages/app-aco/src/components/FolderTree/List/utils.ts b/packages/app-aco/src/components/FolderTree/List/utils.ts index 4659211a1e5..d3b7010315c 100644 --- a/packages/app-aco/src/components/FolderTree/List/utils.ts +++ b/packages/app-aco/src/components/FolderTree/List/utils.ts @@ -3,9 +3,9 @@ import { DndFolderItemData, FolderItem } from "~/types"; import { ROOT_FOLDER } from "~/constants"; /** - * Transform an array of folders returned by useFolders hook into an array of elements for the tree component. + * Transform an array of folders returned by folders cache into an array of elements for the tree component. * - * @param folders list of folders returned by useFolders hook. + * @param folders list of folders returned by folders cache. * @param focusedNodeId id of the current folder selected/focused. * @param hiddenFolderIds list ids of the folder you don't want to show within the list. * @return array of elements to render the tree component. @@ -37,7 +37,7 @@ export const createTreeData = ( * Return an array of ids of open folders, based on the current focused folder id, its parent folders and the folders * opened by user interaction. * - * @param folders list of folders returned by useFolders hook. + * @param folders list of folders returned by folders cache. * @param openIds list of open folders ids. * @param focusedId id of the current folder selected/focused. * @return array of ids of open folders. diff --git a/packages/app-aco/src/components/FolderTree/index.tsx b/packages/app-aco/src/components/FolderTree/index.tsx index b5e6edb5f8e..b874066ae59 100644 --- a/packages/app-aco/src/components/FolderTree/index.tsx +++ b/packages/app-aco/src/components/FolderTree/index.tsx @@ -1,6 +1,6 @@ import React, { useMemo } from "react"; import { Tooltip } from "@webiny/ui/Tooltip"; -import { useFolders } from "~/hooks/useFolders"; +import { useGetFolderLevelPermission, useListFolders } from "~/features"; import { CreateButton } from "./ButtonCreate"; import { Empty } from "./Empty"; import { Loader } from "./Loader"; @@ -29,7 +29,10 @@ export const FolderTree = ({ onFolderClick, rootFolderLabel }: FolderTreeProps) => { - const { folders, folderLevelPermissions: flp } = useFolders(); + const { loading, folders } = useListFolders(); + const { getFolderLevelPermission: canManageStructure } = + useGetFolderLevelPermission("canManageStructure"); + const localFolders = useMemo(() => { if (!folders) { return []; @@ -44,13 +47,13 @@ export const FolderTree = ({ }, [folders]); const renderList = () => { - if (!folders) { + if (loading.INIT || loading.LIST) { return ; } let createButton = null; if (enableCreate) { - const canCreate = flp.canManageStructure(focusedFolderId!); + const canCreate = canManageStructure(focusedFolderId!); createButton = ; diff --git a/packages/app-aco/src/contexts/FoldersApi/FoldersApiProvider.tsx b/packages/app-aco/src/contexts/FoldersApi/FoldersApiProvider.tsx deleted file mode 100644 index 379afcbde0a..00000000000 --- a/packages/app-aco/src/contexts/FoldersApi/FoldersApiProvider.tsx +++ /dev/null @@ -1,350 +0,0 @@ -import React, { ReactNode, useEffect, useRef, useState } from "react"; -import { useApolloClient } from "@apollo/react-hooks"; -import { - CREATE_FOLDER, - DELETE_FOLDER, - GET_FOLDER, - LIST_FOLDERS, - UPDATE_FOLDER -} from "~/graphql/folders.gql"; - -import { - CreateFolderResponse, - CreateFolderVariables, - DeleteFolderResponse, - DeleteFolderVariables, - FolderItem, - GetFolderQueryVariables, - GetFolderResponse, - ListFoldersQueryVariables, - ListFoldersResponse, - UpdateFolderResponse, - UpdateFolderVariables -} from "~/types"; -import { ROOT_FOLDER } from "~/constants"; - -interface OffCacheUpdate { - (): void; -} - -export interface OnCacheUpdate { - (folders: FolderItem[]): void; -} - -export interface FoldersApiContext { - listFolders: ( - type: string, - options?: Partial<{ invalidateCache: boolean }> - ) => Promise; - getFolder: (type: string, id: string) => Promise; - createFolder: (type: string, folder: Omit) => Promise; - updateFolder: ( - type: string, - folder: Omit< - FolderItem, - | "type" - | "canManagePermissions" - | "canManageStructure" - | "canManageContent" - | "hasNonInheritedPermissions" - | "createdOn" - | "createdBy" - | "savedOn" - | "savedBy" - | "modifiedOn" - | "modifiedBy" - > - ) => Promise; - - deleteFolder(type: string, id: string): Promise; - - invalidateCache(folderType: string): FoldersApiContext; - - getDescendantFolders(type: string, id?: string): FolderItem[]; - - onFoldersChanged(type: string, cb: OnCacheUpdate): OffCacheUpdate; -} - -export const FoldersApiContext = React.createContext(undefined); - -interface Props { - children: ReactNode; -} - -const rootFolder: FolderItem = { - id: ROOT_FOLDER, - title: "Home", - permissions: [], - parentId: "0", - slug: "", - createdOn: "", - createdBy: { - id: "", - displayName: "", - type: "" - }, - hasNonInheritedPermissions: false, - canManagePermissions: true, - canManageStructure: true, - canManageContent: true, - savedOn: "", - savedBy: { - id: "", - displayName: "", - type: "" - }, - modifiedOn: null, - modifiedBy: null, - type: "$ROOT" -}; - -interface FoldersByType { - [type: string]: FolderItem[]; -} - -export const FoldersApiProvider = ({ children }: Props) => { - const client = useApolloClient(); - const folderObservers = useRef(new Map>()); - const [cache, setCache] = useState({}); - - useEffect(() => { - folderObservers.current.forEach((observers, type) => { - observers.forEach(observer => observer(cache[type])); - }); - }, [cache]); - - useEffect(() => { - return () => { - folderObservers.current.clear(); - }; - }, []); - - const context: FoldersApiContext = { - onFoldersChanged: (type, cb) => { - if (!folderObservers.current.has(type)) { - folderObservers.current.set(type, new Set()); - } - - folderObservers.current.get(type)!.add(cb); - return () => { - folderObservers.current.get(type)?.delete(cb); - }; - }, - invalidateCache: folderType => { - setCache(cache => { - const cacheClone = structuredClone(cache); - delete cacheClone[folderType]; - return cacheClone; - }); - return context; - }, - async listFolders(type, options) { - const invalidateCache = options?.invalidateCache === true; - if (cache[type] && !invalidateCache) { - return cache[type]; - } - - const { data: response } = await client.query< - ListFoldersResponse, - ListFoldersQueryVariables - >({ - query: LIST_FOLDERS, - variables: { - type, - limit: 10000 - }, - fetchPolicy: "network-only" - }); - - if (!response) { - throw new Error("Network error while listing folders."); - } - - const { data, error } = response.aco.listFolders; - - if (!data) { - throw new Error(error?.message || "Could not fetch folders"); - } - - const foldersWithRoot = [rootFolder, ...(data || [])]; - - setCache(cache => ({ - ...cache, - [type]: foldersWithRoot - })); - - return foldersWithRoot; - }, - - async getFolder(type, id) { - if (!id) { - throw new Error("Folder `id` is mandatory"); - } - - const folder = cache[type]?.find(folder => folder.id === id); - if (folder) { - return folder; - } - - const { data: response } = await client.query< - GetFolderResponse, - GetFolderQueryVariables - >({ - query: GET_FOLDER, - variables: { id } - }); - - if (!response) { - throw new Error("Network error while fetch folder."); - } - - const { data, error } = response.aco.getFolder; - - if (!data) { - throw new Error(error?.message || `Could not fetch folder with id: ${id}`); - } - - return data; - }, - - async createFolder(type, folder) { - const { data: response } = await client.mutate< - CreateFolderResponse, - CreateFolderVariables - >({ - mutation: CREATE_FOLDER, - variables: { - data: { - ...folder, - type - } - } - }); - - if (!response) { - throw new Error("Network error while creating folder."); - } - - const { data, error } = response.aco.createFolder; - - if (!data) { - throw new Error(error?.message || "Could not create folder"); - } - - setCache(cache => ({ - ...cache, - [type]: [...cache[type], data] - })); - - return data; - }, - - async updateFolder(type, folder) { - const { id, title, slug, permissions, parentId } = folder; - - const { data: response } = await client.mutate< - UpdateFolderResponse, - UpdateFolderVariables - >({ - mutation: UPDATE_FOLDER, - variables: { - id, - data: { - title, - slug, - permissions, - parentId - } - } - }); - - if (!response) { - throw new Error("Network error while updating folder."); - } - - const { data, error } = response.aco.updateFolder; - - if (!data) { - throw new Error(error?.message || "Could not update folder"); - } - - const folderIndex = cache[type]?.findIndex(f => f.id === id); - if (folderIndex > -1) { - setCache(cache => ({ - ...cache, - [type]: [ - ...cache[type].slice(0, folderIndex), - { - ...cache[type][folderIndex], - ...data - }, - ...cache[type].slice(folderIndex + 1) - ] - })); - } - - return data; - }, - - async deleteFolder(type, id) { - const { data: response } = await client.mutate< - DeleteFolderResponse, - DeleteFolderVariables - >({ - mutation: DELETE_FOLDER, - variables: { - id - } - }); - - if (!response) { - throw new Error("Network error while deleting folder"); - } - - const { data, error } = response.aco.deleteFolder; - - if (!data) { - throw new Error(error?.message || "Could not delete folder"); - } - - setCache(cache => ({ - ...cache, - [type]: cache[type].filter(f => f.id !== id) - })); - - return true; - }, - - getDescendantFolders(type, id) { - const currentFolders = cache[type]; - - if (!id || id === ROOT_FOLDER || !currentFolders?.length) { - return []; - } - - const folderMap = new Map(currentFolders.map(folder => [folder.id, folder])); - const result: FolderItem[] = []; - - const findChildren = (folderId: string) => { - const folder = folderMap.get(folderId); - if (!folder) { - return; - } - - result.push(folder); - - currentFolders.forEach(child => { - if (child.parentId === folder.id) { - findChildren(child.id); - } - }); - }; - - findChildren(id); - - return result; - } - }; - - return {children}; -}; diff --git a/packages/app-aco/src/contexts/FoldersApi/index.ts b/packages/app-aco/src/contexts/FoldersApi/index.ts deleted file mode 100644 index 69c1b07b6b8..00000000000 --- a/packages/app-aco/src/contexts/FoldersApi/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export * from "./FoldersApiProvider"; -export * from "./useFoldersApi"; diff --git a/packages/app-aco/src/contexts/FoldersApi/useFoldersApi.ts b/packages/app-aco/src/contexts/FoldersApi/useFoldersApi.ts deleted file mode 100644 index d0ccc58a998..00000000000 --- a/packages/app-aco/src/contexts/FoldersApi/useFoldersApi.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { useContext } from "react"; -import { FoldersApiContext } from "./FoldersApiProvider"; - -export function useFoldersApi() { - const context = useContext(FoldersApiContext); - if (!context) { - throw new Error(`Missing "FoldersApiProvider" in the component hierarchy!`); - } - - return context; -} diff --git a/packages/app-aco/src/contexts/acoList.tsx b/packages/app-aco/src/contexts/acoList.tsx index b598807097f..53f4709a91b 100644 --- a/packages/app-aco/src/contexts/acoList.tsx +++ b/packages/app-aco/src/contexts/acoList.tsx @@ -12,6 +12,7 @@ import { SearchRecordItem } from "~/types"; import { useAcoApp, useNavigateFolder } from "~/hooks"; +import { useGetDescendantFolders, useListFolders } from "~/features"; import { FoldersContext } from "~/contexts/folders"; import { SearchRecordsContext } from "~/contexts/records"; import { sortTableItems, validateOrGetDefaultDbSort } from "~/sorting"; @@ -120,6 +121,8 @@ export const AcoListProvider = ({ children, ...props }: AcoListProviderProps) => const { identity } = useSecurity(); const { currentFolderId } = useNavigateFolder(); const { folderIdPath, folderIdInPath } = useAcoApp(); + const { folders: originalFolders, loading: foldersLoading } = useListFolders(); + const { getDescendantFolders } = useGetDescendantFolders(); const folderContext = useContext(FoldersContext); const searchContext = useContext(SearchRecordsContext); @@ -132,12 +135,6 @@ export const AcoListProvider = ({ children, ...props }: AcoListProviderProps) => const [listTitle, setListTitle] = useStateIfMounted(undefined); const [state, setState] = useStateIfMounted>(initializeAcoListState()); - const { - folders: originalFolders, - loading: foldersLoading, - listFolders, - getDescendantFolders - } = folderContext; const { records: originalRecords, loading: recordsLoading, listRecords, meta } = searchContext; /** @@ -155,10 +152,6 @@ export const AcoListProvider = ({ children, ...props }: AcoListProviderProps) => return; } - if (!originalFolders) { - listFolders(); - } - setState(state => { return { ...state, @@ -253,6 +246,10 @@ export const AcoListProvider = ({ children, ...props }: AcoListProviderProps) => // Initialize an empty object let where = {}; + if (!state.folderId) { + return where; + } + // Check if the current folder ID is not the ROOT_FOLDER folder if (state.folderId !== ROOT_FOLDER) { // Get descendant folder IDs of the current folder @@ -331,7 +328,7 @@ export const AcoListProvider = ({ children, ...props }: AcoListProviderProps) => const { hasMoreItems } = meta; // Retrieve all descendant folders of the current folderId - const folderWithChildren = getDescendantFolders(folderId); + const folderWithChildren = folderId ? getDescendantFolders(folderId) : []; // Compute the lengths of various arrays for later comparisons const foldersLength = folders.length; diff --git a/packages/app-aco/src/contexts/folders.tsx b/packages/app-aco/src/contexts/folders.tsx index a6b355d919c..6f243c9949c 100644 --- a/packages/app-aco/src/contexts/folders.tsx +++ b/packages/app-aco/src/contexts/folders.tsx @@ -1,38 +1,8 @@ -import React, { ReactNode, useContext, useEffect, useMemo } from "react"; -import { useWcp } from "@webiny/app-wcp/hooks/useWcp"; -import { useStateIfMounted } from "@webiny/app-admin"; -import { dataLoader, loadingHandler } from "~/handlers"; -import { FolderItem, Loading, LoadingActions } from "~/types"; +import React, { ReactNode, useContext, useMemo } from "react"; import { AcoAppContext } from "~/contexts/app"; -import { useFoldersApi } from "~/hooks"; -import { ROOT_FOLDER } from "~/constants"; - -export interface FoldersContextFolderLevelPermissions { - canManageStructure(folderId: string): boolean; - - canManagePermissions(folderId: string): boolean; - - canManageContent(folderId: string): boolean; -} interface FoldersContext { - folders?: FolderItem[] | null; - loading: Loading; - listFolders: () => Promise; - getFolder: (id: string) => Promise; - createFolder: (folder: Omit) => Promise; - updateFolder: ( - folder: Omit, - options?: Partial<{ - refetchFoldersList: boolean; - }> - ) => Promise; - - deleteFolder(folder: Pick): Promise; - - getDescendantFolders(id?: string): FolderItem[]; - - folderLevelPermissions: FoldersContextFolderLevelPermissions; + type?: string | null; } export const FoldersContext = React.createContext(undefined); @@ -42,132 +12,22 @@ interface Props { children: ReactNode; } -const defaultLoading: Record = { - INIT: true, - LIST: false, - LIST_MORE: false, - GET: false, - MOVE: false, - CREATE: false, - UPDATE: false, - DELETE: false -}; - export const FoldersProvider = ({ children, ...props }: Props) => { const appContext = useContext(AcoAppContext); - const [folders, setFolders] = useStateIfMounted(null); - const [loading, setLoading] = useStateIfMounted>(defaultLoading); - const foldersApi = useFoldersApi(); - const { canUseFolderLevelPermissions } = useWcp(); const app = appContext ? appContext.app : undefined; const type = props.type ?? app?.id; + if (!type) { throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); } - useEffect(() => { - return foldersApi.onFoldersChanged(type, folders => { - setFolders(folders); - }); - }, []); - - const folderLevelPermissions: FoldersContextFolderLevelPermissions = useMemo(() => { - const createCanManage = - (callback: (folder: FolderItem) => boolean) => (folderId: string) => { - if (!canUseFolderLevelPermissions() || folderId === ROOT_FOLDER) { - return true; - } - - const folder = folders?.find(folder => folder.id === folderId); - if (!folder) { - return false; - } - - return callback(folder); - }; - - return { - canManageStructure: createCanManage(folder => folder.canManageStructure), - canManagePermissions: createCanManage(folder => folder.canManagePermissions), - canManageContent: createCanManage(folder => folder.canManageContent) - }; - }, [folders]); - const context = useMemo(() => { return { - folders, - loading, - async listFolders() { - const folders = await dataLoader(loadingHandler("LIST", setLoading), () => - foldersApi.listFolders(type) - ); - - setFolders(() => folders); - - setLoading(prev => ({ - ...prev, - INIT: false - })); - - return folders; - }, - - async getFolder(id) { - if (!id) { - throw new Error("Folder `id` is mandatory"); - } - - return await dataLoader(loadingHandler("GET", setLoading), () => - foldersApi.getFolder(type, id) - ); - }, - - async createFolder(folder) { - return await dataLoader(loadingHandler("CREATE", setLoading), () => - foldersApi.createFolder(type, folder) - ); - }, - - async updateFolder(folder, options) { - const { id, title, slug, permissions, parentId } = folder; - - // We must omit all inherited permissions. - const filteredPermissions = permissions.filter(p => !p.inheritedFrom); - - return await dataLoader(loadingHandler("UPDATE", setLoading), async () => { - const response = await foldersApi.updateFolder(type, { - id, - title, - slug, - permissions: filteredPermissions, - parentId - }); - - if (options?.refetchFoldersList) { - foldersApi.listFolders(type, { invalidateCache: true }).then(setFolders); - } - - return response; - }); - }, - - async deleteFolder(folder) { - const { id } = folder; - - return await dataLoader(loadingHandler("DELETE", setLoading), () => - foldersApi.deleteFolder(type, id) - ); - }, - - getDescendantFolders(id) { - return foldersApi.getDescendantFolders(type, id); - }, - - folderLevelPermissions + type }; - }, [folders, loading, setLoading, setFolders]); + }, [type]); return {children}; }; diff --git a/packages/app-aco/src/dialogs/useCreateDialog.tsx b/packages/app-aco/src/dialogs/useCreateDialog.tsx index 1920abe3593..c3194deabb2 100644 --- a/packages/app-aco/src/dialogs/useCreateDialog.tsx +++ b/packages/app-aco/src/dialogs/useCreateDialog.tsx @@ -10,7 +10,7 @@ import { validation } from "@webiny/validation"; import { FolderTree } from "~/components"; import { useDialogs } from "@webiny/app-admin"; import { DialogFoldersContainer } from "~/dialogs/styled"; -import { useFolders } from "~/hooks"; +import { useCreateFolder } from "~/features"; import { ROOT_FOLDER } from "~/constants"; import { FolderItem } from "~/types"; @@ -81,7 +81,7 @@ const FormComponent = ({ currentParentId = null }: FormComponentProps) => { export const useCreateDialog = (): UseCreateDialogResponse => { const dialogs = useDialogs(); - const { createFolder } = useFolders(); + const { createFolder } = useCreateFolder(); const { showSnackbar } = useSnackbar(); const onAccept = useCallback(async (data: FolderItem) => { diff --git a/packages/app-aco/src/dialogs/useDeleteDialog.tsx b/packages/app-aco/src/dialogs/useDeleteDialog.tsx index 0f30ea82867..54f8f643f6e 100644 --- a/packages/app-aco/src/dialogs/useDeleteDialog.tsx +++ b/packages/app-aco/src/dialogs/useDeleteDialog.tsx @@ -1,7 +1,7 @@ import { useSnackbar } from "@webiny/app-admin"; import { useDialogs } from "@webiny/app-admin"; -import { useFolders } from "~/hooks"; +import { useDeleteFolder } from "~/features"; import { FolderItem } from "~/types"; import { useCallback } from "react"; @@ -15,18 +15,13 @@ interface UseDeleteDialogResponse { export const useDeleteDialog = (): UseDeleteDialogResponse => { const dialogs = useDialogs(); - const { deleteFolder } = useFolders(); + const { deleteFolder } = useDeleteFolder(); const { showSnackbar } = useSnackbar(); const onAccept = useCallback(async (folder: FolderItem) => { try { - const result = await deleteFolder(folder); - - if (result) { - showSnackbar(`The folder "${folder.title}" was deleted successfully.`); - } else { - throw new Error(`Error while deleting folder "${folder.title}"!`); - } + await deleteFolder(folder); + showSnackbar(`The folder "${folder.title}" was deleted successfully.`); } catch (error) { showSnackbar(error.message); } diff --git a/packages/app-aco/src/dialogs/useEditDialog.tsx b/packages/app-aco/src/dialogs/useEditDialog.tsx index 39ee1e9367a..e9027d28832 100644 --- a/packages/app-aco/src/dialogs/useEditDialog.tsx +++ b/packages/app-aco/src/dialogs/useEditDialog.tsx @@ -10,7 +10,7 @@ import { FolderTree } from "~/components"; import { ROOT_FOLDER } from "~/constants"; import { useDialogs } from "@webiny/app-admin"; import { DialogFoldersContainer } from "~/dialogs/styled"; -import { useFolders } from "~/hooks"; +import { useUpdateFolder } from "~/features"; import { FolderItem } from "~/types"; interface ShowDialogParams { @@ -72,21 +72,16 @@ const FormComponent = ({ folder }: FormComponentProps) => { export const useEditDialog = (): UseEditDialogResponse => { const dialog = useDialogs(); - const { updateFolder } = useFolders(); + const { updateFolder } = useUpdateFolder(); const { showSnackbar } = useSnackbar(); const onAccept = useCallback(async (folder: FolderItem, data: GenericFormData) => { try { - const result = await updateFolder({ + await updateFolder({ ...folder, ...data }); - - if (result) { - showSnackbar(`The folder "${result.title}" was updated successfully!`); - } else { - throw new Error(`Error while updating folder "${folder.title}"!`); - } + showSnackbar(`The folder "${data.title}" was updated successfully!`); } catch (error) { showSnackbar(error.message); } diff --git a/packages/app-aco/src/dialogs/useSetPermissionsDialog.tsx b/packages/app-aco/src/dialogs/useSetPermissionsDialog.tsx index 352414caa24..9505cdd4139 100644 --- a/packages/app-aco/src/dialogs/useSetPermissionsDialog.tsx +++ b/packages/app-aco/src/dialogs/useSetPermissionsDialog.tsx @@ -9,7 +9,7 @@ import { UsersTeamsSelection } from "./DialogSetPermissions/UsersTeamsSelection" import { LIST_FOLDER_LEVEL_PERMISSIONS_TARGETS } from "./DialogSetPermissions/graphql"; import { useDialogs } from "@webiny/app-admin"; -import { useFolders } from "~/hooks"; +import { useUpdateFolder } from "~/features"; import { FolderItem, FolderLevelPermissionsTarget, FolderPermission } from "~/types"; interface ShowDialogParams { @@ -115,14 +115,14 @@ const FormComponent = ({ folder }: FormComponentProps) => { export const useSetPermissionsDialog = (): UseSetPermissionsDialogResponse => { const dialogs = useDialogs(); - const { updateFolder } = useFolders(); + const { updateFolder } = useUpdateFolder(); const { showSnackbar } = useSnackbar(); const onAccept = useCallback(async (folder: FolderItem, data: Partial) => { const updateData = { ...folder, ...data }; try { - await updateFolder(updateData, { refetchFoldersList: true }); + await updateFolder(updateData); showSnackbar("Folder permissions updated successfully!"); } catch (error) { showSnackbar(error.message); diff --git a/packages/app-aco/src/features/folder/Folder.ts b/packages/app-aco/src/features/folder/Folder.ts new file mode 100644 index 00000000000..23abccd015d --- /dev/null +++ b/packages/app-aco/src/features/folder/Folder.ts @@ -0,0 +1,62 @@ +import { CmsIdentity, FolderPermission } from "~/types"; + +export interface FolderData { + id?: string; + title: string; + slug: string; + type: string; + parentId: string | null; + permissions: FolderPermission[]; + hasNonInheritedPermissions?: boolean; + canManagePermissions?: boolean; + canManageStructure?: boolean; + canManageContent?: boolean; + createdBy?: CmsIdentity; + createdOn?: string; + savedBy?: CmsIdentity; + savedOn?: string; + modifiedBy?: CmsIdentity | null; + modifiedOn?: string | null; +} + +export class Folder { + public id: string; + public title: string; + public slug: string; + public type: string; + public parentId: string | null; + public permissions: FolderPermission[]; + public hasNonInheritedPermissions?: boolean; + public canManagePermissions?: boolean; + public canManageStructure?: boolean; + public canManageContent?: boolean; + public createdBy?: CmsIdentity; + public createdOn?: string; + public savedBy?: CmsIdentity; + public savedOn?: string; + public modifiedBy?: CmsIdentity | null; + public modifiedOn?: string | null; + + protected constructor(folder: FolderData) { + this.id = folder.id ?? ""; + this.title = folder.title; + this.slug = folder.slug; + this.type = folder.type; + this.parentId = folder.parentId; + this.permissions = folder.permissions; + this.hasNonInheritedPermissions = folder.hasNonInheritedPermissions; + this.canManagePermissions = folder.canManagePermissions; + this.canManageStructure = folder.canManageStructure; + this.canManageContent = folder.canManageContent; + this.createdBy = folder.createdBy; + this.createdOn = folder.createdOn; + this.savedBy = folder.savedBy; + this.savedOn = folder.savedOn; + this.modifiedBy = folder.modifiedBy; + this.modifiedOn = folder.modifiedOn; + } + + static create(folder: FolderData) { + return new Folder(folder); + } +} diff --git a/packages/app-aco/src/features/folder/cache/FoldersCacheFactory.ts b/packages/app-aco/src/features/folder/cache/FoldersCacheFactory.ts new file mode 100644 index 00000000000..581b8e67790 --- /dev/null +++ b/packages/app-aco/src/features/folder/cache/FoldersCacheFactory.ts @@ -0,0 +1,22 @@ +import { Folder } from "../Folder"; +import { ListCache } from "~/features/folder/cache/ListCache"; + +export class FoldersCacheFactory { + private cache: Map> = new Map(); + + getCache(namespace: string) { + const cacheKey = this.getCacheKey(namespace); + + if (!this.cache.has(cacheKey)) { + this.cache.set(cacheKey, new ListCache()); + } + + return this.cache.get(cacheKey) as ListCache; + } + + private getCacheKey(namespace: string) { + return namespace; + } +} + +export const folderCacheFactory = new FoldersCacheFactory(); diff --git a/packages/app-aco/src/features/folder/cache/ListCache.ts b/packages/app-aco/src/features/folder/cache/ListCache.ts new file mode 100644 index 00000000000..b45475794cb --- /dev/null +++ b/packages/app-aco/src/features/folder/cache/ListCache.ts @@ -0,0 +1,74 @@ +import { makeAutoObservable, runInAction, toJS } from "mobx"; + +export type Constructor = new (...args: any[]) => T; + +export interface IListCachePredicate { + (item: T): boolean; +} + +export interface IListCacheItemUpdater { + (item: T): T; +} + +export interface IListCache { + count(): number; + clear(): void; + hasItems(): boolean; + getItems(): T[]; + getItem(predicate: IListCachePredicate): T | undefined; + addItems(items: T[]): void; + updateItems(updater: IListCacheItemUpdater): void; + removeItems(predicate: IListCachePredicate): void; +} + +export class ListCache implements IListCache { + private state: T[]; + + constructor() { + this.state = []; + + makeAutoObservable(this); + } + + count() { + return this.state.length; + } + + clear() { + runInAction(() => { + this.state = []; + }); + } + + hasItems() { + return this.state.length > 0; + } + + getItems() { + return [...this.state.map(item => toJS(item))]; + } + + getItem(predicate: IListCachePredicate): T | undefined { + const item = this.state.find(item => predicate(item)); + + return item ? toJS(item) : undefined; + } + + addItems(items: T[]) { + runInAction(() => { + this.state = [...this.state, ...items]; + }); + } + + updateItems(updater: IListCacheItemUpdater) { + runInAction(() => { + this.state = [...this.state.map(item => updater(item))]; + }); + } + + removeItems(predicate: IListCachePredicate) { + runInAction(() => { + this.state = this.state.filter(item => !predicate(item)); + }); + } +} diff --git a/packages/app-aco/src/features/folder/cache/index.ts b/packages/app-aco/src/features/folder/cache/index.ts new file mode 100644 index 00000000000..ecb9ffa44c9 --- /dev/null +++ b/packages/app-aco/src/features/folder/cache/index.ts @@ -0,0 +1,2 @@ +export * from "./FoldersCacheFactory"; +export * from "./ListCache"; diff --git a/packages/app-aco/src/features/folder/createFolder/CreateFolder.test.ts b/packages/app-aco/src/features/folder/createFolder/CreateFolder.test.ts new file mode 100644 index 00000000000..ecdcdced489 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/CreateFolder.test.ts @@ -0,0 +1,44 @@ +import { CreateFolder } from "./CreateFolder"; +import { folderCacheFactory } from "../cache/FoldersCacheFactory"; + +describe("CreateFolder", () => { + const type = "abc"; + const gateway = { + execute: jest.fn().mockResolvedValue({ + id: "any-folder-id", + title: "New Folder", + slug: "new-folder", + type + }) + }; + const foldersCache = folderCacheFactory.getCache(type); + + beforeEach(() => { + foldersCache.clear(); + }); + + it("should be able to create a new folder", async () => { + const createFolder = CreateFolder.instance(type, gateway); + + expect(foldersCache.hasItems()).toBeFalse(); + + await createFolder.execute({ + title: "New Folder", + slug: "new-folder", + parentId: null, + permissions: [], + type + }); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + expect(foldersCache.hasItems()).toBeTrue(); + + const item = foldersCache.getItem(folder => folder.slug === "new-folder"); + + expect(item).toBeDefined(); + expect(item?.id).toEqual("any-folder-id"); + expect(item?.type).toEqual(type); + expect(item?.title).toEqual("New Folder"); + expect(item?.slug).toEqual("new-folder"); + }); +}); diff --git a/packages/app-aco/src/features/folder/createFolder/CreateFolder.ts b/packages/app-aco/src/features/folder/createFolder/CreateFolder.ts new file mode 100644 index 00000000000..1182098d6d4 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/CreateFolder.ts @@ -0,0 +1,17 @@ +import { loadingRepositoryFactory } from "@webiny/app-utils"; +import { ICreateFolderUseCase } from "./ICreateFolderUseCase"; +import { ICreateFolderGateway } from "./ICreateFolderGateway"; +import { CreateFolderRepository } from "./CreateFolderRepository"; +import { CreateFolderUseCase } from "./CreateFolderUseCase"; +import { CreateFolderUseCaseWithLoading } from "./CreateFolderUseCaseWithLoading"; +import { folderCacheFactory } from "../cache"; + +export class CreateFolder { + public static instance(type: string, gateway: ICreateFolderGateway): ICreateFolderUseCase { + const foldersCache = folderCacheFactory.getCache(type); + const loadingRepository = loadingRepositoryFactory.getRepository(type); + const repository = new CreateFolderRepository(foldersCache, gateway, type); + const useCase = new CreateFolderUseCase(repository); + return new CreateFolderUseCaseWithLoading(loadingRepository, useCase); + } +} diff --git a/packages/app-aco/src/features/folder/createFolder/CreateFolderGqlGateway.ts b/packages/app-aco/src/features/folder/createFolder/CreateFolderGqlGateway.ts new file mode 100644 index 00000000000..4db71c88118 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/CreateFolderGqlGateway.ts @@ -0,0 +1,110 @@ +import ApolloClient from "apollo-client"; +import gql from "graphql-tag"; +import { ICreateFolderGateway } from "./ICreateFolderGateway"; +import { FolderDto } from "./FolderDto"; +import { AcoError, FolderItem } from "~/types"; + +export interface CreateFolderResponse { + aco: { + createFolder: { + data: FolderItem; + error: AcoError | null; + }; + }; +} + +export interface CreateFolderVariables { + data: Omit< + FolderItem, + | "id" + | "createdOn" + | "createdBy" + | "savedOn" + | "savedBy" + | "modifiedOn" + | "modifiedBy" + | "hasNonInheritedPermissions" + | "canManageContent" + | "canManagePermissions" + | "canManageStructure" + >; +} + +export const CREATE_FOLDER = gql` + mutation CreateFolder($data: FolderCreateInput!) { + aco { + createFolder(data: $data) { + data { + id + title + slug + permissions { + target + level + inheritedFrom + } + hasNonInheritedPermissions + canManagePermissions + canManageStructure + canManageContent + parentId + type + savedOn + savedBy { + id + displayName + } + createdOn + createdBy { + id + displayName + } + modifiedOn + modifiedBy { + id + displayName + } + } + error { + code + data + message + } + } + } + } +`; + +export class CreateFolderGqlGateway implements ICreateFolderGateway { + private client: ApolloClient; + + constructor(client: ApolloClient) { + this.client = client; + } + + async execute(folder: FolderDto) { + const { data: response } = await this.client.mutate< + CreateFolderResponse, + CreateFolderVariables + >({ + mutation: CREATE_FOLDER, + variables: { + data: { + ...folder + } + } + }); + + if (!response) { + throw new Error("Network error while creating folder."); + } + + const { data, error } = response.aco.createFolder; + + if (!data) { + throw new Error(error?.message || "Could not create folder"); + } + + return data; + } +} diff --git a/packages/app-aco/src/features/folder/createFolder/CreateFolderRepository.ts b/packages/app-aco/src/features/folder/createFolder/CreateFolderRepository.ts new file mode 100644 index 00000000000..1166f4f86b8 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/CreateFolderRepository.ts @@ -0,0 +1,30 @@ +import { ICreateFolderRepository } from "./ICreateFolderRepository"; +import { ListCache } from "../cache"; +import { Folder } from "../Folder"; +import { ICreateFolderGateway } from "./ICreateFolderGateway"; +import { FolderDto } from "./FolderDto"; + +export class CreateFolderRepository implements ICreateFolderRepository { + private cache: ListCache; + private gateway: ICreateFolderGateway; + private readonly type: string; + + constructor(cache: ListCache, gateway: ICreateFolderGateway, type: string) { + this.cache = cache; + this.gateway = gateway; + this.type = type; + } + + async execute(folder: Folder) { + const dto: FolderDto = { + title: folder.title, + slug: folder.slug, + permissions: folder.permissions, + type: this.type, + parentId: folder.parentId + }; + + const result = await this.gateway.execute(dto); + this.cache.addItems([Folder.create(result)]); + } +} diff --git a/packages/app-aco/src/features/folder/createFolder/CreateFolderUseCase.ts b/packages/app-aco/src/features/folder/createFolder/CreateFolderUseCase.ts new file mode 100644 index 00000000000..fb42c3be0e5 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/CreateFolderUseCase.ts @@ -0,0 +1,23 @@ +import { CreateFolderParams, ICreateFolderUseCase } from "./ICreateFolderUseCase"; +import { ICreateFolderRepository } from "./ICreateFolderRepository"; +import { Folder } from "../Folder"; + +export class CreateFolderUseCase implements ICreateFolderUseCase { + private repository: ICreateFolderRepository; + + constructor(repository: ICreateFolderRepository) { + this.repository = repository; + } + + async execute(params: CreateFolderParams) { + await this.repository.execute( + Folder.create({ + title: params.title, + slug: params.slug, + type: params.type, + parentId: params.parentId, + permissions: params.permissions + }) + ); + } +} diff --git a/packages/app-aco/src/features/folder/createFolder/CreateFolderUseCaseWithLoading.ts b/packages/app-aco/src/features/folder/createFolder/CreateFolderUseCaseWithLoading.ts new file mode 100644 index 00000000000..75578757903 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/CreateFolderUseCaseWithLoading.ts @@ -0,0 +1,20 @@ +import { ILoadingRepository } from "@webiny/app-utils"; +import { CreateFolderParams, ICreateFolderUseCase } from "./ICreateFolderUseCase"; +import { LoadingActionsEnum } from "~/types"; + +export class CreateFolderUseCaseWithLoading implements ICreateFolderUseCase { + private loadingRepository: ILoadingRepository; + private useCase: ICreateFolderUseCase; + + constructor(loadingRepository: ILoadingRepository, useCase: ICreateFolderUseCase) { + this.loadingRepository = loadingRepository; + this.useCase = useCase; + } + + async execute(params: CreateFolderParams) { + await this.loadingRepository.runCallBack( + this.useCase.execute(params), + LoadingActionsEnum.create + ); + } +} diff --git a/packages/app-aco/src/features/folder/createFolder/FolderDto.ts b/packages/app-aco/src/features/folder/createFolder/FolderDto.ts new file mode 100644 index 00000000000..09733f3ede8 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/FolderDto.ts @@ -0,0 +1,9 @@ +import { FolderPermission } from "~/types"; + +export interface FolderDto { + title: string; + slug: string; + permissions: FolderPermission[]; + type: string; + parentId: string | null; +} diff --git a/packages/app-aco/src/features/folder/createFolder/FolderGqlDto.ts b/packages/app-aco/src/features/folder/createFolder/FolderGqlDto.ts new file mode 100644 index 00000000000..0d2ff62504c --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/FolderGqlDto.ts @@ -0,0 +1,20 @@ +import { CmsIdentity, FolderPermission } from "~/types"; + +export interface FolderGqlDto { + id: string; + title: string; + slug: string; + permissions: FolderPermission[]; + hasNonInheritedPermissions: boolean; + canManagePermissions: boolean; + canManageStructure: boolean; + canManageContent: boolean; + type: string; + parentId: string | null; + createdBy: CmsIdentity; + createdOn: string; + savedBy: CmsIdentity; + savedOn: string; + modifiedBy: CmsIdentity | null; + modifiedOn: string | null; +} diff --git a/packages/app-aco/src/features/folder/createFolder/ICreateFolderGateway.ts b/packages/app-aco/src/features/folder/createFolder/ICreateFolderGateway.ts new file mode 100644 index 00000000000..ba10ce21f07 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/ICreateFolderGateway.ts @@ -0,0 +1,6 @@ +import { FolderDto } from "./FolderDto"; +import { FolderGqlDto } from "./FolderGqlDto"; + +export interface ICreateFolderGateway { + execute: (folderDto: FolderDto) => Promise; +} diff --git a/packages/app-aco/src/features/folder/createFolder/ICreateFolderRepository.ts b/packages/app-aco/src/features/folder/createFolder/ICreateFolderRepository.ts new file mode 100644 index 00000000000..d4cf5b909b9 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/ICreateFolderRepository.ts @@ -0,0 +1,5 @@ +import { Folder } from "../Folder"; + +export interface ICreateFolderRepository { + execute: (folder: Folder) => Promise; +} diff --git a/packages/app-aco/src/features/folder/createFolder/ICreateFolderUseCase.ts b/packages/app-aco/src/features/folder/createFolder/ICreateFolderUseCase.ts new file mode 100644 index 00000000000..45517a3200e --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/ICreateFolderUseCase.ts @@ -0,0 +1,13 @@ +import { FolderPermission } from "~/types"; + +export interface CreateFolderParams { + title: string; + slug: string; + type: string; + parentId: string | null; + permissions: FolderPermission[]; +} + +export interface ICreateFolderUseCase { + execute: (params: CreateFolderParams) => Promise; +} diff --git a/packages/app-aco/src/features/folder/createFolder/index.ts b/packages/app-aco/src/features/folder/createFolder/index.ts new file mode 100644 index 00000000000..26134b3d69d --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/index.ts @@ -0,0 +1 @@ +export * from "./useCreateFolder"; diff --git a/packages/app-aco/src/features/folder/createFolder/useCreateFolder.ts b/packages/app-aco/src/features/folder/createFolder/useCreateFolder.ts new file mode 100644 index 00000000000..99193b73097 --- /dev/null +++ b/packages/app-aco/src/features/folder/createFolder/useCreateFolder.ts @@ -0,0 +1,35 @@ +import { useCallback, useContext } from "react"; +import { useApolloClient } from "@apollo/react-hooks"; +import { CreateFolderGqlGateway } from "./CreateFolderGqlGateway"; +import { CreateFolderParams } from "./ICreateFolderUseCase"; +import { CreateFolder } from "./CreateFolder"; +import { FoldersContext } from "~/contexts/folders"; + +export const useCreateFolder = () => { + const client = useApolloClient(); + const gateway = new CreateFolderGqlGateway(client); + + const foldersContext = useContext(FoldersContext); + + if (!foldersContext) { + throw new Error("useCreateFolder must be used within a FoldersProvider"); + } + + const { type } = foldersContext; + + if (!type) { + throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); + } + + const createFolder = useCallback( + (params: CreateFolderParams) => { + const instance = CreateFolder.instance(type, gateway); + return instance.execute(params); + }, + [type, gateway] + ); + + return { + createFolder + }; +}; diff --git a/packages/app-aco/src/features/folder/deleteFolder/DeleteFolder.test.ts b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolder.test.ts new file mode 100644 index 00000000000..d587c352579 --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolder.test.ts @@ -0,0 +1,45 @@ +import { DeleteFolder } from "./DeleteFolder"; +import { folderCacheFactory } from "../cache/FoldersCacheFactory"; +import { Folder } from "../Folder"; + +describe("DeleteFolder", () => { + const type = "abc"; + const gateway = { + execute: jest.fn().mockResolvedValue(true) + }; + const foldersCache = folderCacheFactory.getCache(type); + + beforeEach(() => { + foldersCache.clear(); + foldersCache.addItems([ + Folder.create({ + id: "any-folder-id", + title: "New Folder", + slug: "new-folder", + parentId: null, + permissions: [], + type + }) + ]); + }); + + it("should be able to delete a folder", async () => { + const deleteFolder = DeleteFolder.instance(type, gateway); + + expect(foldersCache.hasItems()).toBeTrue(); + const item = foldersCache.getItem(folder => folder.id === "any-folder-id"); + expect(item?.id).toEqual("any-folder-id"); + + await deleteFolder.execute({ + id: "any-folder-id", + title: "New Folder", + slug: "new-folder", + parentId: null, + permissions: [], + type + }); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + expect(foldersCache.hasItems()).toBeFalse(); + }); +}); diff --git a/packages/app-aco/src/features/folder/deleteFolder/DeleteFolder.ts b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolder.ts new file mode 100644 index 00000000000..41aebf02144 --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolder.ts @@ -0,0 +1,17 @@ +import { loadingRepositoryFactory } from "@webiny/app-utils"; +import { IDeleteFolderUseCase } from "./IDeleteFolderUseCase"; +import { DeleteFolderRepository } from "./DeleteFolderRepository"; +import { DeleteFolderUseCase } from "./DeleteFolderUseCase"; +import { DeleteFolderUseCaseWithLoading } from "./DeleteFolderUseCaseWithLoading"; +import { IDeleteFolderGateway } from "./IDeleteFolderGateway"; +import { folderCacheFactory } from "../cache"; + +export class DeleteFolder { + public static instance(type: string, gateway: IDeleteFolderGateway): IDeleteFolderUseCase { + const foldersCache = folderCacheFactory.getCache(type); + const loadingRepository = loadingRepositoryFactory.getRepository(type); + const repository = new DeleteFolderRepository(foldersCache, gateway); + const useCase = new DeleteFolderUseCase(repository); + return new DeleteFolderUseCaseWithLoading(loadingRepository, useCase); + } +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderGqlGateway.ts b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderGqlGateway.ts new file mode 100644 index 00000000000..1de619638e6 --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderGqlGateway.ts @@ -0,0 +1,64 @@ +import ApolloClient from "apollo-client"; +import gql from "graphql-tag"; +import { IDeleteFolderGateway } from "./IDeleteFolderGateway"; +import { AcoError } from "~/types"; + +export interface DeleteFolderVariables { + id: string; +} + +export interface DeleteFolderResponse { + aco: { + deleteFolder: { + data: boolean; + error: AcoError | null; + }; + }; +} + +export const DELETE_FOLDER = gql` + mutation DeleteFolder($id: ID!) { + aco { + deleteFolder(id: $id) { + data + error { + code + data + message + } + } + } + } +`; + +export class DeleteFolderGqlGateway implements IDeleteFolderGateway { + private client: ApolloClient; + + constructor(client: ApolloClient) { + this.client = client; + } + + async execute(id: string) { + const { data: response } = await this.client.mutate< + DeleteFolderResponse, + DeleteFolderVariables + >({ + mutation: DELETE_FOLDER, + variables: { + id + } + }); + + if (!response) { + throw new Error("Network error while deleting folder"); + } + + const { data, error } = response.aco.deleteFolder; + + if (!data) { + throw new Error(error?.message || "Could not delete folder"); + } + + return; + } +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderRepository.ts b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderRepository.ts new file mode 100644 index 00000000000..9192780de14 --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderRepository.ts @@ -0,0 +1,19 @@ +import { IDeleteFolderRepository } from "./IDeleteFolderRepository"; +import { ListCache } from "../cache"; +import { Folder } from "../Folder"; +import { IDeleteFolderGateway } from "./IDeleteFolderGateway"; + +export class DeleteFolderRepository implements IDeleteFolderRepository { + private cache: ListCache; + private gateway: IDeleteFolderGateway; + + constructor(cache: ListCache, gateway: IDeleteFolderGateway) { + this.cache = cache; + this.gateway = gateway; + } + + async execute(folder: Folder) { + await this.gateway.execute(folder.id); + this.cache.removeItems(f => f.id === folder.id); + } +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderUseCase.ts b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderUseCase.ts new file mode 100644 index 00000000000..9dcbfb942ef --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderUseCase.ts @@ -0,0 +1,24 @@ +import { DeleteFolderParams, IDeleteFolderUseCase } from "./IDeleteFolderUseCase"; +import { IDeleteFolderRepository } from "./IDeleteFolderRepository"; +import { Folder } from "../Folder"; + +export class DeleteFolderUseCase implements IDeleteFolderUseCase { + private repository: IDeleteFolderRepository; + + constructor(repository: IDeleteFolderRepository) { + this.repository = repository; + } + + async execute(params: DeleteFolderParams) { + await this.repository.execute( + Folder.create({ + id: params.id, + title: params.title, + slug: params.slug, + type: params.type, + parentId: params.parentId, + permissions: params.permissions + }) + ); + } +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderUseCaseWithLoading.ts b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderUseCaseWithLoading.ts new file mode 100644 index 00000000000..12717658c02 --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/DeleteFolderUseCaseWithLoading.ts @@ -0,0 +1,20 @@ +import { ILoadingRepository } from "@webiny/app-utils"; +import { LoadingActionsEnum } from "~/types"; +import { DeleteFolderParams, IDeleteFolderUseCase } from "./IDeleteFolderUseCase"; + +export class DeleteFolderUseCaseWithLoading implements IDeleteFolderUseCase { + private loadingRepository: ILoadingRepository; + private useCase: IDeleteFolderUseCase; + + constructor(loadingRepository: ILoadingRepository, useCase: IDeleteFolderUseCase) { + this.loadingRepository = loadingRepository; + this.useCase = useCase; + } + + async execute(params: DeleteFolderParams) { + await this.loadingRepository.runCallBack( + this.useCase.execute(params), + LoadingActionsEnum.delete + ); + } +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderGateway.ts b/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderGateway.ts new file mode 100644 index 00000000000..852a065ec5e --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderGateway.ts @@ -0,0 +1,3 @@ +export interface IDeleteFolderGateway { + execute: (id: string) => Promise; +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderRepository.ts b/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderRepository.ts new file mode 100644 index 00000000000..d771713a47c --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderRepository.ts @@ -0,0 +1,5 @@ +import { Folder } from "../Folder"; + +export interface IDeleteFolderRepository { + execute: (folder: Folder) => Promise; +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderUseCase.ts b/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderUseCase.ts new file mode 100644 index 00000000000..a257dd5b0e0 --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/IDeleteFolderUseCase.ts @@ -0,0 +1,14 @@ +import { FolderPermission } from "~/types"; + +export interface DeleteFolderParams { + id: string; + title: string; + slug: string; + type: string; + parentId: string | null; + permissions: FolderPermission[]; +} + +export interface IDeleteFolderUseCase { + execute: (params: DeleteFolderParams) => Promise; +} diff --git a/packages/app-aco/src/features/folder/deleteFolder/index.ts b/packages/app-aco/src/features/folder/deleteFolder/index.ts new file mode 100644 index 00000000000..87fcddfbf54 --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/index.ts @@ -0,0 +1 @@ +export * from "./useDeleteFolder"; diff --git a/packages/app-aco/src/features/folder/deleteFolder/useDeleteFolder.ts b/packages/app-aco/src/features/folder/deleteFolder/useDeleteFolder.ts new file mode 100644 index 00000000000..e0344e7be1f --- /dev/null +++ b/packages/app-aco/src/features/folder/deleteFolder/useDeleteFolder.ts @@ -0,0 +1,35 @@ +import { useCallback, useContext } from "react"; +import { useApolloClient } from "@apollo/react-hooks"; +import { DeleteFolderGqlGateway } from "./DeleteFolderGqlGateway"; +import { DeleteFolderParams } from "./IDeleteFolderUseCase"; +import { DeleteFolder } from "./DeleteFolder"; +import { FoldersContext } from "~/contexts/folders"; + +export const useDeleteFolder = () => { + const client = useApolloClient(); + const gateway = new DeleteFolderGqlGateway(client); + + const foldersContext = useContext(FoldersContext); + + if (!foldersContext) { + throw new Error("useDeleteFolder must be used within a FoldersProvider"); + } + + const { type } = foldersContext; + + if (!type) { + throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); + } + + const deleteFolder = useCallback( + (params: DeleteFolderParams) => { + const instance = DeleteFolder.instance(type, gateway); + return instance.execute(params); + }, + [type, gateway] + ); + + return { + deleteFolder + }; +}; diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/FolderDto.ts b/packages/app-aco/src/features/folder/getDescendantFolders/FolderDto.ts new file mode 100644 index 00000000000..766a6f65f29 --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/FolderDto.ts @@ -0,0 +1,10 @@ +import { FolderPermission } from "~/types"; + +export interface FolderDto { + id: string; + title: string; + slug: string; + permissions: FolderPermission[]; + type: string; + parentId: string | null; +} diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFolders.test.ts b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFolders.test.ts new file mode 100644 index 00000000000..c4fb3a4a4a6 --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFolders.test.ts @@ -0,0 +1,110 @@ +import { GetDescendantFolders } from "./GetDescendantFolders"; +import { folderCacheFactory } from "../cache/FoldersCacheFactory"; +import { Folder } from "../Folder"; + +describe("GetDescendantFolders", () => { + const type = "abc"; + const foldersCache = folderCacheFactory.getCache(type); + + beforeEach(() => { + foldersCache.clear(); + foldersCache.addItems([ + Folder.create({ + id: "folder-1", + title: "Folder 1", + slug: "folder-1", + parentId: null, + permissions: [], + type + }), + Folder.create({ + id: "folder-2", + title: "Folder 2", + slug: "folder-2", + parentId: null, + permissions: [], + type + }), + Folder.create({ + id: "folder-3", + title: "Folder 3", + slug: "folder-3", + parentId: "folder-2", + permissions: [], + type + }), + Folder.create({ + id: "folder-4", + title: "Folder 4", + slug: "folder-4", + parentId: "folder-3", + permissions: [], + type + }) + ]); + }); + + it("should return all descendants of a folder", async () => { + const getDescendantFolders = GetDescendantFolders.instance(type); + + const descendants = getDescendantFolders.execute({ + id: "folder-2" + }); + + expect(descendants).toEqual([ + { + id: "folder-2", + title: "Folder 2", + slug: "folder-2", + parentId: null, + permissions: [], + type + }, + { + id: "folder-3", + title: "Folder 3", + slug: "folder-3", + parentId: "folder-2", + permissions: [], + type + }, + { + id: "folder-4", + title: "Folder 4", + slug: "folder-4", + parentId: "folder-3", + permissions: [], + type + } + ]); + }); + + it("should return the folder it self in case no descendants are found", async () => { + const getDescendantFolders = GetDescendantFolders.instance(type); + + const descendants = getDescendantFolders.execute({ + id: "folder-1" + }); + + expect(descendants).toEqual([ + { + id: "folder-1", + title: "Folder 1", + slug: "folder-1", + parentId: null, + permissions: [], + type + } + ]); + }); + + it("should return empty array if folder does not exist", async () => { + const getDescendantFolders = GetDescendantFolders.instance(type); + + const descendants = getDescendantFolders.execute({ + id: "non-existent-folder" + }); + + expect(descendants).toEqual([]); + }); +}); diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFolders.ts b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFolders.ts new file mode 100644 index 00000000000..ac33309677f --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFolders.ts @@ -0,0 +1,12 @@ +import { IGetDescendantFoldersUseCase } from "./IGetDescendantFoldersUseCase"; +import { GetDescendantFoldersRepository } from "./GetDescendantFoldersRepository"; +import { GetDescendantFoldersUseCase } from "./GetDescendantFoldersUseCase"; +import { folderCacheFactory } from "../cache"; + +export class GetDescendantFolders { + public static instance(type: string): IGetDescendantFoldersUseCase { + const foldersCache = folderCacheFactory.getCache(type); + const repository = new GetDescendantFoldersRepository(foldersCache); + return new GetDescendantFoldersUseCase(repository); + } +} diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFoldersRepository.ts b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFoldersRepository.ts new file mode 100644 index 00000000000..6dc2268b71b --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFoldersRepository.ts @@ -0,0 +1,42 @@ +import { IGetDescendantFoldersRepository } from "./IGetDescendantFoldersRepository"; +import { ListCache } from "../cache"; +import { Folder } from "../Folder"; +import { ROOT_FOLDER } from "~/constants"; + +export class GetDescendantFoldersRepository implements IGetDescendantFoldersRepository { + private readonly cache: ListCache; + + constructor(cache: ListCache) { + this.cache = cache; + } + + execute(id: string): Folder[] { + const currentFolders = this.cache.getItems(); + + if (!id || id === ROOT_FOLDER || !currentFolders.length) { + return []; + } + + const folderMap = new Map(currentFolders.map(folder => [folder.id, folder])); + const result: Folder[] = []; + + const findChildren = (folderId: string) => { + const folder = folderMap.get(folderId); + if (!folder) { + return; + } + + result.push(folder); + + currentFolders.forEach(child => { + if (child.parentId === folder.id) { + findChildren(child.id); + } + }); + }; + + findChildren(id); + + return result; + } +} diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFoldersUseCase.ts b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFoldersUseCase.ts new file mode 100644 index 00000000000..af95700892d --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/GetDescendantFoldersUseCase.ts @@ -0,0 +1,26 @@ +import { IGetDescendantFoldersRepository } from "./IGetDescendantFoldersRepository"; +import { + GetDescendantFoldersParams, + IGetDescendantFoldersUseCase +} from "./IGetDescendantFoldersUseCase"; + +export class GetDescendantFoldersUseCase implements IGetDescendantFoldersUseCase { + private repository: IGetDescendantFoldersRepository; + + constructor(repository: IGetDescendantFoldersRepository) { + this.repository = repository; + } + + execute(params: GetDescendantFoldersParams) { + const folders = this.repository.execute(params.id); + + return folders.map(folder => ({ + id: folder.id, + title: folder.title, + slug: folder.slug, + permissions: folder.permissions, + type: folder.type, + parentId: folder.parentId + })); + } +} diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/IGetDescendantFoldersRepository.ts b/packages/app-aco/src/features/folder/getDescendantFolders/IGetDescendantFoldersRepository.ts new file mode 100644 index 00000000000..97cbe4c59ff --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/IGetDescendantFoldersRepository.ts @@ -0,0 +1,5 @@ +import { Folder } from "../Folder"; + +export interface IGetDescendantFoldersRepository { + execute: (id: string) => Folder[]; +} diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/IGetDescendantFoldersUseCase.ts b/packages/app-aco/src/features/folder/getDescendantFolders/IGetDescendantFoldersUseCase.ts new file mode 100644 index 00000000000..142753f7c72 --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/IGetDescendantFoldersUseCase.ts @@ -0,0 +1,9 @@ +import { FolderDto } from "./FolderDto"; + +export interface GetDescendantFoldersParams { + id: string; +} + +export interface IGetDescendantFoldersUseCase { + execute: (params: GetDescendantFoldersParams) => FolderDto[]; +} diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/index.ts b/packages/app-aco/src/features/folder/getDescendantFolders/index.ts new file mode 100644 index 00000000000..b08ebc4e8ec --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/index.ts @@ -0,0 +1 @@ +export * from "./useGetDescendantFolders"; diff --git a/packages/app-aco/src/features/folder/getDescendantFolders/useGetDescendantFolders.ts b/packages/app-aco/src/features/folder/getDescendantFolders/useGetDescendantFolders.ts new file mode 100644 index 00000000000..10509563968 --- /dev/null +++ b/packages/app-aco/src/features/folder/getDescendantFolders/useGetDescendantFolders.ts @@ -0,0 +1,29 @@ +import { useCallback, useContext } from "react"; +import { GetDescendantFolders } from "./GetDescendantFolders"; +import { FoldersContext } from "~/contexts/folders"; + +export const useGetDescendantFolders = () => { + const foldersContext = useContext(FoldersContext); + + if (!foldersContext) { + throw new Error("useCreateFolder must be used within a FoldersProvider"); + } + + const { type } = foldersContext; + + if (!type) { + throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); + } + + const getDescendantFolders = useCallback( + (id: string) => { + const instance = GetDescendantFolders.instance(type); + return instance.execute({ id }); + }, + [type] + ); + + return { + getDescendantFolders + }; +}; diff --git a/packages/app-aco/src/features/folder/getFolder/GetFolder.ts b/packages/app-aco/src/features/folder/getFolder/GetFolder.ts new file mode 100644 index 00000000000..ef82ff0841f --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/GetFolder.ts @@ -0,0 +1,17 @@ +import { loadingRepositoryFactory } from "@webiny/app-utils"; +import { IGetFolderUseCase } from "./IGetFolderUseCase"; +import { IGetFolderGateway } from "./IGetFolderGateway"; +import { GetFolderRepository } from "./GetFolderRepository"; +import { GetFolderUseCase } from "./GetFolderUseCase"; +import { GetFolderUseCaseWithLoading } from "./GetFolderUseCaseWithLoading"; +import { folderCacheFactory } from "../cache"; + +export class GetFolder { + public static instance(type: string, gateway: IGetFolderGateway): IGetFolderUseCase { + const foldersCache = folderCacheFactory.getCache(type); + const loadingRepository = loadingRepositoryFactory.getRepository(type); + const repository = new GetFolderRepository(foldersCache, gateway); + const useCase = new GetFolderUseCase(repository); + return new GetFolderUseCaseWithLoading(loadingRepository, useCase); + } +} diff --git a/packages/app-aco/src/features/folder/getFolder/GetFolderGqlGateway.ts b/packages/app-aco/src/features/folder/getFolder/GetFolderGqlGateway.ts new file mode 100644 index 00000000000..3439569c7de --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/GetFolderGqlGateway.ts @@ -0,0 +1,97 @@ +import ApolloClient from "apollo-client"; +import gql from "graphql-tag"; +import { IGetFolderGateway } from "./IGetFolderGateway"; +import { FolderItem, AcoError } from "~/types"; + +export interface GetFolderResponse { + aco: { + getFolder: { + data: FolderItem | null; + error: AcoError | null; + }; + }; +} + +export interface GetFolderQueryVariables { + id: string; +} + +export const GET_FOLDER = gql` + query GetFolder($id: ID!) { + aco { + getFolder(id: $id) { + data { + id + title + slug + permissions { + target + level + inheritedFrom + } + hasNonInheritedPermissions + canManagePermissions + canManageStructure + canManageContent + parentId + type + savedOn + savedBy { + id + displayName + } + createdOn + createdBy { + id + displayName + } + modifiedOn + modifiedBy { + id + displayName + } + } + error { + code + data + message + } + } + } + } +`; + +export class GetFolderGqlGateway implements IGetFolderGateway { + private client: ApolloClient; + + constructor(client: ApolloClient) { + this.client = client; + } + + async execute(id: string) { + if (!id) { + throw new Error("Folder `id` is mandatory"); + } + + const { data: response } = await this.client.query< + GetFolderResponse, + GetFolderQueryVariables + >({ + query: GET_FOLDER, + variables: { id }, + fetchPolicy: "network-only" + }); + + if (!response) { + throw new Error("Network error while fetch folder."); + } + + const { data, error } = response.aco.getFolder; + + if (!data) { + throw new Error(error?.message || `Could not fetch folder with id: ${id}`); + } + + return data; + } +} diff --git a/packages/app-aco/src/features/folder/getFolder/GetFolderRepository.ts b/packages/app-aco/src/features/folder/getFolder/GetFolderRepository.ts new file mode 100644 index 00000000000..37750d1f1a1 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/GetFolderRepository.ts @@ -0,0 +1,19 @@ +import { Folder } from "../Folder"; +import { ListCache } from "../cache"; +import { IGetFolderRepository } from "./IGetFolderRepository"; +import { IGetFolderGateway } from "./IGetFolderGateway"; + +export class GetFolderRepository implements IGetFolderRepository { + private cache: ListCache; + private gateway: IGetFolderGateway; + + constructor(cache: ListCache, gateway: IGetFolderGateway) { + this.cache = cache; + this.gateway = gateway; + } + + async execute(id: string) { + const response = await this.gateway.execute(id); + this.cache.addItems([Folder.create(response)]); + } +} diff --git a/packages/app-aco/src/features/folder/getFolder/GetFolderUseCase.ts b/packages/app-aco/src/features/folder/getFolder/GetFolderUseCase.ts new file mode 100644 index 00000000000..68cd0e14f4d --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/GetFolderUseCase.ts @@ -0,0 +1,14 @@ +import { GetFolderParams, IGetFolderUseCase } from "./IGetFolderUseCase"; +import { IGetFolderRepository } from "./IGetFolderRepository"; + +export class GetFolderUseCase implements IGetFolderUseCase { + private repository: IGetFolderRepository; + + constructor(repository: IGetFolderRepository) { + this.repository = repository; + } + + async execute(params: GetFolderParams) { + await this.repository.execute(params.id); + } +} diff --git a/packages/app-aco/src/features/folder/getFolder/GetFolderUseCaseWithLoading.ts b/packages/app-aco/src/features/folder/getFolder/GetFolderUseCaseWithLoading.ts new file mode 100644 index 00000000000..a1ee2e0816d --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/GetFolderUseCaseWithLoading.ts @@ -0,0 +1,20 @@ +import { GetFolderParams, IGetFolderUseCase } from "./IGetFolderUseCase"; +import { ILoadingRepository } from "@webiny/app-utils"; +import { LoadingActionsEnum } from "~/types"; + +export class GetFolderUseCaseWithLoading implements IGetFolderUseCase { + private loadingRepository: ILoadingRepository; + private useCase: IGetFolderUseCase; + + constructor(loadingRepository: ILoadingRepository, useCase: IGetFolderUseCase) { + this.loadingRepository = loadingRepository; + this.useCase = useCase; + } + + async execute(params: GetFolderParams) { + await this.loadingRepository.runCallBack( + this.useCase.execute(params), + LoadingActionsEnum.get + ); + } +} diff --git a/packages/app-aco/src/features/folder/getFolder/IGetFolderGateway.ts b/packages/app-aco/src/features/folder/getFolder/IGetFolderGateway.ts new file mode 100644 index 00000000000..401dcb4f004 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/IGetFolderGateway.ts @@ -0,0 +1,24 @@ +import { CmsIdentity, FolderPermission } from "~/types"; + +export interface FolderDto { + id: string; + title: string; + slug: string; + permissions: FolderPermission[]; + hasNonInheritedPermissions: boolean; + canManagePermissions: boolean; + canManageStructure: boolean; + canManageContent: boolean; + type: string; + parentId: string | null; + createdBy: CmsIdentity; + createdOn: string; + savedBy: CmsIdentity; + savedOn: string; + modifiedBy: CmsIdentity | null; + modifiedOn: string | null; +} + +export interface IGetFolderGateway { + execute: (id: string) => Promise; +} diff --git a/packages/app-aco/src/features/folder/getFolder/IGetFolderRepository.ts b/packages/app-aco/src/features/folder/getFolder/IGetFolderRepository.ts new file mode 100644 index 00000000000..48bcaf7f250 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/IGetFolderRepository.ts @@ -0,0 +1,3 @@ +export interface IGetFolderRepository { + execute: (id: string) => Promise; +} diff --git a/packages/app-aco/src/features/folder/getFolder/IGetFolderUseCase.ts b/packages/app-aco/src/features/folder/getFolder/IGetFolderUseCase.ts new file mode 100644 index 00000000000..22cf8bcc44a --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/IGetFolderUseCase.ts @@ -0,0 +1,7 @@ +export interface GetFolderParams { + id: string; +} + +export interface IGetFolderUseCase { + execute: (params: GetFolderParams) => Promise; +} diff --git a/packages/app-aco/src/features/folder/getFolder/index.ts b/packages/app-aco/src/features/folder/getFolder/index.ts new file mode 100644 index 00000000000..ed6c1f0e032 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/index.ts @@ -0,0 +1 @@ +export * from "./useGetFolder"; diff --git a/packages/app-aco/src/features/folder/getFolder/useGetFolder.ts b/packages/app-aco/src/features/folder/getFolder/useGetFolder.ts new file mode 100644 index 00000000000..c0d606a38e3 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolder/useGetFolder.ts @@ -0,0 +1,35 @@ +import { useCallback, useContext } from "react"; +import { useApolloClient } from "@apollo/react-hooks"; +import { GetFolderGqlGateway } from "./GetFolderGqlGateway"; +import { GetFolderParams } from "./IGetFolderUseCase"; +import { GetFolder } from "./GetFolder"; +import { FoldersContext } from "~/contexts/folders"; + +export const useGetFolder = () => { + const client = useApolloClient(); + const gateway = new GetFolderGqlGateway(client); + + const foldersContext = useContext(FoldersContext); + + if (!foldersContext) { + throw new Error("useGetFolder must be used within a FoldersProvider"); + } + + const { type } = foldersContext; + + if (!type) { + throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); + } + + const getFolder = useCallback( + (params: GetFolderParams) => { + const instance = GetFolder.instance(type, gateway); + return instance.execute(params); + }, + [type, gateway] + ); + + return { + getFolder + }; +}; diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/FolderPermissionName.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/FolderPermissionName.ts new file mode 100644 index 00000000000..a9483befe9b --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/FolderPermissionName.ts @@ -0,0 +1,4 @@ +export type FolderPermissionName = + | "canManagePermissions" + | "canManageStructure" + | "canManageContent"; diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermission.test.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermission.test.ts new file mode 100644 index 00000000000..9d8a8c1e7ee --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermission.test.ts @@ -0,0 +1,225 @@ +import { GetFolderLevelPermission } from "./GetFolderLevelPermission"; +import { folderCacheFactory } from "../cache/FoldersCacheFactory"; +import { Folder } from "../Folder"; + +describe("GetFolderLevelPermission", () => { + const type = "abc"; + const foldersCache = folderCacheFactory.getCache(type); + + beforeEach(() => { + foldersCache.clear(); + foldersCache.addItems([ + Folder.create({ + id: "folder-canManageContent", + title: "Folder canManageContent", + slug: "folder-canManageContent", + parentId: null, + permissions: [], + canManageContent: true, + type + }), + Folder.create({ + id: "folder-canManageStructure", + title: "Folder canManageStructure", + slug: "folder-canManageStructure", + parentId: null, + permissions: [], + canManageStructure: true, + type + }), + Folder.create({ + id: "folder-canManagePermissions", + title: "Folder canManagePermissions3", + slug: "folder-canManagePermissions", + parentId: null, + permissions: [], + canManagePermissions: true, + type + }), + Folder.create({ + id: "folder-no-permissions", + title: "Folder No Permissions", + slug: "folder-no-permissions", + parentId: null, + permissions: [], + type + }) + ]); + }); + + it("should return true in case a specific permission is set at folder level and FLP is enabled", async () => { + // canManagePermissions + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManagePermissions", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-canManagePermissions" + }); + expect(result).toBeTrue(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageStructure", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-canManageStructure" + }); + expect(result).toBeTrue(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageContent", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-canManageContent" + }); + expect(result).toBeTrue(); + } + }); + + it("should return false in case a specific permission is not set at folder level and FLP is enabled", async () => { + // canManagePermissions + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManagePermissions", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-no-permissions" + }); + expect(result).toBeFalse(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageStructure", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-no-permissions" + }); + expect(result).toBeFalse(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageContent", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-no-permissions" + }); + expect(result).toBeFalse(); + } + }); + + it("should return always false in case the folder is not found", async () => { + // canManagePermissions + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManagePermissions", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "not-existing-folder" + }); + expect(result).toBeFalse(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageStructure", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "not-existing-folder" + }); + expect(result).toBeFalse(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageContent", + true + ); + + const result = getFolderLevelPermission.execute({ + id: "not-existing-folder" + }); + expect(result).toBeFalse(); + } + }); + + it("should return always true in case FLP is not enabled", async () => { + // canManagePermissions + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManagePermissions", + false + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-no-permissions" + }); + expect(result).toBeTrue(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageStructure", + false + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-no-permissions" + }); + expect(result).toBeTrue(); + } + + // canManageStructure + { + const getFolderLevelPermission = GetFolderLevelPermission.instance( + type, + "canManageContent", + false + ); + + const result = getFolderLevelPermission.execute({ + id: "folder-no-permissions" + }); + expect(result).toBeTrue(); + } + }); +}); diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermission.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermission.ts new file mode 100644 index 00000000000..eb7d50c208d --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermission.ts @@ -0,0 +1,23 @@ +import { IGetFolderLevelPermissionUseCase } from "./IGetFolderLevelPermissionUseCase"; +import { GetFolderLevelPermissionRepository } from "./GetFolderLevelPermissionRepository"; +import { GetFolderLevelPermissionWithFlpUseCase } from "./GetFolderLevelPermissionWithFlpUseCase"; +import { GetFolderLevelPermissionUseCase } from "./GetFolderLevelPermissionUseCase"; +import { FolderPermissionName } from "./FolderPermissionName"; +import { folderCacheFactory } from "../cache"; + +export class GetFolderLevelPermission { + public static instance( + type: string, + permissionName: FolderPermissionName, + canUseFlp: boolean + ): IGetFolderLevelPermissionUseCase { + const foldersCache = folderCacheFactory.getCache(type); + const repository = new GetFolderLevelPermissionRepository(foldersCache, permissionName); + + if (canUseFlp) { + return new GetFolderLevelPermissionWithFlpUseCase(repository); + } + + return new GetFolderLevelPermissionUseCase(); + } +} diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionRepository.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionRepository.ts new file mode 100644 index 00000000000..c3d9e2b9940 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionRepository.ts @@ -0,0 +1,24 @@ +import { ListCache } from "../cache"; +import { IGetFolderLevelPermissionRepository } from "./IGetFolderLevelPermissionRepository"; +import { FolderPermissionName } from "./FolderPermissionName"; +import { Folder } from "~/features/folder"; + +export class GetFolderLevelPermissionRepository implements IGetFolderLevelPermissionRepository { + private cache: ListCache; + private readonly permissionName: FolderPermissionName; + + constructor(cache: ListCache, permissionName: FolderPermissionName) { + this.cache = cache; + this.permissionName = permissionName; + } + + execute(id: string) { + const folder = this.cache.getItem(folder => folder.id === id); + + if (!folder) { + return false; + } + + return folder[this.permissionName] ?? false; + } +} diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionUseCase.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionUseCase.ts new file mode 100644 index 00000000000..982a2d1a442 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionUseCase.ts @@ -0,0 +1,7 @@ +import { IGetFolderLevelPermissionUseCase } from "./IGetFolderLevelPermissionUseCase"; + +export class GetFolderLevelPermissionUseCase implements IGetFolderLevelPermissionUseCase { + execute() { + return true; + } +} diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionWithFlpUseCase.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionWithFlpUseCase.ts new file mode 100644 index 00000000000..351aad0167e --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/GetFolderLevelPermissionWithFlpUseCase.ts @@ -0,0 +1,17 @@ +import { + GetFolderLevelPermissionParams, + IGetFolderLevelPermissionUseCase +} from "./IGetFolderLevelPermissionUseCase"; +import { IGetFolderLevelPermissionRepository } from "./IGetFolderLevelPermissionRepository"; + +export class GetFolderLevelPermissionWithFlpUseCase implements IGetFolderLevelPermissionUseCase { + private repository: IGetFolderLevelPermissionRepository; + + constructor(repository: IGetFolderLevelPermissionRepository) { + this.repository = repository; + } + + execute(params: GetFolderLevelPermissionParams) { + return this.repository.execute(params.id); + } +} diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/IGetFolderLevelPermissionRepository.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/IGetFolderLevelPermissionRepository.ts new file mode 100644 index 00000000000..69585a37ed3 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/IGetFolderLevelPermissionRepository.ts @@ -0,0 +1,3 @@ +export interface IGetFolderLevelPermissionRepository { + execute: (id: string) => boolean; +} diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/IGetFolderLevelPermissionUseCase.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/IGetFolderLevelPermissionUseCase.ts new file mode 100644 index 00000000000..443205e3c38 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/IGetFolderLevelPermissionUseCase.ts @@ -0,0 +1,7 @@ +export interface GetFolderLevelPermissionParams { + id: string; +} + +export interface IGetFolderLevelPermissionUseCase { + execute: (params: GetFolderLevelPermissionParams) => boolean; +} diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/index.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/index.ts new file mode 100644 index 00000000000..ad2572bc388 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/index.ts @@ -0,0 +1 @@ +export * from "./useGetFolderLevelPermission"; diff --git a/packages/app-aco/src/features/folder/getFolderLevelPermission/useGetFolderLevelPermission.ts b/packages/app-aco/src/features/folder/getFolderLevelPermission/useGetFolderLevelPermission.ts new file mode 100644 index 00000000000..de1dab3dc07 --- /dev/null +++ b/packages/app-aco/src/features/folder/getFolderLevelPermission/useGetFolderLevelPermission.ts @@ -0,0 +1,37 @@ +import { useCallback, useContext } from "react"; +import { useWcp } from "@webiny/app-wcp"; +import { FoldersContext } from "~/contexts/folders"; +import { GetFolderLevelPermission } from "./GetFolderLevelPermission"; +import { FolderPermissionName } from "./FolderPermissionName"; + +export const useGetFolderLevelPermission = (permissionName: FolderPermissionName) => { + const { canUseFolderLevelPermissions } = useWcp(); + + const foldersContext = useContext(FoldersContext); + + if (!foldersContext) { + throw new Error("useGetCanManageContent must be used within a FoldersProvider"); + } + + const { type } = foldersContext; + + if (!type) { + throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); + } + + const getFolderLevelPermission = useCallback( + (id: string) => { + const instance = GetFolderLevelPermission.instance( + type, + permissionName, + canUseFolderLevelPermissions() + ); + return instance.execute({ id }); + }, + [type, canUseFolderLevelPermissions] + ); + + return { + getFolderLevelPermission + }; +}; diff --git a/packages/app-aco/src/features/folder/index.ts b/packages/app-aco/src/features/folder/index.ts new file mode 100644 index 00000000000..fc6678906ca --- /dev/null +++ b/packages/app-aco/src/features/folder/index.ts @@ -0,0 +1,9 @@ +export * from "./Folder"; +export * from "./cache"; +export * from "./createFolder"; +export * from "./deleteFolder"; +export * from "./getDescendantFolders"; +export * from "./getFolder"; +export * from "./getFolderLevelPermission"; +export * from "./listFolders"; +export * from "./updateFolder"; diff --git a/packages/app-aco/src/features/folder/listFolders/FolderDto.ts b/packages/app-aco/src/features/folder/listFolders/FolderDto.ts new file mode 100644 index 00000000000..053a523ebc9 --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/FolderDto.ts @@ -0,0 +1,53 @@ +import { CmsIdentity, FolderPermission } from "~/types"; +import { Folder } from "../Folder"; +import { ROOT_FOLDER } from "~/constants"; + +export interface FolderDto { + id: string; + title: string; + slug: string; + type: string; + parentId: string; + permissions: FolderPermission[]; + hasNonInheritedPermissions: boolean; + canManagePermissions: boolean; + canManageStructure: boolean; + canManageContent: boolean; + createdBy: CmsIdentity; + createdOn: string; + savedBy: CmsIdentity; + savedOn: string; + modifiedBy: CmsIdentity; + modifiedOn: string; +} + +export class FolderDtoMapper { + static toDTO(folder: Folder): FolderDto { + return { + id: folder.id, + title: folder.title, + canManageContent: folder.canManageContent ?? false, + canManagePermissions: folder.canManagePermissions ?? false, + canManageStructure: folder.canManageStructure ?? false, + createdBy: this.createIdentity(folder.createdBy), + createdOn: folder.createdOn ?? "", + hasNonInheritedPermissions: folder.hasNonInheritedPermissions ?? false, + modifiedBy: this.createIdentity(folder.modifiedBy), + modifiedOn: folder.modifiedOn ?? "", + parentId: folder.parentId ?? ROOT_FOLDER, + permissions: folder.permissions ?? [], + savedBy: this.createIdentity(folder.savedBy), + savedOn: folder.savedOn ?? "", + slug: folder.slug, + type: folder.type + }; + } + + private static createIdentity(identity?: CmsIdentity | null): CmsIdentity { + return { + id: identity?.id || "", + displayName: identity?.displayName || "", + type: identity?.type || "" + }; + } +} diff --git a/packages/app-aco/src/features/folder/listFolders/FolderGqlDto.ts b/packages/app-aco/src/features/folder/listFolders/FolderGqlDto.ts new file mode 100644 index 00000000000..0d2ff62504c --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/FolderGqlDto.ts @@ -0,0 +1,20 @@ +import { CmsIdentity, FolderPermission } from "~/types"; + +export interface FolderGqlDto { + id: string; + title: string; + slug: string; + permissions: FolderPermission[]; + hasNonInheritedPermissions: boolean; + canManagePermissions: boolean; + canManageStructure: boolean; + canManageContent: boolean; + type: string; + parentId: string | null; + createdBy: CmsIdentity; + createdOn: string; + savedBy: CmsIdentity; + savedOn: string; + modifiedBy: CmsIdentity | null; + modifiedOn: string | null; +} diff --git a/packages/app-aco/src/features/folder/listFolders/IListFoldersGateway.ts b/packages/app-aco/src/features/folder/listFolders/IListFoldersGateway.ts new file mode 100644 index 00000000000..f3806afb4aa --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/IListFoldersGateway.ts @@ -0,0 +1,9 @@ +import { FolderGqlDto } from "./FolderGqlDto"; + +export interface ListFoldersGatewayParams { + type: string; +} + +export interface IListFoldersGateway { + execute: (params: ListFoldersGatewayParams) => Promise; +} diff --git a/packages/app-aco/src/features/folder/listFolders/IListFoldersRepository.ts b/packages/app-aco/src/features/folder/listFolders/IListFoldersRepository.ts new file mode 100644 index 00000000000..f9aedf93d50 --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/IListFoldersRepository.ts @@ -0,0 +1,3 @@ +export interface IListFoldersRepository { + execute: () => Promise; +} diff --git a/packages/app-aco/src/features/folder/listFolders/IListFoldersUseCase.ts b/packages/app-aco/src/features/folder/listFolders/IListFoldersUseCase.ts new file mode 100644 index 00000000000..7024ce59ede --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/IListFoldersUseCase.ts @@ -0,0 +1,3 @@ +export interface IListFoldersUseCase { + execute: () => Promise; +} diff --git a/packages/app-aco/src/features/folder/listFolders/ListFolders.test.ts b/packages/app-aco/src/features/folder/listFolders/ListFolders.test.ts new file mode 100644 index 00000000000..519c01cc587 --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/ListFolders.test.ts @@ -0,0 +1,119 @@ +import { ListFolders } from "./ListFolders"; +import { folderCacheFactory } from "../cache/FoldersCacheFactory"; + +describe("ListFolders", () => { + const type = "abc"; + const gateway = { + execute: jest.fn().mockResolvedValue([ + { + id: "folder-1", + title: "Folder 1", + slug: "folder-1", + type + }, + { + id: "folder-2", + title: "Folder 2", + slug: "folder-1", + type + }, + { + id: "folder-3", + title: "Folder 3", + slug: "folder-3", + type + } + ]) + }; + const foldersCache = folderCacheFactory.getCache(type); + + beforeEach(() => { + foldersCache.clear(); + jest.clearAllMocks(); + }); + + it("should be able to list folders", async () => { + const listFolders = ListFolders.instance(type, gateway); + + expect(foldersCache.hasItems()).toBeFalse(); + + await listFolders.useCase.execute(); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + expect(foldersCache.hasItems()).toBeTrue(); + + const items = foldersCache.getItems(); + expect(items.length).toEqual(3); + }); + + it("should return empty array if no folders are found", async () => { + const emptyGateway = { + execute: jest.fn().mockResolvedValue([]) + }; + const listFolders = ListFolders.instance(type, emptyGateway); + + expect(foldersCache.hasItems()).toBeFalse(); + + await listFolders.useCase.execute(); + + expect(emptyGateway.execute).toHaveBeenCalledTimes(1); + expect(foldersCache.hasItems()).toBeFalse(); + + const items = foldersCache.getItems(); + expect(items.length).toEqual(0); + }); + + it("should handle gateway errors gracefully", async () => { + const errorGateway = { + execute: jest.fn().mockRejectedValue(new Error("Gateway error")) + }; + const listFolders = ListFolders.instance(type, errorGateway); + + expect(foldersCache.hasItems()).toBeFalse(); + + await expect(listFolders.useCase.execute()).rejects.toThrow("Gateway error"); + + expect(errorGateway.execute).toHaveBeenCalledTimes(1); + expect(foldersCache.hasItems()).toBeFalse(); + }); + + it("should NOT cache folders after listing", async () => { + const listFolders = ListFolders.instance(type, gateway); + + expect(foldersCache.hasItems()).toBeFalse(); + + await listFolders.useCase.execute(); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + expect(foldersCache.hasItems()).toBeTrue(); + + const items = foldersCache.getItems(); + expect(items.length).toEqual(3); + + // Execute again, it should execute the gateway again + await listFolders.useCase.execute(); + expect(gateway.execute).toHaveBeenCalledTimes(2); + }); + + it("should clear cache when type changes", async () => { + const listFolders = ListFolders.instance(type, gateway); + + expect(foldersCache.hasItems()).toBeFalse(); + + await listFolders.useCase.execute(); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + expect(foldersCache.hasItems()).toBeTrue(); + + const newType = "xyz"; + const newFoldersCache = folderCacheFactory.getCache(newType); + const newListFolders = ListFolders.instance(newType, gateway); + + expect(newFoldersCache.hasItems()).toBeFalse(); + + await newListFolders.useCase.execute(); + + expect(gateway.execute).toHaveBeenCalledTimes(2); + expect(newFoldersCache.hasItems()).toBeTrue(); + }); +}); diff --git a/packages/app-aco/src/features/folder/listFolders/ListFolders.ts b/packages/app-aco/src/features/folder/listFolders/ListFolders.ts new file mode 100644 index 00000000000..3623c849fae --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/ListFolders.ts @@ -0,0 +1,30 @@ +import { LoadingRepository, loadingRepositoryFactory } from "@webiny/app-utils"; +import { IListFoldersUseCase } from "./IListFoldersUseCase"; +import { IListFoldersGateway } from "./IListFoldersGateway"; +import { ListFoldersRepository } from "./ListFoldersRepository"; +import { ListFoldersUseCaseWithLoading } from "./ListFoldersUseCaseWithLoading"; +import { ListFoldersUseCase } from "./ListFoldersUseCase"; +import { folderCacheFactory, ListCache } from "../cache"; +import { Folder } from "~/features/folder"; + +interface IListFoldersInstance { + useCase: IListFoldersUseCase; + folders: ListCache; + loading: LoadingRepository; +} + +export class ListFolders { + public static instance(type: string, gateway: IListFoldersGateway): IListFoldersInstance { + const foldersCache = folderCacheFactory.getCache(type); + const loadingRepository = loadingRepositoryFactory.getRepository(type); + const repository = new ListFoldersRepository(foldersCache, gateway, type); + const useCase = new ListFoldersUseCase(repository); + const useCaseWithLoading = new ListFoldersUseCaseWithLoading(loadingRepository, useCase); + + return { + useCase: useCaseWithLoading, + folders: foldersCache, + loading: loadingRepository + }; + } +} diff --git a/packages/app-aco/src/features/folder/listFolders/ListFoldersGqlGateway.ts b/packages/app-aco/src/features/folder/listFolders/ListFoldersGqlGateway.ts new file mode 100644 index 00000000000..7bfba60a75c --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/ListFoldersGqlGateway.ts @@ -0,0 +1,129 @@ +import ApolloClient from "apollo-client"; +import gql from "graphql-tag"; +import { IListFoldersGateway, ListFoldersGatewayParams } from "./IListFoldersGateway"; +import { AcoError, FolderItem } from "~/types"; +import { ROOT_FOLDER } from "~/constants"; + +export interface ListFoldersResponse { + aco: { + listFolders: { + data: FolderItem[] | null; + error: AcoError | null; + }; + }; +} + +export interface ListFoldersQueryVariables { + type: string; + limit: number; + sort?: Record; + after?: string | null; +} + +export const LIST_FOLDERS = gql` + query ListFolders($type: String!, $limit: Int!) { + aco { + listFolders(where: { type: $type }, limit: $limit) { + data { + id + title + slug + permissions { + target + level + inheritedFrom + } + hasNonInheritedPermissions + canManagePermissions + canManageStructure + canManageContent + parentId + type + savedOn + savedBy { + id + displayName + } + createdOn + createdBy { + id + displayName + } + modifiedOn + modifiedBy { + id + displayName + } + } + error { + code + data + message + } + } + } + } +`; + +export class ListFoldersGqlGateway implements IListFoldersGateway { + private client: ApolloClient; + + constructor(client: ApolloClient) { + this.client = client; + } + + async execute(params: ListFoldersGatewayParams) { + const { data: response } = await this.client.query< + ListFoldersResponse, + ListFoldersQueryVariables + >({ + query: LIST_FOLDERS, + variables: { + ...params, + limit: 10000 + }, + fetchPolicy: "network-only" + }); + + if (!response) { + throw new Error("Network error while listing folders."); + } + + const { data, error } = response.aco.listFolders; + + if (!data) { + throw new Error(error?.message || "Could not fetch folders"); + } + + return [this.getRootFolder(), ...(data || [])]; + } + + private getRootFolder(): FolderItem { + return { + id: ROOT_FOLDER, + title: "Home", + permissions: [], + parentId: "0", + slug: "", + createdOn: "", + createdBy: { + id: "", + displayName: "", + type: "" + }, + hasNonInheritedPermissions: false, + canManagePermissions: true, + canManageStructure: true, + canManageContent: true, + savedOn: "", + savedBy: { + id: "", + displayName: "", + type: "" + }, + modifiedOn: null, + modifiedBy: null, + type: "$ROOT" + }; + } +} diff --git a/packages/app-aco/src/features/folder/listFolders/ListFoldersRepository.ts b/packages/app-aco/src/features/folder/listFolders/ListFoldersRepository.ts new file mode 100644 index 00000000000..ddec23543e5 --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/ListFoldersRepository.ts @@ -0,0 +1,22 @@ +import { ListCache } from "../cache"; +import { Folder } from "../Folder"; +import { IListFoldersGateway } from "./IListFoldersGateway"; +import { IListFoldersRepository } from "./IListFoldersRepository"; + +export class ListFoldersRepository implements IListFoldersRepository { + private cache: ListCache; + private gateway: IListFoldersGateway; + private type: string; + + constructor(cache: ListCache, gateway: IListFoldersGateway, type: string) { + this.cache = cache; + this.gateway = gateway; + this.type = type; + } + + async execute() { + const items = await this.gateway.execute({ type: this.type }); + this.cache.clear(); + this.cache.addItems(items.map(item => Folder.create(item))); + } +} diff --git a/packages/app-aco/src/features/folder/listFolders/ListFoldersUseCase.ts b/packages/app-aco/src/features/folder/listFolders/ListFoldersUseCase.ts new file mode 100644 index 00000000000..ac913080897 --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/ListFoldersUseCase.ts @@ -0,0 +1,14 @@ +import { IListFoldersUseCase } from "./IListFoldersUseCase"; +import { IListFoldersRepository } from "./IListFoldersRepository"; + +export class ListFoldersUseCase implements IListFoldersUseCase { + private repository: IListFoldersRepository; + + constructor(repository: IListFoldersRepository) { + this.repository = repository; + } + + async execute() { + await this.repository.execute(); + } +} diff --git a/packages/app-aco/src/features/folder/listFolders/ListFoldersUseCaseWithLoading.ts b/packages/app-aco/src/features/folder/listFolders/ListFoldersUseCaseWithLoading.ts new file mode 100644 index 00000000000..ff772fbb964 --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/ListFoldersUseCaseWithLoading.ts @@ -0,0 +1,17 @@ +import { ILoadingRepository } from "@webiny/app-utils"; +import { LoadingActionsEnum } from "~/types"; +import { IListFoldersUseCase } from "./IListFoldersUseCase"; + +export class ListFoldersUseCaseWithLoading implements IListFoldersUseCase { + private loadingRepository: ILoadingRepository; + private useCase: IListFoldersUseCase; + + constructor(loadingRepository: ILoadingRepository, useCase: IListFoldersUseCase) { + this.loadingRepository = loadingRepository; + this.useCase = useCase; + } + + async execute() { + await this.loadingRepository.runCallBack(this.useCase.execute(), LoadingActionsEnum.list); + } +} diff --git a/packages/app-aco/src/features/folder/listFolders/index.ts b/packages/app-aco/src/features/folder/listFolders/index.ts new file mode 100644 index 00000000000..92780101758 --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/index.ts @@ -0,0 +1,2 @@ +export * from "./useListFolders"; +export * from "./FolderDto"; diff --git a/packages/app-aco/src/features/folder/listFolders/useListFolders.ts b/packages/app-aco/src/features/folder/listFolders/useListFolders.ts new file mode 100644 index 00000000000..96db86776cf --- /dev/null +++ b/packages/app-aco/src/features/folder/listFolders/useListFolders.ts @@ -0,0 +1,82 @@ +import { useCallback, useContext, useEffect, useMemo, useState } from "react"; +import { autorun } from "mobx"; +import { useApolloClient } from "@apollo/react-hooks"; +import { ListFoldersGqlGateway } from "./ListFoldersGqlGateway"; +import { ListFolders } from "./ListFolders"; +import { FolderDtoMapper } from "./FolderDto"; +import { FoldersContext } from "~/contexts/folders"; +import { FolderItem } from "~/types"; + +export const useListFolders = () => { + const client = useApolloClient(); + const gateway = new ListFoldersGqlGateway(client); + + const [vm, setVm] = useState<{ + folders: FolderItem[]; + loading: Record; + }>({ + folders: [], + loading: { + INIT: true + } + }); + + const foldersContext = useContext(FoldersContext); + + if (!foldersContext) { + throw new Error("useCreateFolder must be used within a FoldersProvider"); + } + + const { type } = foldersContext; + + if (!type) { + throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); + } + + const { + useCase, + folders: foldersCache, + loading + } = useMemo(() => { + return ListFolders.instance(type, gateway); + }, [type, gateway]); + + const listFolders = useCallback(() => { + return useCase.execute(); + }, [useCase]); + + useEffect(() => { + if (foldersCache.hasItems()) { + return; // Skip if we already have folders in the cache. + } + + listFolders(); + }, []); + + useEffect(() => { + return autorun(() => { + const folders = foldersCache.getItems().map(folder => FolderDtoMapper.toDTO(folder)); + + setVm(vm => ({ + ...vm, + folders + })); + }); + }, [foldersCache]); + + useEffect(() => { + return autorun(() => { + const loadingState = loading.get(); + + setVm(vm => ({ + ...vm, + loading: loadingState + })); + }); + }, [loading]); + + return { + ...vm, + listFolders + }; +}; diff --git a/packages/app-aco/src/features/folder/updateFolder/FolderDto.ts b/packages/app-aco/src/features/folder/updateFolder/FolderDto.ts new file mode 100644 index 00000000000..b162e2cc1e3 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/FolderDto.ts @@ -0,0 +1,9 @@ +import { FolderPermission } from "~/types"; + +export interface FolderDto { + id: string; + title: string; + slug: string; + permissions: FolderPermission[]; + parentId: string | null; +} diff --git a/packages/app-aco/src/features/folder/updateFolder/FolderGqlDto.ts b/packages/app-aco/src/features/folder/updateFolder/FolderGqlDto.ts new file mode 100644 index 00000000000..0d2ff62504c --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/FolderGqlDto.ts @@ -0,0 +1,20 @@ +import { CmsIdentity, FolderPermission } from "~/types"; + +export interface FolderGqlDto { + id: string; + title: string; + slug: string; + permissions: FolderPermission[]; + hasNonInheritedPermissions: boolean; + canManagePermissions: boolean; + canManageStructure: boolean; + canManageContent: boolean; + type: string; + parentId: string | null; + createdBy: CmsIdentity; + createdOn: string; + savedBy: CmsIdentity; + savedOn: string; + modifiedBy: CmsIdentity | null; + modifiedOn: string | null; +} diff --git a/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderGateway.ts b/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderGateway.ts new file mode 100644 index 00000000000..f6003e70efc --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderGateway.ts @@ -0,0 +1,6 @@ +import { FolderDto } from "./FolderDto"; +import { FolderGqlDto } from "./FolderGqlDto"; + +export interface IUpdateFolderGateway { + execute: (folder: FolderDto) => Promise; +} diff --git a/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderRepository.ts b/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderRepository.ts new file mode 100644 index 00000000000..76c55d36275 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderRepository.ts @@ -0,0 +1,5 @@ +import { Folder } from "../Folder"; + +export interface IUpdateFolderRepository { + execute: (folder: Folder) => Promise; +} diff --git a/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderUseCase.ts b/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderUseCase.ts new file mode 100644 index 00000000000..173404f49ae --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/IUpdateFolderUseCase.ts @@ -0,0 +1,14 @@ +import { FolderPermission } from "~/types"; + +export interface UpdateFolderParams { + id: string; + title: string; + slug: string; + type: string; + parentId: string | null; + permissions: FolderPermission[]; +} + +export interface IUpdateFolderUseCase { + execute: (params: UpdateFolderParams) => Promise; +} diff --git a/packages/app-aco/src/features/folder/updateFolder/UpdateFolder.test.ts b/packages/app-aco/src/features/folder/updateFolder/UpdateFolder.test.ts new file mode 100644 index 00000000000..1ad82996072 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/UpdateFolder.test.ts @@ -0,0 +1,111 @@ +import { UpdateFolder } from "./UpdateFolder"; +import { folderCacheFactory } from "../cache/FoldersCacheFactory"; +import { Folder } from "../Folder"; + +describe("UpdateFolder", () => { + const type = "abc"; + + const foldersCache = folderCacheFactory.getCache(type); + + beforeEach(() => { + jest.clearAllMocks(); + foldersCache.clear(); + foldersCache.addItems([ + Folder.create({ + id: "any-folder-id", + title: "Any Folder", + slug: "any-folder", + parentId: null, + permissions: [], + type + }) + ]); + }); + + it("should be able to update a folder", async () => { + const gateway = { + execute: jest.fn().mockResolvedValue({ + id: "any-folder-id", + title: "Updated Folder", + slug: "updated-folder", + parentId: "another-id", + permissions: [], + type + }) + }; + + const updateFolder = UpdateFolder.instance(type, gateway); + + expect(foldersCache.hasItems()).toBeTrue(); + const item = foldersCache.getItem(folder => folder.id === "any-folder-id"); + expect(item?.id).toEqual("any-folder-id"); + expect(item?.title).toEqual("Any Folder"); + + await updateFolder.execute({ + id: "any-folder-id", + title: "Updated Folder", + slug: "updated-folder", + parentId: "another-id", + permissions: [], + type + }); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + const updatedItem = foldersCache.getItem(folder => folder.id === "any-folder-id"); + + expect(updatedItem).toBeDefined(); + expect(updatedItem?.id).toEqual("any-folder-id"); + expect(updatedItem?.type).toEqual(type); + expect(updatedItem?.title).toEqual("Updated Folder"); + expect(updatedItem?.slug).toEqual("updated-folder"); + expect(updatedItem?.parentId).toEqual("another-id"); + }); + + it("should not update a folder if id is missing", async () => { + const gateway = { + execute: jest.fn().mockResolvedValue(null) + }; + + const updateFolder = UpdateFolder.instance(type, gateway); + + await updateFolder.execute({ + id: "", + title: "Updated Folder", + slug: "updated-folder", + parentId: "another-id", + permissions: [], + type + }); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + const updatedItem = foldersCache.getItem(folder => folder.id === "any-folder-id"); + + expect(updatedItem).toBeDefined(); + expect(updatedItem?.id).toEqual("any-folder-id"); + expect(updatedItem?.type).toEqual(type); + expect(updatedItem?.title).toEqual("Any Folder"); + expect(updatedItem?.slug).toEqual("any-folder"); + expect(updatedItem?.parentId).toEqual(null); + }); + + it("should handle gateway errors gracefully", async () => { + const gateway = { + execute: jest.fn().mockRejectedValue(new Error("Gateway error")) + }; + + const updateFolder = UpdateFolder.instance(type, gateway); + + await expect( + updateFolder.execute({ + id: "any-folder-id", + title: "Updated Folder", + slug: "updated-folder", + parentId: "another-id", + permissions: [], + type + }) + ).rejects.toThrow("Gateway error"); + + expect(gateway.execute).toHaveBeenCalledTimes(1); + }); +}); diff --git a/packages/app-aco/src/features/folder/updateFolder/UpdateFolder.ts b/packages/app-aco/src/features/folder/updateFolder/UpdateFolder.ts new file mode 100644 index 00000000000..fec60dc8be8 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/UpdateFolder.ts @@ -0,0 +1,23 @@ +import { loadingRepositoryFactory } from "@webiny/app-utils"; +import { IUpdateFolderUseCase } from "./IUpdateFolderUseCase"; +import { IUpdateFolderGateway } from "./IUpdateFolderGateway"; +import { UpdateFolderRepository } from "./UpdateFolderRepository"; +import { UpdateFolderUseCase } from "./UpdateFolderUseCase"; +import { UpdateFolderUseCaseWithLoading } from "./UpdateFolderUseCaseWithLoading"; +import { UpdateFolderUseCaseWithoutInheritedPermissions } from "./UpdateFolderUseCaseWithoutInheritedPermissions"; +import { folderCacheFactory } from "../cache"; + +export class UpdateFolder { + public static instance(type: string, gateway: IUpdateFolderGateway): IUpdateFolderUseCase { + const foldersCache = folderCacheFactory.getCache(type); + const loadingRepository = loadingRepositoryFactory.getRepository(type); + const repository = new UpdateFolderRepository(foldersCache, gateway); + const useCase = new UpdateFolderUseCase(repository); + const useCaseWithoutInheritedPermissions = + new UpdateFolderUseCaseWithoutInheritedPermissions(useCase); + return new UpdateFolderUseCaseWithLoading( + loadingRepository, + useCaseWithoutInheritedPermissions + ); + } +} diff --git a/packages/app-aco/src/features/folder/updateFolder/UpdateFolderGqlGateway.ts b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderGqlGateway.ts new file mode 100644 index 00000000000..21855ab4b44 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderGqlGateway.ts @@ -0,0 +1,110 @@ +import ApolloClient from "apollo-client"; +import gql from "graphql-tag"; +import { IUpdateFolderGateway } from "./IUpdateFolderGateway"; +import { FolderDto } from "./FolderDto"; +import { AcoError, FolderItem } from "~/types"; +import { ROOT_FOLDER } from "~/constants"; + +export interface UpdateFolderResponse { + aco: { + updateFolder: { + data: FolderItem; + error: AcoError | null; + }; + }; +} + +export interface UpdateFolderVariables { + id: string; + data: Partial< + Omit< + FolderItem, + "id" | "createdOn" | "createdBy" | "savedOn" | "savedBy" | "modifiedOn" | "modifiedBy" + > + >; +} + +export const UPDATE_FOLDER = gql` + mutation UpdateFolder($id: ID!, $data: FolderUpdateInput!) { + aco { + updateFolder(id: $id, data: $data) { + data { + id + title + slug + permissions { + target + level + inheritedFrom + } + hasNonInheritedPermissions + canManagePermissions + canManageStructure + canManageContent + parentId + type + savedOn + savedBy { + id + displayName + } + createdOn + createdBy { + id + displayName + } + modifiedOn + modifiedBy { + id + displayName + } + } + error { + code + data + message + } + } + } + } +`; + +export class UpdateFolderGqlGateway implements IUpdateFolderGateway { + private client: ApolloClient; + + constructor(client: ApolloClient) { + this.client = client; + } + + async execute(folder: FolderDto) { + const { id, title, slug, permissions, parentId } = folder; + + const { data: response } = await this.client.mutate< + UpdateFolderResponse, + UpdateFolderVariables + >({ + mutation: UPDATE_FOLDER, + variables: { + id, + data: { + title, + slug, + parentId: parentId === ROOT_FOLDER ? null : parentId, + permissions: permissions.filter(p => !p.inheritedFrom) + } + } + }); + + if (!response) { + throw new Error("Network error while updating folder."); + } + + const { data, error } = response.aco.updateFolder; + + if (!data) { + throw new Error(error?.message || "Could not update folder"); + } + + return data; + } +} diff --git a/packages/app-aco/src/features/folder/updateFolder/UpdateFolderRepository.ts b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderRepository.ts new file mode 100644 index 00000000000..e30b7523761 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderRepository.ts @@ -0,0 +1,34 @@ +import { IUpdateFolderRepository } from "./IUpdateFolderRepository"; +import { ListCache } from "../cache"; +import { Folder } from "../Folder"; +import { IUpdateFolderGateway } from "./IUpdateFolderGateway"; +import { FolderDto } from "./FolderDto"; + +export class UpdateFolderRepository implements IUpdateFolderRepository { + private cache: ListCache; + private gateway: IUpdateFolderGateway; + + constructor(cache: ListCache, gateway: IUpdateFolderGateway) { + this.cache = cache; + this.gateway = gateway; + } + + async execute(folder: Folder) { + const dto: FolderDto = { + id: folder.id, + title: folder.title, + slug: folder.slug, + permissions: folder.permissions, + parentId: folder.parentId + }; + + const result = await this.gateway.execute(dto); + this.cache.updateItems(f => { + if (f.id === folder.id) { + return Folder.create(result); + } + + return Folder.create(f); + }); + } +} diff --git a/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCase.ts b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCase.ts new file mode 100644 index 00000000000..f86690f6025 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCase.ts @@ -0,0 +1,24 @@ +import { UpdateFolderParams, IUpdateFolderUseCase } from "./IUpdateFolderUseCase"; +import { IUpdateFolderRepository } from "./IUpdateFolderRepository"; +import { Folder } from "../Folder"; + +export class UpdateFolderUseCase implements IUpdateFolderUseCase { + private repository: IUpdateFolderRepository; + + constructor(repository: IUpdateFolderRepository) { + this.repository = repository; + } + + async execute(params: UpdateFolderParams) { + await this.repository.execute( + Folder.create({ + id: params.id, + title: params.title, + slug: params.slug, + type: params.type, + parentId: params.parentId, + permissions: params.permissions + }) + ); + } +} diff --git a/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCaseWithLoading.ts b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCaseWithLoading.ts new file mode 100644 index 00000000000..6c75936edbc --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCaseWithLoading.ts @@ -0,0 +1,20 @@ +import { ILoadingRepository } from "@webiny/app-utils"; +import { LoadingActionsEnum } from "~/types"; +import { IUpdateFolderUseCase, UpdateFolderParams } from "./IUpdateFolderUseCase"; + +export class UpdateFolderUseCaseWithLoading implements IUpdateFolderUseCase { + private loadingRepository: ILoadingRepository; + private useCase: IUpdateFolderUseCase; + + constructor(loadingRepository: ILoadingRepository, useCase: IUpdateFolderUseCase) { + this.loadingRepository = loadingRepository; + this.useCase = useCase; + } + + async execute(params: UpdateFolderParams) { + await this.loadingRepository.runCallBack( + this.useCase.execute(params), + LoadingActionsEnum.update + ); + } +} diff --git a/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCaseWithoutInheritedPermissions.ts b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCaseWithoutInheritedPermissions.ts new file mode 100644 index 00000000000..02b75e371f3 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/UpdateFolderUseCaseWithoutInheritedPermissions.ts @@ -0,0 +1,19 @@ +import { IUpdateFolderUseCase, UpdateFolderParams } from "./IUpdateFolderUseCase"; + +export class UpdateFolderUseCaseWithoutInheritedPermissions implements IUpdateFolderUseCase { + private useCase: IUpdateFolderUseCase; + + constructor(useCase: IUpdateFolderUseCase) { + this.useCase = useCase; + } + + async execute(params: UpdateFolderParams) { + // We must omit all inherited permissions. + const permissions = params.permissions.filter(p => !p.inheritedFrom); + + await this.useCase.execute({ + ...params, + permissions + }); + } +} diff --git a/packages/app-aco/src/features/folder/updateFolder/index.ts b/packages/app-aco/src/features/folder/updateFolder/index.ts new file mode 100644 index 00000000000..776c694e3e8 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/index.ts @@ -0,0 +1 @@ +export * from "./useUpdateFolder"; diff --git a/packages/app-aco/src/features/folder/updateFolder/useUpdateFolder.ts b/packages/app-aco/src/features/folder/updateFolder/useUpdateFolder.ts new file mode 100644 index 00000000000..962d04a0d69 --- /dev/null +++ b/packages/app-aco/src/features/folder/updateFolder/useUpdateFolder.ts @@ -0,0 +1,35 @@ +import { useCallback, useContext } from "react"; +import { useApolloClient } from "@apollo/react-hooks"; +import { UpdateFolderGqlGateway } from "./UpdateFolderGqlGateway"; +import { UpdateFolder } from "./UpdateFolder"; +import { UpdateFolderParams } from "./IUpdateFolderUseCase"; +import { FoldersContext } from "~/contexts/folders"; + +export const useUpdateFolder = () => { + const client = useApolloClient(); + const gateway = new UpdateFolderGqlGateway(client); + + const foldersContext = useContext(FoldersContext); + + if (!foldersContext) { + throw new Error("useUpdateFolder must be used within a FoldersProvider"); + } + + const { type } = foldersContext; + + if (!type) { + throw Error(`FoldersProvider requires a "type" prop or an AcoAppContext to be available!`); + } + + const updateFolder = useCallback( + (params: UpdateFolderParams) => { + const instance = UpdateFolder.instance(type, gateway); + return instance.execute(params); + }, + [type, gateway] + ); + + return { + updateFolder + }; +}; diff --git a/packages/app-aco/src/features/index.ts b/packages/app-aco/src/features/index.ts new file mode 100644 index 00000000000..40d34a8d6d6 --- /dev/null +++ b/packages/app-aco/src/features/index.ts @@ -0,0 +1 @@ +export * from "./folder"; diff --git a/packages/app-aco/src/graphql/folders.gql.ts b/packages/app-aco/src/graphql/folders.gql.ts deleted file mode 100644 index add1cece388..00000000000 --- a/packages/app-aco/src/graphql/folders.gql.ts +++ /dev/null @@ -1,98 +0,0 @@ -import gql from "graphql-tag"; - -const ERROR_FIELD = /* GraphQL */ ` - { - code - data - message - } -`; - -const DATA_FIELD = /* GraphQL */ ` - { - id - title - slug - permissions { - target - level - inheritedFrom - } - hasNonInheritedPermissions - canManagePermissions - canManageStructure - canManageContent - parentId - type - savedOn - savedBy { - id - displayName - } - createdOn - createdBy { - id - displayName - } - modifiedOn - modifiedBy { - id - displayName - } - } -`; - -export const CREATE_FOLDER = gql` - mutation CreateFolder($data: FolderCreateInput!) { - aco { - createFolder(data: $data) { - data ${DATA_FIELD} - error ${ERROR_FIELD} - } - } - } -`; - -export const LIST_FOLDERS = gql` - query ListFolders ($type: String!, $limit: Int!) { - aco { - listFolders(where: { type: $type }, limit: $limit) { - data ${DATA_FIELD} - error ${ERROR_FIELD} - } - } - } -`; - -export const GET_FOLDER = gql` - query GetFolder ($id: ID!) { - aco { - getFolder(id: $id) { - data ${DATA_FIELD} - error ${ERROR_FIELD} - } - } - } -`; - -export const UPDATE_FOLDER = gql` - mutation UpdateFolder($id: ID!, $data: FolderUpdateInput!) { - aco { - updateFolder(id: $id, data: $data) { - data ${DATA_FIELD} - error ${ERROR_FIELD} - } - } - } -`; - -export const DELETE_FOLDER = gql` - mutation DeleteFolder($id: ID!) { - aco { - deleteFolder(id: $id) { - data - error ${ERROR_FIELD} - } - } - } -`; diff --git a/packages/app-aco/src/hooks/index.ts b/packages/app-aco/src/hooks/index.ts index a25f3b8d2b4..60907b72441 100644 --- a/packages/app-aco/src/hooks/index.ts +++ b/packages/app-aco/src/hooks/index.ts @@ -5,4 +5,3 @@ export * from "./useFolders"; export * from "./useRecords"; export * from "./useTags"; export * from "./useNavigateFolder"; -export { useFoldersApi } from "../contexts/FoldersApi"; diff --git a/packages/app-aco/src/hooks/useFolders.ts b/packages/app-aco/src/hooks/useFolders.ts index b8d6282a243..a493b1c546e 100644 --- a/packages/app-aco/src/hooks/useFolders.ts +++ b/packages/app-aco/src/hooks/useFolders.ts @@ -1,40 +1,45 @@ -import { useContext, useEffect, useMemo } from "react"; -import { FoldersContext } from "~/contexts/folders"; +import { + useCreateFolder, + useDeleteFolder, + useGetDescendantFolders, + useGetFolder, + useGetFolderLevelPermission, + useListFolders, + useUpdateFolder +} from "~/features"; +/** + * Custom hook to manage folder operations. + * + * @deprecated This hook is deprecated. Use the individual hooks directly from "~/features" instead. + */ export const useFolders = () => { - const context = useContext(FoldersContext); - if (!context) { - throw new Error("useFolders must be used within a FoldersProvider"); - } + const { createFolder } = useCreateFolder(); + const { deleteFolder } = useDeleteFolder(); + const { listFolders, folders, loading } = useListFolders(); + const { updateFolder } = useUpdateFolder(); + const { getDescendantFolders } = useGetDescendantFolders(); + const { getFolder } = useGetFolder(); + const { getFolderLevelPermission: canManageStructure } = + useGetFolderLevelPermission("canManageStructure"); + const { getFolderLevelPermission: canManagePermissions } = + useGetFolderLevelPermission("canManagePermissions"); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); - const { folders, loading, listFolders, ...other } = context; - - useEffect(() => { - /** - * On first mount, call `listFolders`, which will either issue a network request, or load folders from cache. - * We don't need to store the result of it to any local state; that is managed by the context provider. - * - * IMPORTANT: we check if the folders array exists: the hook can be used from multiple components and - * fetch the outdated list from Apollo Cache. Since the state is managed locally, we fetch the folders only - * at the first mount. - */ - if (folders) { - return; + return { + folders, + loading, + listFolders, + getFolder, + getDescendantFolders, + createFolder, + updateFolder, + deleteFolder, + folderLevelPermissions: { + canManageStructure, + canManagePermissions, + canManageContent } - listFolders(); - }, []); - - return useMemo( - () => ({ - /** - * NOTE: do NOT expose listFolders from this hook, because you already have folders in the `folders` property. - * You'll never need to call `listFolders` from any component. As soon as you call `useFolders()`, you'll initiate - * fetching of `folders`, which is managed by the FoldersContext. - */ - loading, - folders, - ...other - }), - [folders, loading] - ); + }; }; diff --git a/packages/app-aco/src/index.ts b/packages/app-aco/src/index.ts index c7c297a3415..34b2023afd9 100644 --- a/packages/app-aco/src/index.ts +++ b/packages/app-aco/src/index.ts @@ -2,6 +2,6 @@ export * from "./components"; export * from "./config"; export * from "./contexts"; export * from "./hooks"; +export * from "./features"; export * from "./dialogs"; -export * from "./Folders"; export * from "./sorting"; diff --git a/packages/app-aco/src/types.ts b/packages/app-aco/src/types.ts index 73ef6344160..1f3c61190fc 100644 --- a/packages/app-aco/src/types.ts +++ b/packages/app-aco/src/types.ts @@ -67,6 +67,17 @@ export type LoadingActions = | "DELETE" | "MOVE"; +export enum LoadingActionsEnum { + init = "INIT", + list = "LIST", + listMore = "LIST_MORE", + get = "GET", + create = "CREATE", + update = "UPDATE", + delete = "DELETE", + move = "MOVE" +} + export interface AcoError { code: string; message: string; @@ -142,7 +153,17 @@ export interface CreateFolderResponse { export interface CreateFolderVariables { data: Omit< FolderItem, - "id" | "createdOn" | "createdBy" | "savedOn" | "savedBy" | "modifiedOn" | "modifiedBy" + | "id" + | "createdOn" + | "createdBy" + | "savedOn" + | "savedBy" + | "modifiedOn" + | "modifiedBy" + | "hasNonInheritedPermissions" + | "canManageContent" + | "canManagePermissions" + | "canManageStructure" >; } diff --git a/packages/app-aco/tsconfig.build.json b/packages/app-aco/tsconfig.build.json index e9fd945cba2..bda102d5c14 100644 --- a/packages/app-aco/tsconfig.build.json +++ b/packages/app-aco/tsconfig.build.json @@ -6,6 +6,7 @@ { "path": "../app-admin/tsconfig.build.json" }, { "path": "../app-headless-cms-common/tsconfig.build.json" }, { "path": "../app-security/tsconfig.build.json" }, + { "path": "../app-utils/tsconfig.build.json" }, { "path": "../app-wcp/tsconfig.build.json" }, { "path": "../form/tsconfig.build.json" }, { "path": "../plugins/tsconfig.build.json" }, diff --git a/packages/app-aco/tsconfig.json b/packages/app-aco/tsconfig.json index 83c93d96212..f7cb73a64e3 100644 --- a/packages/app-aco/tsconfig.json +++ b/packages/app-aco/tsconfig.json @@ -6,6 +6,7 @@ { "path": "../app-admin" }, { "path": "../app-headless-cms-common" }, { "path": "../app-security" }, + { "path": "../app-utils" }, { "path": "../app-wcp" }, { "path": "../form" }, { "path": "../plugins" }, @@ -30,6 +31,8 @@ "@webiny/app-headless-cms-common": ["../app-headless-cms-common/src"], "@webiny/app-security/*": ["../app-security/src/*"], "@webiny/app-security": ["../app-security/src"], + "@webiny/app-utils/*": ["../app-utils/src/*"], + "@webiny/app-utils": ["../app-utils/src"], "@webiny/app-wcp/*": ["../app-wcp/src/*"], "@webiny/app-wcp": ["../app-wcp/src"], "@webiny/form/*": ["../form/src/*"], diff --git a/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/FileManagerViewContext.tsx b/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/FileManagerViewContext.tsx index 38545cfbee6..af9731b1560 100644 --- a/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/FileManagerViewContext.tsx +++ b/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/FileManagerViewContext.tsx @@ -8,7 +8,7 @@ import { initializeState, State } from "./state"; import { FolderItem, ListMeta, ListSearchRecordsSort } from "@webiny/app-aco/types"; import { UploadOptions } from "@webiny/app/types"; import { sortTableItems } from "@webiny/app-aco/sorting"; -import { useFolders, useNavigateFolder } from "@webiny/app-aco"; +import { useListFolders, useNavigateFolder } from "@webiny/app-aco"; import { ListFilesQueryVariables } from "~/modules/FileManagerApiProvider/graphql"; import { useListFiles } from "./useListFiles"; import { useTags } from "./useTags"; @@ -106,7 +106,7 @@ export const FileManagerViewProvider = ({ children, ...props }: FileManagerViewP const shiftKeyPressed = useShiftKey(); const modifiers = { scope: props.scope, own: props.own, accept: props.accept }; const fileManager = useFileManagerApi(); - const { folders: originalFolders, loading: foldersLoading } = useFolders(); + const { folders: originalFolders, loading: foldersLoading } = useListFolders(); const { currentFolderId = ROOT_FOLDER, navigateToFolder } = useNavigateFolder(); const tags = useTags(modifiers); const [state, setState] = useStateIfMounted(initializeState()); diff --git a/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/useListFiles.ts b/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/useListFiles.ts index 401b4f8128f..fcbde481908 100644 --- a/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/useListFiles.ts +++ b/packages/app-file-manager/src/modules/FileManagerRenderer/FileManagerViewProvider/useListFiles.ts @@ -1,6 +1,6 @@ import isEqual from "lodash/isEqual"; import { validateOrGetDefaultDbSort } from "@webiny/app-aco/sorting"; -import { useFolders } from "@webiny/app-aco"; +import { useGetDescendantFolders } from "@webiny/app-aco"; import { ListMeta } from "@webiny/app-aco/types"; import { useSecurity } from "@webiny/app-security"; import { FileItem } from "@webiny/app-admin/types"; @@ -47,7 +47,7 @@ const defaultMeta: ListMeta = { export function useListFiles({ modifiers, folderId, state }: UseListFilesParams) { const { identity } = useSecurity(); const fileManager = useFileManagerApi(); - const { getDescendantFolders } = useFolders(); + const { getDescendantFolders } = useGetDescendantFolders(); const [meta, setMeta] = useStateIfMounted(defaultMeta); const [files, setFiles] = useStateIfMounted([]); const [loading, setLoading] = useStateIfMounted>({}); diff --git a/packages/app-headless-cms/src/admin/views/contentEntries/Table/Main.tsx b/packages/app-headless-cms/src/admin/views/contentEntries/Table/Main.tsx index c25540eef1f..3fa59331597 100644 --- a/packages/app-headless-cms/src/admin/views/contentEntries/Table/Main.tsx +++ b/packages/app-headless-cms/src/admin/views/contentEntries/Table/Main.tsx @@ -1,6 +1,6 @@ import React, { useCallback, useEffect, useMemo, useRef, useState } from "react"; import debounce from "lodash/debounce"; -import { useCreateDialog, useFolders } from "@webiny/app-aco"; +import { useCreateDialog, useGetFolderLevelPermission } from "@webiny/app-aco"; import { Scrollbar } from "@webiny/ui/Scrollbar"; import { Empty } from "~/admin/components/ContentEntries/Empty"; import { Filters } from "~/admin/components/ContentEntries/Filters"; @@ -30,15 +30,18 @@ export const Main = ({ folderId: initialFolderId }: MainProps) => { // We check permissions on two layers - security and folder level permissions. const { canCreate, contentModel } = useContentEntry(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); + const { getFolderLevelPermission: canManageStructure } = + useGetFolderLevelPermission("canManageStructure"); const canCreateFolder = useMemo(() => { - return flp.canManageStructure(folderId); - }, [flp, folderId]); + return canManageStructure(folderId); + }, [canManageStructure, folderId]); const canCreateContent = useMemo(() => { - return canCreate && flp.canManageContent(folderId); - }, [flp, folderId]); + return canCreate && canManageContent(folderId); + }, [canManageContent, folderId]); const createEntry = useCallback(() => { const folder = folderId ? `&folderId=${encodeURIComponent(folderId)}` : ""; diff --git a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDelete.tsx b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDelete.tsx index 8af05ee53f3..225683e4aba 100644 --- a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDelete.tsx +++ b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDelete.tsx @@ -1,5 +1,5 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { observer } from "mobx-react-lite"; import { PageListConfig } from "~/admin/config/pages"; import { usePagesPermissions } from "~/hooks/permissions"; @@ -7,18 +7,17 @@ import { ActionDelete as ActionDeleteBase } from "~/admin/components/BulkActions export const SecureActionDelete = observer(() => { const { canDelete } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { useWorker } = PageListConfig.Browser.BulkAction; const worker = useWorker(); const canDeleteAll = useMemo(() => { return worker.items.every(item => { - return ( - canDelete(item.data.createdBy.id) && flp.canManageContent(item.location?.folderId) - ); + return canDelete(item.data.createdBy.id) && canManageContent(item.location?.folderId); }); - }, [worker.items]); + }, [worker.items, canManageContent]); if (!canDeleteAll) { console.log("You don't have permissions to delete pages."); diff --git a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDuplicate.tsx b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDuplicate.tsx index 527d7f2903d..2e333ddac8d 100644 --- a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDuplicate.tsx +++ b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionDuplicate.tsx @@ -1,5 +1,5 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { observer } from "mobx-react-lite"; import { PageListConfig } from "~/admin/config/pages"; import { usePagesPermissions } from "~/hooks/permissions"; @@ -8,7 +8,8 @@ import { ActionDuplicate } from "~/admin/components/BulkActions/ActionDuplicate" export const SecureActionDuplicate = ActionDuplicate.createDecorator(Original => { return observer(() => { const { canWrite: pagesCanWrite } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { useWorker } = PageListConfig.Browser.BulkAction; const worker = useWorker(); @@ -17,10 +18,10 @@ export const SecureActionDuplicate = ActionDuplicate.createDecorator(Original => return worker.items.every(item => { return ( pagesCanWrite(item.data.createdBy.id) && - flp.canManageContent(item.location?.folderId) + canManageContent(item.location?.folderId) ); }); - }, [worker.items]); + }, [worker.items, canManageContent]); if (!canDuplicateAll) { console.log("You don't have permissions to duplicate pages."); diff --git a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionMove.tsx b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionMove.tsx index 733d0ca65e1..0e12c532ed8 100644 --- a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionMove.tsx +++ b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionMove.tsx @@ -1,5 +1,5 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { observer } from "mobx-react-lite"; import { PageListConfig } from "~/admin/config/pages"; import { ActionMove as ActionMoveBase } from "~/admin/components/BulkActions"; @@ -7,13 +7,14 @@ import { ActionMove as ActionMoveBase } from "~/admin/components/BulkActions"; export const SecureActionMove = observer(() => { const { useWorker } = PageListConfig.Browser.BulkAction; const worker = useWorker(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const canMoveAll = useMemo(() => { return worker.items.every(item => { - return flp.canManageContent(item.location?.folderId); + return canManageContent(item.location?.folderId); }); - }, [worker.items]); + }, [worker.items, canManageContent]); if (!canMoveAll) { return null; diff --git a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionPublish.tsx b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionPublish.tsx index 051d25e6d8d..80dc43dbd92 100644 --- a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionPublish.tsx +++ b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionPublish.tsx @@ -1,5 +1,5 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { observer } from "mobx-react-lite"; import { PageListConfig } from "~/admin/config/pages"; import { usePagesPermissions } from "~/hooks/permissions"; @@ -8,16 +8,17 @@ import { ActionPublish as ActionPublishBase } from "~/admin/components/BulkActio export const SecureActionPublish = observer(() => { const { canPublish } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { useWorker } = PageListConfig.Browser.BulkAction; const worker = useWorker(); const canPublishAll = useMemo(() => { return worker.items.every(item => { - return canPublish() && flp.canManageContent(item.location?.folderId); + return canPublish() && canManageContent(item.location?.folderId); }); - }, [worker.items]); + }, [worker.items, canManageContent]); if (!canPublishAll) { console.log("You don't have permissions to publish pages."); diff --git a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionUnpublish.tsx b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionUnpublish.tsx index 9683f12a73f..6025b05b730 100644 --- a/packages/app-page-builder/src/admin/components/BulkActions/SecureActionUnpublish.tsx +++ b/packages/app-page-builder/src/admin/components/BulkActions/SecureActionUnpublish.tsx @@ -1,13 +1,14 @@ import React, { useMemo } from "react"; import { observer } from "mobx-react-lite"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { PageListConfig } from "~/admin/config/pages"; import { usePagesPermissions } from "~/hooks/permissions"; import { ActionUnpublish as ActionUnpublishBase } from "~/admin/components/BulkActions"; export const SecureActionUnpublish = observer(() => { const { canUnpublish } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { useWorker } = PageListConfig.Browser.BulkAction; @@ -15,9 +16,9 @@ export const SecureActionUnpublish = observer(() => { const canUnpublishAll = useMemo(() => { return worker.items.every(item => { - return canUnpublish() && flp.canManageContent(item.location?.folderId); + return canUnpublish() && canManageContent(item.location?.folderId); }); - }, [worker.items]); + }, [worker.items, canManageContent]); if (!canUnpublishAll) { console.log("You don't have permissions to unpublish pages."); diff --git a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureChangePageStatus.tsx b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureChangePageStatus.tsx index 13a2ba0618e..46cc3d4e423 100644 --- a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureChangePageStatus.tsx +++ b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureChangePageStatus.tsx @@ -1,12 +1,13 @@ import React from "react"; import { usePage } from "~/admin/views/Pages/hooks/usePage"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { ChangePageStatus } from "./ChangePageStatus"; export const SecureChangePageStatus = () => { const { page } = usePage(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { hasPermissions, canPublish, canUnpublish } = usePagesPermissions(); if (!hasPermissions()) { @@ -14,7 +15,7 @@ export const SecureChangePageStatus = () => { } const { folderId } = page.location; - if (!flp.canManageContent(folderId)) { + if (!canManageContent(folderId)) { return null; } diff --git a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDeletePage.tsx b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDeletePage.tsx index e06a04da33e..fc1672a286f 100644 --- a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDeletePage.tsx +++ b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDeletePage.tsx @@ -1,18 +1,19 @@ import React, { useMemo } from "react"; import { usePage } from "~/admin/views/Pages/hooks/usePage"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { DeletePage } from "./DeletePage"; export const SecureDeletePage = () => { const { page } = usePage(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { canDelete: pagesCanDelete } = usePagesPermissions(); const { folderId } = page.location; const canDelete = useMemo(() => { - return pagesCanDelete(page.data.createdBy.id) && flp.canManageContent(folderId); - }, [flp, folderId]); + return pagesCanDelete(page.data.createdBy.id) && canManageContent(folderId); + }, [canManageContent, folderId]); if (!canDelete) { return null; diff --git a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDuplicatePage.tsx b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDuplicatePage.tsx index 9af2fdbcbe0..c894783d806 100644 --- a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDuplicatePage.tsx +++ b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureDuplicatePage.tsx @@ -1,5 +1,5 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { usePagesPermissions } from "~/hooks/permissions"; import { usePage } from "~/admin/views/Pages/hooks/usePage"; import { DuplicatePage } from "./DuplicatePage"; @@ -7,14 +7,15 @@ import { DuplicatePage } from "./DuplicatePage"; export const SecureDuplicatePage = DuplicatePage.createDecorator(Original => { return function SecureDuplicatePageRenderer() { const { page } = usePage(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { canWrite: pagesCanWrite } = usePagesPermissions(); const { folderId } = page.location; const canDuplicate = useMemo(() => { - return pagesCanWrite(page.data.createdBy.id) && flp.canManageContent(folderId); - }, [flp, folderId]); + return pagesCanWrite(page.data.createdBy.id) && canManageContent(folderId); + }, [canManageContent, folderId]); if (!canDuplicate) { return null; diff --git a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureEditPage.tsx b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureEditPage.tsx index e6a342f059e..79a4a7127d4 100644 --- a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureEditPage.tsx +++ b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureEditPage.tsx @@ -1,18 +1,19 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { usePage } from "~/admin/views/Pages/hooks/usePage"; import { usePagesPermissions } from "~/hooks/permissions"; import { EditPage } from "./EditPage"; export const SecureEditPage = () => { const { page } = usePage(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { canUpdate: pagesCanUpdate } = usePagesPermissions(); const { folderId } = page.location; const canEdit = useMemo(() => { - return pagesCanUpdate(page.data.createdBy.id) && flp.canManageContent(folderId); - }, [flp, folderId]); + return pagesCanUpdate(page.data.createdBy.id) && canManageContent(folderId); + }, [canManageContent, folderId]); if (!canEdit) { return null; diff --git a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureMovePage.tsx b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureMovePage.tsx index eece9aa980a..515a1c3bfeb 100644 --- a/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureMovePage.tsx +++ b/packages/app-page-builder/src/admin/components/Table/Table/Actions/SecureMovePage.tsx @@ -1,17 +1,17 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { usePage } from "~/admin/views/Pages/hooks/usePage"; import { MovePage } from "./MovePage"; export const SecureMovePage = () => { const { page } = usePage(); - - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { folderId } = page.location; const canMove = useMemo(() => { - return flp.canManageContent(folderId); - }, [flp, folderId]); + return canManageContent(folderId); + }, [canManageContent, folderId]); if (!canMove) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/header/deletePage/DeletePage.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/header/deletePage/DeletePage.tsx index 089fa129c45..7c7578f3b41 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/header/deletePage/DeletePage.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/header/deletePage/DeletePage.tsx @@ -4,7 +4,7 @@ import { Tooltip } from "@webiny/ui/Tooltip"; import { ReactComponent as DeleteIcon } from "~/admin/assets/delete.svg"; import { usePagesPermissions } from "~/hooks/permissions"; import { useDeletePage } from "~/admin/views/Pages/hooks/useDeletePage"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { makeDecoratable } from "@webiny/react-composition"; import { usePage } from "~/admin/views/Pages/PageDetails"; @@ -15,14 +15,15 @@ export interface DeletePageProps { const DeletePage = makeDecoratable("DeletePage", (props: DeletePageProps) => { const { onDelete } = props; const { page } = usePage(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { canDelete: pagesCanDelete } = usePagesPermissions(); const { openDialogDeletePage } = useDeletePage({ page, onDelete }); const folderId = page.wbyAco_location?.folderId; const canDelete = useMemo(() => { - return pagesCanDelete(page.createdBy?.id) && flp.canManageContent(folderId); - }, [flp, folderId]); + return pagesCanDelete(page.createdBy?.id) && canManageContent(folderId); + }, [canManageContent, folderId]); if (!canDelete) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/header/editRevision/EditRevision.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/header/editRevision/EditRevision.tsx index 889ada66ccf..f88a42b88d8 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/header/editRevision/EditRevision.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/header/editRevision/EditRevision.tsx @@ -9,7 +9,7 @@ import { i18n } from "@webiny/app/i18n"; import { useMutation } from "@apollo/react-hooks"; import { usePagesPermissions } from "~/hooks/permissions"; import { useNavigatePage } from "~/admin/hooks/useNavigatePage"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { usePage } from "~/admin/views/Pages/PageDetails"; import { makeDecoratable } from "@webiny/react-composition"; @@ -17,7 +17,8 @@ const t = i18n.ns("app-headless-cms/app-page-builder/page-details/header/edit"); const EditRevision = makeDecoratable("EditRevision", () => { const { canUpdate: pagesCanUpdate } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const [inProgress, setInProgress] = useState(); const { showSnackbar } = useSnackbar(); const [createPageFrom] = useMutation(CREATE_PAGE); @@ -46,8 +47,8 @@ const EditRevision = makeDecoratable("EditRevision", () => { const folderId = page.wbyAco_location?.folderId; const canEdit = useMemo(() => { - return pagesCanUpdate(page.createdBy?.id) && flp.canManageContent(folderId); - }, [flp, folderId]); + return pagesCanUpdate(page.createdBy?.id) && canManageContent(folderId); + }, [canManageContent, folderId]); if (!canEdit) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/DuplicatePage/SecureDuplicatePage.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/DuplicatePage/SecureDuplicatePage.tsx index 855a1d4b5ab..77245349ea4 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/DuplicatePage/SecureDuplicatePage.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/DuplicatePage/SecureDuplicatePage.tsx @@ -1,5 +1,5 @@ import React, { useMemo } from "react"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { usePage } from "~/admin/views/Pages/PageDetails"; import { usePagesPermissions } from "~/hooks/permissions"; import { DuplicatePage } from "./DuplicatePage"; @@ -7,7 +7,8 @@ import { DuplicatePage } from "./DuplicatePage"; export const SecureDuplicatePage = DuplicatePage.createDecorator(Original => { return function SecurePageDetailsDuplicatePageRenderer() { const { page } = usePage(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { canWrite: pagesCanWrite } = usePagesPermissions(); const canDuplicate = useMemo(() => { @@ -17,10 +18,9 @@ export const SecureDuplicatePage = DuplicatePage.createDecorator(Original => { } return ( - pagesCanWrite(page.createdBy.id) && - flp.canManageContent(page.wbyAco_location.folderId) + pagesCanWrite(page.createdBy.id) && canManageContent(page.wbyAco_location.folderId) ); - }, [flp, page]); + }, [canManageContent, page]); if (!canDuplicate) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/PageOptionsMenu.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/PageOptionsMenu.tsx index 406ac6a4d3e..ad6513dc471 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/PageOptionsMenu.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/header/pageOptionsMenu/PageOptionsMenu.tsx @@ -16,7 +16,7 @@ import { plugins } from "@webiny/plugins"; import { PbPageData, PbPageDetailsHeaderRightOptionsMenuItemPlugin, PbPageTemplate } from "~/types"; import { SecureView } from "@webiny/app-security"; import { useAdminPageBuilder } from "~/admin/hooks/useAdminPageBuilder"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { useTemplatesPermissions } from "~/hooks/permissions"; import { PreviewPage } from "./PreviewPage"; import { DuplicatePage } from "./DuplicatePage"; @@ -71,13 +71,14 @@ const PageOptionsMenu = (props: PageOptionsMenuProps) => { [page] ); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { canCreate: templatesCanCreate } = useTemplatesPermissions(); const canCreateTemplate = templatesCanCreate(); const folderId = page.wbyAco_location?.folderId; - const flpCanManageContent = flp.canManageContent(folderId); + const flpCanManageContent = canManageContent(folderId); const isTemplatePage = page.content?.data?.template; return ( diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/header/publishRevision/PublishRevision.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/header/publishRevision/PublishRevision.tsx index 6c1b1c70926..73f7988a011 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/header/publishRevision/PublishRevision.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/header/publishRevision/PublishRevision.tsx @@ -7,7 +7,7 @@ import { ReactComponent as PublishIcon } from "@material-design-icons/svg/round/ import { ReactComponent as UnpublishIcon } from "@material-design-icons/svg/round/settings_backup_restore.svg"; import { makeDecoratable } from "@webiny/app-admin"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { usePage } from "~/admin/views/Pages/PageDetails"; import { usePublishRevisionHandler } from "../../pageRevisions/usePublishRevisionHandler"; @@ -15,7 +15,8 @@ const t = i18n.ns("app-headless-cms/app-page-builder/page-details/header/publish const PublishRevision = () => { const { canPublish, canUnpublish, hasPermissions } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const { page } = usePage(); const { publishRevision, unpublishRevision } = usePublishRevisionHandler(); @@ -45,7 +46,7 @@ const PublishRevision = () => { }); const folderId = page.wbyAco_location?.folderId; - if (!hasPermissions() || !flp.canManageContent(folderId)) { + if (!hasPermissions() || !canManageContent(folderId)) { return null; } diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureDeleteRevisionMenuOption.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureDeleteRevisionMenuOption.tsx index cf50324561c..646a7e4fb44 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureDeleteRevisionMenuOption.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureDeleteRevisionMenuOption.tsx @@ -1,6 +1,6 @@ import React, { useMemo } from "react"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { DeleteRevisionMenuOption, @@ -10,14 +10,14 @@ import { export const SecureDeleteRevisionMenuOption = (props: DeleteRevisionMenuOptionProps) => { const { page } = props; const { canDelete: pagesCanDelete } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const hasAccess = useMemo(() => { return ( - pagesCanDelete(page?.createdBy?.id) && - flp.canManageContent(page.wbyAco_location?.folderId) + pagesCanDelete(page?.createdBy?.id) && canManageContent(page.wbyAco_location?.folderId) ); - }, [page]); + }, [page, canManageContent]); if (!hasAccess) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureEditRevisionMenuOption.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureEditRevisionMenuOption.tsx index 0e6bccc1f44..193e4de88a5 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureEditRevisionMenuOption.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureEditRevisionMenuOption.tsx @@ -1,20 +1,20 @@ import React, { useMemo } from "react"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { EditRevisionMenuOption, EditRevisionMenuOptionProps } from "./EditRevisionMenuOption"; export const SecureEditRevisionMenuOption = (props: EditRevisionMenuOptionProps) => { const { page } = props; const { canUpdate: pagesCanUpdate } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const hasAccess = useMemo(() => { return ( - pagesCanUpdate(page?.createdBy?.id) && - flp.canManageContent(page.wbyAco_location?.folderId) + pagesCanUpdate(page?.createdBy?.id) && canManageContent(page.wbyAco_location?.folderId) ); - }, [page]); + }, [page, canManageContent]); if (!hasAccess) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureNewRevisionFromCurrent.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureNewRevisionFromCurrent.tsx index 409fa88e58b..b0b02da0fb9 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureNewRevisionFromCurrent.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureNewRevisionFromCurrent.tsx @@ -1,17 +1,18 @@ import React, { useMemo } from "react"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { NewRevisionFromCurrent, NewRevisionFromCurrentProps } from "./NewRevisionFromCurrent"; export const SecureNewRevisionFromCurrent = (props: NewRevisionFromCurrentProps) => { const { page } = props; const { canCreate: pagesCanCreate } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const hasAccess = useMemo(() => { - return pagesCanCreate() && flp.canManageContent(page.wbyAco_location?.folderId); - }, [page]); + return pagesCanCreate() && canManageContent(page.wbyAco_location?.folderId); + }, [page, canManageContent]); if (!hasAccess) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecurePublishPageMenuOption.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecurePublishPageMenuOption.tsx index 90646e2cb29..9310ee05826 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecurePublishPageMenuOption.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecurePublishPageMenuOption.tsx @@ -1,6 +1,6 @@ import React, { useMemo } from "react"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { PublishPageMenuOptionProps, PublishPageMenuOption } from "./PublishPageMenuOption"; import { usePage } from "~/admin/views/Pages/PageDetails"; @@ -8,11 +8,12 @@ import { usePage } from "~/admin/views/Pages/PageDetails"; export const SecurePublishPageMenuOption = (props: PublishPageMenuOptionProps) => { const { page } = usePage(); const { canPublish: pagesCanPublish } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const hasAccess = useMemo(() => { - return pagesCanPublish() && flp.canManageContent(page.wbyAco_location?.folderId); - }, [page]); + return pagesCanPublish() && canManageContent(page.wbyAco_location?.folderId); + }, [page, canManageContent]); if (!hasAccess) { return null; diff --git a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureUnpublishPageMenuOption.tsx b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureUnpublishPageMenuOption.tsx index e5bd121e5a3..6e130ca81ce 100644 --- a/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureUnpublishPageMenuOption.tsx +++ b/packages/app-page-builder/src/admin/plugins/pageDetails/pageRevisions/MenuOptions/SecureUnpublishPageMenuOption.tsx @@ -1,17 +1,18 @@ import React, { useMemo } from "react"; import { usePagesPermissions } from "~/hooks/permissions"; -import { useFolders } from "@webiny/app-aco"; +import { useGetFolderLevelPermission } from "@webiny/app-aco"; import { UnpublishPageMenuOption, UnpublishPageMenuOptionProps } from "./UnpublishPageMenuOption"; export const SecureUnpublishPageMenuOption = (props: UnpublishPageMenuOptionProps) => { const { page } = props; const { canUnpublish: pagesCanUnpublish } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const hasAccess = useMemo(() => { - return pagesCanUnpublish() && flp.canManageContent(page.wbyAco_location?.folderId); - }, [page]); + return pagesCanUnpublish() && canManageContent(page.wbyAco_location?.folderId); + }, [page, canManageContent]); if (!hasAccess) { return null; diff --git a/packages/app-page-builder/src/admin/views/Pages/Table/Main.tsx b/packages/app-page-builder/src/admin/views/Pages/Table/Main.tsx index bdc654fda97..fdbde7f3998 100644 --- a/packages/app-page-builder/src/admin/views/Pages/Table/Main.tsx +++ b/packages/app-page-builder/src/admin/views/Pages/Table/Main.tsx @@ -1,7 +1,7 @@ import React, { useCallback, useEffect, useMemo, useRef, useState } from "react"; import debounce from "lodash/debounce"; import { i18n } from "@webiny/app/i18n"; -import { useCreateDialog, useFolders } from "@webiny/app-aco"; +import { useCreateDialog, useGetFolderLevelPermission } from "@webiny/app-aco"; import { CircularProgress } from "@webiny/ui/Progress"; import { Scrollbar } from "@webiny/ui/Scrollbar"; import CategoriesDialog from "~/admin/views/Categories/CategoriesDialog"; @@ -46,15 +46,18 @@ export const Main = ({ folderId: initialFolderId }: Props) => { // We check permissions on two layers - security and folder level permissions. const { canCreate } = usePagesPermissions(); - const { folderLevelPermissions: flp } = useFolders(); + const { getFolderLevelPermission: canManageStructure } = + useGetFolderLevelPermission("canManageStructure"); + const { getFolderLevelPermission: canManageContent } = + useGetFolderLevelPermission("canManageContent"); const canCreateFolder = useMemo(() => { - return flp.canManageStructure(folderId); - }, [flp, folderId]); + return canManageStructure(folderId); + }, [canManageStructure, folderId]); const canCreateContent = useMemo(() => { - return canCreate() && flp.canManageContent(folderId); - }, [flp, folderId]); + return canCreate() && canManageContent(folderId); + }, [canManageContent, folderId]); const { innerHeight: windowHeight } = window; const [tableHeight, setTableHeight] = useState(0); diff --git a/packages/app-serverless-cms/package.json b/packages/app-serverless-cms/package.json index 570d83d3c75..843085c84c8 100644 --- a/packages/app-serverless-cms/package.json +++ b/packages/app-serverless-cms/package.json @@ -11,7 +11,6 @@ "dependencies": { "@emotion/react": "11.10.8", "@webiny/app": "0.0.0", - "@webiny/app-aco": "0.0.0", "@webiny/app-admin": "0.0.0", "@webiny/app-admin-rmwc": "0.0.0", "@webiny/app-apw": "0.0.0", diff --git a/packages/app-serverless-cms/src/Admin.tsx b/packages/app-serverless-cms/src/Admin.tsx index c092173b7ca..b3c090f3040 100644 --- a/packages/app-serverless-cms/src/Admin.tsx +++ b/packages/app-serverless-cms/src/Admin.tsx @@ -28,7 +28,6 @@ import { AuditLogs } from "@webiny/app-audit-logs"; import { LexicalEditorPlugin } from "@webiny/lexical-editor-pb-element"; import { LexicalEditorActions } from "@webiny/lexical-editor-actions"; import { Module as MailerSettings } from "@webiny/app-mailer"; -import { Folders } from "@webiny/app-aco"; import { Websockets } from "@webiny/app-websockets"; import { RecordLocking } from "@webiny/app-record-locking"; import { TrashBinConfigs } from "@webiny/app-trash-bin"; @@ -52,7 +51,6 @@ const App = (props: AdminProps) => { - diff --git a/packages/app-serverless-cms/tsconfig.build.json b/packages/app-serverless-cms/tsconfig.build.json index 421e090f48e..56c04020d82 100644 --- a/packages/app-serverless-cms/tsconfig.build.json +++ b/packages/app-serverless-cms/tsconfig.build.json @@ -3,7 +3,6 @@ "include": ["src"], "references": [ { "path": "../app/tsconfig.build.json" }, - { "path": "../app-aco/tsconfig.build.json" }, { "path": "../app-admin/tsconfig.build.json" }, { "path": "../app-admin-rmwc/tsconfig.build.json" }, { "path": "../app-apw/tsconfig.build.json" }, diff --git a/packages/app-serverless-cms/tsconfig.json b/packages/app-serverless-cms/tsconfig.json index 3838a1c5e85..109cb24a6d7 100644 --- a/packages/app-serverless-cms/tsconfig.json +++ b/packages/app-serverless-cms/tsconfig.json @@ -3,7 +3,6 @@ "include": ["src", "__tests__"], "references": [ { "path": "../app" }, - { "path": "../app-aco" }, { "path": "../app-admin" }, { "path": "../app-admin-rmwc" }, { "path": "../app-apw" }, @@ -38,8 +37,6 @@ "~tests/*": ["./__tests__/*"], "@webiny/app/*": ["../app/src/*"], "@webiny/app": ["../app/src"], - "@webiny/app-aco/*": ["../app-aco/src/*"], - "@webiny/app-aco": ["../app-aco/src"], "@webiny/app-admin/*": ["../app-admin/src/*"], "@webiny/app-admin": ["../app-admin/src"], "@webiny/app-admin-rmwc/*": ["../app-admin-rmwc/src/*"], diff --git a/packages/app-utils/src/fta/Domain/Repositories/Loading/LoadingRepositoryFactory.ts b/packages/app-utils/src/fta/Domain/Repositories/Loading/LoadingRepositoryFactory.ts index e761c3768ae..de631f0bb14 100644 --- a/packages/app-utils/src/fta/Domain/Repositories/Loading/LoadingRepositoryFactory.ts +++ b/packages/app-utils/src/fta/Domain/Repositories/Loading/LoadingRepositoryFactory.ts @@ -3,8 +3,8 @@ import { LoadingRepository } from "./LoadingRepository"; export class LoadingRepositoryFactory { private cache: Map = new Map(); - getRepository() { - const cacheKey = this.getCacheKey(); + getRepository(namespace?: string) { + const cacheKey = this.getCacheKey(namespace); if (!this.cache.has(cacheKey)) { this.cache.set(cacheKey, new LoadingRepository()); @@ -13,8 +13,8 @@ export class LoadingRepositoryFactory { return this.cache.get(cacheKey) as LoadingRepository; } - private getCacheKey() { - return Date.now().toString(); + private getCacheKey(namespace?: string) { + return namespace ?? Date.now().toString(); } } diff --git a/yarn.lock b/yarn.lock index c9184c714e3..6c769ccfd69 100644 --- a/yarn.lock +++ b/yarn.lock @@ -13721,6 +13721,7 @@ __metadata: "@webiny/app-admin": "npm:0.0.0" "@webiny/app-headless-cms-common": "npm:0.0.0" "@webiny/app-security": "npm:0.0.0" + "@webiny/app-utils": "npm:0.0.0" "@webiny/app-wcp": "npm:0.0.0" "@webiny/cli": "npm:0.0.0" "@webiny/form": "npm:0.0.0" @@ -14714,7 +14715,6 @@ __metadata: "@emotion/babel-plugin": "npm:^11.11.0" "@emotion/react": "npm:11.10.8" "@webiny/app": "npm:0.0.0" - "@webiny/app-aco": "npm:0.0.0" "@webiny/app-admin": "npm:0.0.0" "@webiny/app-admin-rmwc": "npm:0.0.0" "@webiny/app-apw": "npm:0.0.0"