224 lines
5.9 KiB
JavaScript
224 lines
5.9 KiB
JavaScript
/**
|
|
* 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';
|
|
}
|
|
}
|