/** * SPDX-FileCopyrightText: 2023 Nextcloud GmbH and Nextcloud contributors * SPDX-License-Identifier: AGPL-3.0-or-later */ import type { FilesStore, RootsStore, RootOptions, Service, FilesState, FileSource } from '../types' import type { FileStat, ResponseDataDetailed } from 'webdav' import type { Folder, Node } from '@nextcloud/files' import { davGetDefaultPropfind, davResultToNode, davRootPath } from '@nextcloud/files' import { defineStore } from 'pinia' import { subscribe } from '@nextcloud/event-bus' import logger from '../logger' import Vue from 'vue' import { client } from '../services/WebdavClient.ts' const fetchNode = async (node: Node): Promise => { const propfindPayload = davGetDefaultPropfind() const result = await client.stat(`${davRootPath}${node.path}`, { details: true, data: propfindPayload, }) as ResponseDataDetailed return davResultToNode(result.data) } export const useFilesStore = function(...args) { const store = defineStore('files', { state: (): FilesState => ({ files: {} as FilesStore, roots: {} as RootsStore, }), getters: { /** * Get a file or folder by its source * @param state */ getNode: (state) => (source: FileSource): Node|undefined => state.files[source], /** * Get a list of files or folders by their IDs * Note: does not return undefined values * @param state */ getNodes: (state) => (sources: FileSource[]): Node[] => sources .map(source => state.files[source]) .filter(Boolean), /** * Get files or folders by their file ID * Multiple nodes can have the same file ID but different sources * (e.g. in a shared context) * @param state */ getNodesById: (state) => (fileId: number): Node[] => Object.values(state.files).filter(node => node.fileid === fileId), /** * Get the root folder of a service * @param state */ getRoot: (state) => (service: Service): Folder|undefined => state.roots[service], }, actions: { updateNodes(nodes: Node[]) { // Update the store all at once const files = nodes.reduce((acc, node) => { if (!node.fileid) { logger.error('Trying to update/set a node without fileid', { node }) return acc } acc[node.source] = node return acc }, {} as FilesStore) Vue.set(this, 'files', { ...this.files, ...files }) }, deleteNodes(nodes: Node[]) { nodes.forEach(node => { if (node.source) { Vue.delete(this.files, node.source) } }) }, setRoot({ service, root }: RootOptions) { Vue.set(this.roots, service, root) }, onDeletedNode(node: Node) { this.deleteNodes([node]) }, onCreatedNode(node: Node) { this.updateNodes([node]) }, async onUpdatedNode(node: Node) { if (!node.fileid) { logger.error('Trying to update/set a node without fileid', { node }) return } // If we have multiple nodes with the same file ID, we need to update all of them const nodes = this.getNodesById(node.fileid) if (nodes.length > 1) { await Promise.all(nodes.map(fetchNode)).then(this.updateNodes) logger.debug(nodes.length + ' nodes updated in store', { fileid: node.fileid }) return } // If we have only one node with the file ID, we can update it directly if (node.source === nodes[0].source) { this.updateNodes([node]) return } // Otherwise, it means we receive an event for a node that is not in the store fetchNode(node).then(n => this.updateNodes([n])) }, }, }) const fileStore = store(...args) // Make sure we only register the listeners once if (!fileStore._initialized) { subscribe('files:node:created', fileStore.onCreatedNode) subscribe('files:node:deleted', fileStore.onDeletedNode) subscribe('files:node:updated', fileStore.onUpdatedNode) fileStore._initialized = true } return fileStore }