UNPKG

@jsprismarine/prismarine

Version:

Dedicated Minecraft Bedrock Edition server written in TypeScript

179 lines (172 loc) • 20.6 kB
'use strict'; Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); const errors = require('@jsprismarine/errors'); const confbox = require('confbox'); const fs = require('node:fs'); const path = require('node:path'); const _interopDefault = e => e && e.__esModule ? e : { default: e }; const fs__default = /*#__PURE__*/_interopDefault(fs); const path__default = /*#__PURE__*/_interopDefault(path); const _ = { get: (obj, path2, defaultValue = void 0) => { const travel = (regexp) => String.prototype.split.call(path2, regexp).filter(Boolean).reduce((res, key) => res !== null && res !== void 0 ? res[key] : res, obj); const result = travel(/[,[\]]+?/) || travel(/[,[\].]+?/); return result === void 0 || result === obj ? defaultValue : result; }, set: (obj, path2, value) => { if (Object(obj) !== obj) return obj; if (!Array.isArray(path2)) path2 = path2.toString().match(/[^.[\]]+/g) || []; path2.slice(0, -1).reduce( (a, c, i) => Object(a[c]) === a[c] ? a[c] : a[c] = Math.abs(path2[i + 1]) >> 0 === Number(path2[i + 1]) ? [] : {}, obj )[path2.at(-1)] = value; return obj; }, has: (obj, key) => { const keyParts = key.split("."); return Boolean( obj && (keyParts.length > 1 ? _.has(obj[key.split(".")[0]], keyParts.slice(1).join(".")) : Object.hasOwnProperty.call(obj, key)) ); }, del: (obj, path2) => { const pathArray = Array.isArray(path2) ? path2 : path2.match(/([^[.\]])+/g); pathArray.reduce((acc, key, i) => { if (i === pathArray.length - 1) delete acc[key]; return acc[key]; }, obj); return true; } }; const TypeDefaults = { json: "{}", yaml: " " }; class ConfigBuilder { type; path; /** * Config constructor. */ constructor(filePath) { const pathSplitted = path__default.default.parse(filePath); this.type = pathSplitted.ext.slice(1); if (!Object.keys(TypeDefaults).some((i) => i.toLowerCase() === this.type.toLowerCase())) { throw new Error(`Unsupported config type. (Supported types: ${Object.keys(TypeDefaults).join(", ")})`); } if (!fs__default.default.existsSync(pathSplitted.dir)) fs__default.default.mkdirSync(pathSplitted.dir, { recursive: true }); if (!fs__default.default.existsSync(filePath)) fs__default.default.writeFileSync(filePath, TypeDefaults[this.type], "utf8"); this.path = filePath; } /** * Get path to the config file on the filesystem. * * @returns {string} the path to the config file */ getPath() { return this.path; } /** * Get the config format (eg. type). * * @returns {'yaml' | 'json'} either `yaml` or `json` */ getType() { return this.type; } /** * @private */ setFileData(data = {}) { if (!data) throw new errors.ConfigInvalidDataError(); fs__default.default.writeFileSync(this.path, this.stringify(data), "utf8"); } /** * @private */ stringify(data = {}) { switch (this.type) { case "json": return JSON.stringify(data, null, 4); case "yaml": return confbox.stringifyYAML(data, { indent: 4 }); default: throw new Error(`Unknown config type ${this.type}!`); } } getFileData() { const raw = fs__default.default.readFileSync(this.path, "utf8"); switch (this.type) { case "json": try { return confbox.parseJSON5(raw); } catch { } break; case "yaml": try { return confbox.parseYAML(raw); } catch { } break; default: throw new Error(`Unknown config type: ${this.type}!`); } return {}; } /** * Get a config value from a key. * * @param {string} key - the config key * @param {T} defaults - */ get(key, defaults) { const data = this.getFileData(); let result = _.get(data, key); if (typeof result === "undefined" && typeof defaults !== "undefined") { const newData = _.set(data, key, defaults); this.setFileData(newData); result = defaults; } return result; } /** * Sets a key - value pair in config. * * @returns true if the value was set successfully */ set(key, value) { const data = this.getFileData(); const newData = _.set(data, key, value); try { this.setFileData(newData); return true; } catch { return false; } } /** * Check if config value exists. * * @param key - the config key * @returns true if the config contains that key */ has(key) { const data = this.getFileData(); const result = _.has(data, key); return result; } /** * Delete a config value. * * @param key - the config key * @returns true if the deletion was successful */ del(key) { const data = this.getFileData(); const isSuccessful = _.del(data, key); this.setFileData(data); return isSuccessful; } } exports.ConfigBuilder = ConfigBuilder; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ConfigBuilder.cjs.cjs","sources":["../../src/config/ConfigBuilder.ts"],"sourcesContent":["import { ConfigInvalidDataError } from '@jsprismarine/errors';\nimport { parseJSON5, parseYAML, stringifyYAML } from 'confbox';\nimport fs from 'node:fs';\nimport path from 'node:path';\n\nconst _ = {\n    get: (obj: any, path: string, defaultValue = undefined) => {\n        const travel = (regexp: any) =>\n            String.prototype.split\n                .call(path, regexp)\n                .filter(Boolean)\n                .reduce((res, key) => (res !== null && res !== undefined ? res[key] : res), obj);\n        const result = travel(/[,[\\]]+?/) || travel(/[,[\\].]+?/);\n        return result === undefined || result === obj ? defaultValue : result;\n    },\n    set: (obj: any, path: any, value: any) => {\n        if (Object(obj) !== obj) return obj;\n        if (!Array.isArray(path)) path = path.toString().match(/[^.[\\]]+/g) || [];\n        path\n            .slice(0, -1)\n            .reduce(\n                (a: any, c: any, i: any) =>\n                    Object(a[c]) === a[c]\n                        ? a[c]\n                        : (a[c] = Math.abs(path[i + 1]) >> 0 === Number(path[i + 1]) ? [] : {}),\n                obj\n            )[path.at(-1)] = value;\n        return obj;\n    },\n    has: (obj: any, key: string): boolean => {\n        const keyParts = key.split('.');\n\n        return Boolean(\n            obj &&\n                (keyParts.length > 1\n                    ? _.has(obj[key.split('.')[0]!], keyParts.slice(1).join('.'))\n                    : Object.hasOwnProperty.call(obj, key))\n        );\n    },\n    del: (obj: any, path: any) => {\n        const pathArray = Array.isArray(path) ? path : path.match(/([^[.\\]])+/g);\n\n        pathArray.reduce((acc: any, key: any, i: number) => {\n            // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n            if (i === pathArray.length - 1) delete acc[key];\n            return acc[key];\n        }, obj);\n        return true;\n    }\n};\n\nconst TypeDefaults = {\n    json: '{}',\n    yaml: ' '\n};\n\nexport type ConfigData = {\n    [key: string]: any;\n};\n\n/**\n * General config-file handler.\n */\nexport class ConfigBuilder {\n    private type: 'yaml' | 'json';\n    private path: string;\n\n    /**\n     * Config constructor.\n     */\n    public constructor(filePath: string) {\n        const pathSplitted = path.parse(filePath);\n\n        this.type = pathSplitted.ext.slice(1) as 'yaml' | 'json';\n\n        if (!Object.keys(TypeDefaults).some((i) => i.toLowerCase() === this.type.toLowerCase())) {\n            throw new Error(`Unsupported config type. (Supported types: ${Object.keys(TypeDefaults).join(', ')})`);\n        }\n\n        // Create the config directory if it doesn't exist.\n        if (!fs.existsSync(pathSplitted.dir)) fs.mkdirSync(pathSplitted.dir, { recursive: true });\n        if (!fs.existsSync(filePath)) fs.writeFileSync(filePath, (TypeDefaults as any)[this.type], 'utf8');\n\n        this.path = filePath;\n    }\n\n    /**\n     * Get path to the config file on the filesystem.\n     *\n     * @returns {string} the path to the config file\n     */\n    public getPath(): string {\n        return this.path;\n    }\n\n    /**\n     * Get the config format (eg. type).\n     *\n     * @returns {'yaml' | 'json'} either `yaml` or `json`\n     */\n    public getType(): 'yaml' | 'json' {\n        return this.type;\n    }\n\n    /**\n     * @private\n     */\n    private setFileData(data: ConfigData = {}) {\n        if (!(data as any)) throw new ConfigInvalidDataError();\n        fs.writeFileSync(this.path, this.stringify(data), 'utf8');\n    }\n\n    /**\n     * @private\n     */\n    private stringify(data: ConfigData = {}) {\n        // FIXME: This overwrites comments in the file.\n        switch (this.type) {\n            case 'json':\n                return JSON.stringify(data, null, 4);\n            case 'yaml':\n                return stringifyYAML(data, { indent: 4 });\n            default:\n                throw new Error(`Unknown config type ${this.type}!`);\n        }\n    }\n\n    private getFileData(): ConfigData {\n        const raw = fs.readFileSync(this.path, 'utf8');\n\n        switch (this.type) {\n            case 'json':\n                try {\n                    return parseJSON5(raw);\n                } catch {}\n                break;\n            case 'yaml':\n                try {\n                    return parseYAML(raw);\n                } catch {}\n                break;\n            default:\n                throw new Error(`Unknown config type: ${this.type}!`);\n        }\n\n        return {};\n    }\n\n    /**\n     * Get a config value from a key.\n     *\n     * @param {string} key - the config key\n     * @param {T} defaults -\n     */\n    public get<T = any>(key: string, defaults?: T): T {\n        const data = this.getFileData();\n        let result = _.get(data, key);\n\n        if (typeof result === 'undefined' && typeof defaults !== 'undefined') {\n            const newData = _.set(data, key, defaults);\n            this.setFileData(newData);\n            result = defaults;\n        }\n\n        return result;\n    }\n\n    /**\n     * Sets a key - value pair in config.\n     *\n     * @returns true if the value was set successfully\n     */\n    public set(key: string, value: any) {\n        const data = this.getFileData();\n        const newData = _.set(data, key, value);\n\n        try {\n            this.setFileData(newData);\n            return true;\n        } catch {\n            return false;\n        }\n    }\n\n    /**\n     * Check if config value exists.\n     *\n     * @param key - the config key\n     * @returns true if the config contains that key\n     */\n    public has(key: string): boolean {\n        const data = this.getFileData();\n        const result = _.has(data, key);\n        return result;\n    }\n\n    /**\n     * Delete a config value.\n     *\n     * @param key - the config key\n     * @returns true if the deletion was successful\n     */\n    public del(key: string): boolean {\n        const data = this.getFileData();\n\n        // It mutates the object, we\n        // don't need to define a new\n        // variable.\n        const isSuccessful = _.del(data, key);\n\n        this.setFileData(data);\n        return isSuccessful;\n    }\n}\n"],"names":["path","fs","ConfigInvalidDataError","stringifyYAML","parseJSON5","parseYAML"],"mappings":"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"}