UNPKG

@nephele/adapter-file-system

Version:

File system adapter for the Nephele WebDAV server.

265 lines 8.9 kB
import { platform } from 'node:os'; import { ForbiddenError, PropertyIsProtectedError, PropertyNotFoundError, } from 'nephele'; export default class Properties { constructor({ resource }) { this.resource = resource; } async get(name) { switch (name) { case 'creationdate': { const stats = await this.resource.getStats(); return stats.birthtime.toISOString(); } case 'getcontentlength': return `${await this.resource.getLength()}`; case 'getcontenttype': const mediaType = await this.resource.getMediaType(); if (mediaType == null) { throw new PropertyNotFoundError(`${name} property doesn't exist on resource.`); } return mediaType; case 'getetag': return await this.resource.getEtag(); case 'getlastmodified': { const stats = await this.resource.getStats(); return stats.mtime.toUTCString(); } case 'resourcetype': if (await this.resource.isCollection()) { return { collection: {} }; } else { return {}; } case 'supportedlock': return { lockentry: [ { lockscope: { exclusive: {} }, locktype: { write: {} }, }, { lockscope: { shared: {} }, locktype: { write: {} }, }, ], }; case 'quota-available-bytes': return `${await this.resource.getFreeSpace()}`; case 'quota-used-bytes': return `${await this.resource.getTotalSpace()}`; case 'LCGDM:%%mode': if (platform() === 'win32') { throw new PropertyNotFoundError(`${name} property doesn't exist on resource.`); } try { const stats = await this.resource.getStats(); return `${stats.mode.toString(8)}`; } catch (e) { if (e.code === 'ENOENT') { throw new PropertyNotFoundError(`${name} property doesn't exist on resource.`); } else { throw e; } } } const meta = await this.resource.readMetadataFile(); if (meta.props == null || !(name in meta.props)) { throw new PropertyNotFoundError(`${name} property doesn't exist on resource.`); } return meta.props[name]; } async getByUser(name, _user) { return await this.get(name); } async set(name, value) { const errors = await this.runInstructions([['set', name, value]]); if (errors != null && errors.length) { throw errors[0][1]; } } async setByUser(name, value, user) { const errors = await this.runInstructionsByUser([['set', name, value]], user); if (errors != null && errors.length) { throw errors[0][1]; } } async remove(name) { const errors = await this.runInstructions([['remove', name, undefined]]); if (errors != null && errors.length) { throw errors[0][1]; } } async removeByUser(name, user) { const errors = await this.runInstructionsByUser([['remove', name, undefined]], user); if (errors != null && errors.length) { throw errors[0][1]; } } async runInstructions(instructions) { let meta = {}; let changed = false; let errors = []; let setMode = undefined; const errorEverything = (e) => { const errProps = {}; for (let instruction of instructions) { errProps[instruction[1]] = e; } return Object.entries(errProps); }; try { meta = await this.resource.readMetadataFile(); } catch (e) { return errorEverything(e); } for (let instruction of instructions) { const [action, name, value] = instruction; if ([ 'creationdate', 'getcontentlength', 'getcontenttype', 'getetag', 'getlastmodified', 'resourcetype', 'supportedlock', 'quota-available-bytes', 'quota-used-bytes', ...(platform() === 'win32' ? ['LCGDM:%%mode'] : []), ].includes(name)) { errors.push([ name, new PropertyIsProtectedError(`${name} is a protected property.`), ]); continue; } if (action === 'set') { if (name === 'LCGDM:%%mode') { setMode = parseInt(value, 8); continue; } if (meta.props == null) { meta.props = {}; } meta.props[name] = value; changed = true; } else { if (name === 'LCGDM:%%mode') { errors.push([ name, new ForbiddenError("This property can't be deleted."), ]); continue; } if (meta.props != null && name in meta.props) { delete meta.props[name]; changed = true; } } } if (errors.length) { return errors; } let unsetMode = async () => { }; if (setMode != null) { try { const stats = await this.resource.getStats(); const safeMode = setMode % 0o1000; const mode = stats.mode & 0o777000 & safeMode; unsetMode = async () => { try { await this.resource.setMode(stats.mode); } catch (e) { } }; await this.resource.setMode(mode); } catch (e) { return [...errors, ['LCGDM:%%mode', e]]; } } if (changed) { try { await this.resource.saveMetadataFile(meta); } catch (e) { await unsetMode(); return errorEverything(e); } } if (errors.length) { return errors; } } async runInstructionsByUser(instructions, _user) { return await this.runInstructions(instructions); } async getAll() { const meta = await this.resource.readMetadataFile(); const props = { ...meta.props }; for (let name of [ 'creationdate', 'getcontentlength', 'getcontenttype', 'getetag', 'getlastmodified', 'resourcetype', 'supportedlock', ...(platform() === 'win32' ? [] : ['LCGDM:%%mode']), ]) { try { props[name] = await this.get(name); } catch (e) { if (!(e instanceof PropertyNotFoundError)) { props[name] = e; } } } return props; } async getAllByUser(_user) { return await this.getAll(); } async list() { return [...(await this.listLive()), ...(await this.listDead())]; } async listByUser(user) { return [ ...(await this.listLiveByUser(user)), ...(await this.listDeadByUser(user)), ]; } async listLive() { return [ 'creationdate', 'getcontentlength', 'getcontenttype', 'getetag', 'getlastmodified', 'resourcetype', 'supportedlock', 'quota-available-bytes', 'quota-used-bytes', ...(platform() === 'win32' ? [] : ['LCGDM:%%mode']), ]; } async listLiveByUser(_user) { return await this.listLive(); } async listDead() { let meta = await this.resource.readMetadataFile(); return [ ...Object.keys(meta.props || {}), ]; } async listDeadByUser(_user) { return await this.listDead(); } } //# sourceMappingURL=Properties.js.map