/** * BetterDiscord Utils Module * Copyright (c) 2015-present Jiiks/JsSucks - https://github.com/Jiiks / https://github.com/JsSucks * All rights reserved. * https://betterdiscord.net * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ const path = require('path'), fs = require('fs'), _ = require('lodash'); import { Vendor } from 'modules'; export class Utils { static overload(fn, cb) { const orig = fn; return function (...args) { orig(...args); cb(...args); } } static async tryParseJson(jsonString) { try { return JSON.parse(jsonString); } catch (err) { throw ({ 'message': 'Failed to parse json', err }); } } static toCamelCase(o) { const camelCased = {}; _.forEach(o, (value, key) => { if (_.isPlainObject(value) || _.isArray(value)) { value = this.toCamelCase(value); } camelCased[_.camelCase(key)] = value; }); return camelCased; } } export class FileUtils { static async fileExists(path) { return new Promise((resolve, reject) => { fs.stat(path, (err, stats) => { if (err) return reject({ 'message': `No such file or directory: ${err.path}`, err }); if (!stats.isFile()) return reject({ 'message': `Not a file: ${path}`, stats }); resolve(); }); }); } static async directoryExists(path) { return new Promise((resolve, reject) => { fs.stat(path, (err, stats) => { if (err) return reject({ 'message': `Directory does not exist: ${path}`, err }); if (!stats.isDirectory()) return reject({ 'message': `Not a directory: ${path}`, stats }); resolve(); }); }); } static async createDirectory(path) { return new Promise((resolve, reject) => { fs.mkdir(path, err => { if (err) { if (err.code === 'EEXIST') return resolve(); else return reject(err); } resolve(); }); }); } static async ensureDirectory(path) { try { await this.directoryExists(path); return true; } catch (err) { try { await this.createDirectory(path); return true; } catch (err) { throw err; } } } static async readFile(path) { try { await this.fileExists(path); } catch (err) { throw (err); } return new Promise((resolve, reject) => { fs.readFile(path, 'utf-8', (err, data) => { if (err) reject({ 'message': `Could not read file: ${path}`, err }); resolve(data); }); }); } static async writeFile(path, data) { return new Promise((resolve, reject) => { fs.writeFile(path, data, err => { if (err) return reject(err); resolve(); }); }); } static async readJsonFromFile(path) { let readFile; try { readFile = await this.readFile(path); } catch (err) { throw (err); } try { const parsed = await Utils.tryParseJson(readFile); return parsed; } catch (err) { throw (Object.assign(err, { path })); } } static async writeJsonToFile(path, json) { return this.writeFile(path, JSON.stringify(json)); } static async listDirectory(path) { try { await this.directoryExists(path); return new Promise((resolve, reject) => { fs.readdir(path, (err, files) => { if (err) return reject(err); resolve(files); }); }); } catch (err) { throw err; } } static async readDir(path) { return this.listDirectory(path); } } const logs = []; export class ClientLogger { static err(module, message) { this.log(module, message, 'err'); } static warn(module, message) { this.log(module, message, 'warn'); } static info(module, message) { this.log(module, message, 'info'); } static dbg(module, message) { this.log(module, message, 'dbg'); } static log(module, message, level = 'log') { message = message.message || message; if (typeof message === 'object') { //TODO object handler for logs console.log(message); return; } level = this.parseLevel(level); console[level]('[%cBetter%cDiscord:%s] %s', 'color: #3E82E5', '', `${module}${level === 'debug' ? '|DBG' : ''}`, message); logs.push(`${level.toUpperCase()} : [${Vendor.moment().format('DD/MM/YY hh:mm:ss')}|${module}] ${message}`); window.bdlogs = logs; } static logError(err) { if (!err.module && !err.message) { console.log(err); return; } this.err(err.module, err.message); } static get levels() { return { 'log': 'log', 'warn': 'warn', 'err': 'error', 'error': 'error', 'debug': 'debug', 'dbg': 'debug', 'info': 'info' }; } static parseLevel(level) { return this.levels.hasOwnProperty(level) ? this.levels[level] : 'log'; } }