WIP: Fix builds

This commit is contained in:
Aaron Dewes 2021-08-06 16:40:08 +01:00
parent 18586644e1
commit 5f8adf787d
4 changed files with 721 additions and 548 deletions

View File

@ -23,14 +23,11 @@ jobs:
yarn
yarn devInstall
yarn compile
yarn build
- name: Build/release Electron app
uses: samuelmeuli/action-electron-builder@v1
- name: Release
uses: softprops/action-gh-release@v1
if: startsWith(github.ref, 'refs/tags/')
with:
# GitHub token, automatically provided to the action
# (No need to define this secret in the repo settings)
github_token: ${{ secrets.github_token }}
# If the commit is tagged with a version (e.g. "v1.0.0"),
# release the app after building
release: ${{ startsWith(github.ref, 'refs/tags/v') }}
files:
- 'builds/*'

View File

@ -1,42 +0,0 @@
language: node_js
node_js: "12"
before_install:
- npm i
jobs:
include:
- stage: Linux & Mac Build
os: osx
osx_image: xcode10.2
env:
- ELECTRON_CACHE=$HOME/.cache/electron
- ELECTRON_BUILDER_CACHE=$HOME/.cache/electron-builder
before_cache:
- rm -rf $ELECTRON_BUILDER_CACHE/wine
script:
- npm run devInstall
- npm run build
# - stage: Windows Build
# os: windows
# script:
# - export NPM_CONFIG_PREFIX=C:\\npm_prefix
# - export PATH="/c/npm_prefix:$PATH"
# - npm i -g npm@latest
# - npm run devInstall
# - npm run build
# - stage: GitHub Release
# script:
# - export TRAVIS_TAG=${TRAVIS_TAG:-$(date +'%Y%m%d%H%M%S')-$(git log --format=%h -1)}
# - git tag $TRAVIS_TAG
# deploy:
# provider: releases
# prerelease: true
# api_key: "$GH_TOKEN"
# cleanup: false
# file:
# - builds/lightcord-win32-ia32.zip
# - builds/lightcord-win32.exe
# - builds/lightcord-linux-x64.zip
# - builds/lightcord-darwin.zip
# on:
# tags: true

View File

@ -80,24 +80,22 @@ if (process.arch === 'arm64') {
module.exports = {
consoleLog = (...args) => {},
}
return;
}
} else {
features.declareSupported('voice_panning');
features.declareSupported('voice_multiple_connections');
features.declareSupported('media_devices');
features.declareSupported('media_video');
features.declareSupported('debug_logging');
features.declareSupported('set_audio_device_by_id');
features.declareSupported('set_video_device_by_id');
features.declareSupported('loopback');
features.declareSupported('experiment_config');
features.declareSupported('remote_locus_network_control');
features.declareSupported('connection_replay');
features.declareSupported('simulcast');
features.declareSupported('direct_video');
features.declareSupported('voice_panning');
features.declareSupported('voice_multiple_connections');
features.declareSupported('media_devices');
features.declareSupported('media_video');
features.declareSupported('debug_logging');
features.declareSupported('set_audio_device_by_id');
features.declareSupported('set_video_device_by_id');
features.declareSupported('loopback');
features.declareSupported('experiment_config');
features.declareSupported('remote_locus_network_control');
features.declareSupported('connection_replay');
features.declareSupported('simulcast');
features.declareSupported('direct_video');
if (process.platform === 'win32') {
if (process.platform === 'win32') {
features.declareSupported('voice_legacy_subsystem');
features.declareSupported('soundshare');
features.declareSupported('wumpus_video');
@ -111,9 +109,9 @@ if (process.platform === 'win32') {
// NOTE(jvass): currently there's no experimental encoders! Add this back if you
// add one and want to re-enable the UI for them.
// features.declareSupported('experimental_encoders');
}
}
function bindConnectionInstance(instance) {
function bindConnectionInstance(instance) {
return {
destroy: () => instance.destroy(),
@ -155,15 +153,15 @@ function bindConnectionInstance(instance) {
getFilteredStats: (filter, callback) => instance.getFilteredStats(filter, callback),
startReplay: () => instance.startReplay(),
};
}
}
VoiceEngine.createTransport = VoiceEngine._createTransport;
VoiceEngine.createTransport = VoiceEngine._createTransport;
if (isElectronRenderer) {
if (isElectronRenderer) {
VoiceEngine.setImageDataAllocator((width, height) => new window.ImageData(width, height));
}
}
VoiceEngine.createVoiceConnection = function (audioSSRC, userId, address, port, onConnectCallback, experiments, rids) {
VoiceEngine.createVoiceConnection = function (audioSSRC, userId, address, port, onConnectCallback, experiments, rids) {
let instance = null;
if (rids != null) {
instance = new VoiceEngine.VoiceConnection(audioSSRC, userId, address, port, onConnectCallback, experiments, rids);
@ -173,18 +171,18 @@ VoiceEngine.createVoiceConnection = function (audioSSRC, userId, address, port,
instance = new VoiceEngine.VoiceConnection(audioSSRC, userId, address, port, onConnectCallback);
}
return bindConnectionInstance(instance);
};
VoiceEngine.createOwnStreamConnection = VoiceEngine.createVoiceConnection;
};
VoiceEngine.createOwnStreamConnection = VoiceEngine.createVoiceConnection;
VoiceEngine.createReplayConnection = function (audioEngineId, callback, replayLog) {
VoiceEngine.createReplayConnection = function (audioEngineId, callback, replayLog) {
if (replayLog == null) {
return null;
}
return bindConnectionInstance(new VoiceEngine.VoiceReplayConnection(replayLog, audioEngineId, callback));
};
};
VoiceEngine.setAudioSubsystem = function (subsystem) {
VoiceEngine.setAudioSubsystem = function (subsystem) {
if (appSettings == null) {
console.warn('Unable to access app settings.');
return;
@ -203,9 +201,9 @@ VoiceEngine.setAudioSubsystem = function (subsystem) {
if (isElectronRenderer) {
window.DiscordNative.app.relaunch();
}
};
};
VoiceEngine.setDebugLogging = function (enable) {
VoiceEngine.setDebugLogging = function (enable) {
if (appSettings == null) {
console.warn('Unable to access app settings.');
return;
@ -220,15 +218,15 @@ VoiceEngine.setDebugLogging = function (enable) {
if (isElectronRenderer) {
window.DiscordNative.app.relaunch();
}
};
};
VoiceEngine.getDebugLogging = function () {
VoiceEngine.getDebugLogging = function () {
return debugLogging;
};
};
const videoStreams = {};
const videoStreams = {};
const ensureCanvasContext = function (sinkId) {
const ensureCanvasContext = function (sinkId) {
let canvas = document.getElementById(sinkId);
if (canvas == null) {
for (const popout of window.popouts.values()) {
@ -251,21 +249,21 @@ const ensureCanvasContext = function (sinkId) {
}
return context;
};
};
// [adill] NB: with context isolation it has become extremely costly (both memory & performance) to provide the image
// data directly to clients at any reasonably fast interval so we've replaced setVideoOutputSink with a direct canvas
// renderer via addVideoOutputSink
const setVideoOutputSink = VoiceEngine.setVideoOutputSink;
const clearVideoOutputSink = (streamId) => {
// [adill] NB: with context isolation it has become extremely costly (both memory & performance) to provide the image
// data directly to clients at any reasonably fast interval so we've replaced setVideoOutputSink with a direct canvas
// renderer via addVideoOutputSink
const setVideoOutputSink = VoiceEngine.setVideoOutputSink;
const clearVideoOutputSink = (streamId) => {
// [adill] NB: if you don't pass a frame callback setVideoOutputSink clears the sink
setVideoOutputSink(streamId);
};
const signalVideoOutputSinkReady = VoiceEngine.signalVideoOutputSinkReady;
delete VoiceEngine.setVideoOutputSink;
delete VoiceEngine.signalVideoOutputSinkReady;
};
const signalVideoOutputSinkReady = VoiceEngine.signalVideoOutputSinkReady;
delete VoiceEngine.setVideoOutputSink;
delete VoiceEngine.signalVideoOutputSinkReady;
function addVideoOutputSinkInternal(sinkId, streamId, frameCallback) {
function addVideoOutputSinkInternal(sinkId, streamId, frameCallback) {
let sinks = videoStreams[streamId];
if (sinks == null) {
sinks = videoStreams[streamId] = new Map();
@ -288,9 +286,9 @@ function addVideoOutputSinkInternal(sinkId, streamId, frameCallback) {
}
sinks.set(sinkId, frameCallback);
}
}
VoiceEngine.addVideoOutputSink = function (sinkId, streamId, frameCallback) {
VoiceEngine.addVideoOutputSink = function (sinkId, streamId, frameCallback) {
let canvasContext = null;
addVideoOutputSinkInternal(sinkId, streamId, (imageData) => {
if (canvasContext == null) {
@ -309,9 +307,9 @@ VoiceEngine.addVideoOutputSink = function (sinkId, streamId, frameCallback) {
const leak = canvasContext.getImageData(0, 0, 1, 1);
canvasContext.putImageData(imageData, 0, 0);
});
};
};
VoiceEngine.removeVideoOutputSink = function (sinkId, streamId) {
VoiceEngine.removeVideoOutputSink = function (sinkId, streamId) {
const sinks = videoStreams[streamId];
if (sinks != null) {
sinks.delete(sinkId);
@ -321,10 +319,10 @@ VoiceEngine.removeVideoOutputSink = function (sinkId, streamId) {
clearVideoOutputSink(streamId);
}
}
};
};
let sinkId = 0;
VoiceEngine.getNextVideoOutputFrame = function (streamId) {
let sinkId = 0;
VoiceEngine.getNextVideoOutputFrame = function (streamId) {
const nextVideoFrameSinkId = `getNextVideoFrame_${++sinkId}`;
return new Promise((resolve, reject) => {
@ -342,9 +340,10 @@ VoiceEngine.getNextVideoOutputFrame = function (streamId) {
});
});
});
};
};
console.log(`Initializing voice engine with audio subsystem: ${audioSubsystem}`);
VoiceEngine.initialize({audioSubsystem, logLevel, dataDirectory});
console.log(`Initializing voice engine with audio subsystem: ${audioSubsystem}`);
VoiceEngine.initialize({audioSubsystem, logLevel, dataDirectory});
module.exports = VoiceEngine;
module.exports = VoiceEngine;
}

View File

@ -1,286 +1,505 @@
const child_process = require("child_process")
const path = require("path")
const terser = require("terser")
const util = require("util")
const child_process = require("child_process");
const path = require("path");
const terser = require("terser");
const util = require("util");
const production = true
const includeSourcesMaps = true
const production = true;
const includeSourcesMaps = true;
let fs = require("fs")
let fs = require("fs");
exports.default = async function beforeBuild(context){
await main()
return true
}
exports.default = async function beforeBuild(context) {
await main();
return true;
};
const PROJECT_DIR = path.resolve(__dirname, "..");
console.log = (...args) => {
process.stdout.write(Buffer.from(util.formatWithOptions({colors: true}, ...args)+"\n", "binary").toString("utf8"))
}
process.stdout.write(
Buffer.from(
util.formatWithOptions({ colors: true }, ...args) + "\n",
"binary"
).toString("utf8")
);
};
console.info = (...args) => {
console.log(`\x1b[34m[INFO]\x1b[0m`, ...args)
}
let commit = child_process.execSync("git rev-parse HEAD").toString().split("\n")[0].trim()
console.info(`Obtained commit ${commit} for the build`)
console.log(`\x1b[34m[INFO]\x1b[0m`, ...args);
};
let commit = child_process
.execSync("git rev-parse HEAD")
.toString()
.split("\n")[0]
.trim();
console.info(`Obtained commit ${commit} for the build`);
async function processNextDir(folder, folders, predicate, compile, ignoreModules){
if(typeof ignoreModules === "undefined")ignoreModules = false
let files = fs.readdirSync(folder, {withFileTypes: true})
for(let file of files){
if(file.isFile()){
let isMinified = file.name.endsWith(".min.js") || file.name.endsWith(".min.css")
let filepath = path.join(folder, file.name)
let type = file.name.split(".").pop().toLowerCase()
if(type === file.name)type = ""
if([
"ts",
"md",
"gitignore",
"map"
].includes(type)){
console.warn(`\x1b[33mIgnored file ${path.relative(folders.startDir, filepath)} because of type ${type}\x1b[0m`)
continue
async function processNextDir(
folder,
folders,
predicate,
compile,
ignoreModules
) {
if (typeof ignoreModules === "undefined") ignoreModules = false;
let files = fs.readdirSync(folder, { withFileTypes: true });
for (let file of files) {
if (file.isFile()) {
let isMinified =
file.name.endsWith(".min.js") || file.name.endsWith(".min.css");
let filepath = path.join(folder, file.name);
let type = file.name.split(".").pop().toLowerCase();
if (type === file.name) type = "";
if (["ts", "md", "gitignore", "map"].includes(type)) {
console.warn(
`\x1b[33mIgnored file ${path.relative(
folders.startDir,
filepath
)} because of type ${type}\x1b[0m`
);
continue;
}
if([
"tsconfig.json",
"webpack.config.js"
].includes(file.name)){
console.warn(`\x1b[33mIgnored file ${path.relative(folders.startDir, filepath)} because of name ${file.name}\x1b[0m`)
continue
if (["tsconfig.json", "webpack.config.js"].includes(file.name)) {
console.warn(
`\x1b[33mIgnored file ${path.relative(
folders.startDir,
filepath
)} because of name ${file.name}\x1b[0m`
);
continue;
}
if(folders.exclude && folders.exclude.test(filepath)){
console.warn(`\x1b[33mIgnored file ${path.relative(folders.startDir, filepath)} because regex\x1b[0m`)
continue
if (folders.exclude && folders.exclude.test(filepath)) {
console.warn(
`\x1b[33mIgnored file ${path.relative(
folders.startDir,
filepath
)} because regex\x1b[0m`
);
continue;
}
let hasMinifiedVersion = (type === "js" || type === "css") && !isMinified && files.find(f => {
return f.name === file.name.split(".").slice(0, -1).join(".")+".min."+type
})
if(hasMinifiedVersion){
console.warn(`\x1b[33mIgnored file ${path.relative(folders.startDir, filepath)} because it has a minified version.\x1b[0m`)
continue
let hasMinifiedVersion =
(type === "js" || type === "css") &&
!isMinified &&
files.find((f) => {
return (
f.name ===
file.name.split(".").slice(0, -1).join(".") + ".min." + type
);
});
if (hasMinifiedVersion) {
console.warn(
`\x1b[33mIgnored file ${path.relative(
folders.startDir,
filepath
)} because it has a minified version.\x1b[0m`
);
continue;
}
if(!isMinified && predicate(filepath) && filepath.split(/[\\/]+/).reverse()[1] !== "js"){
await compile(filepath, path.join(filepath.replace(folders.startDir, folders.newDir)), "..")
}else{
if(["js", "css"].includes(type)){
if(!includeSourcesMaps){
console.log(`We don't include sourcemap for this build. Skipping ${file.name}.`)
return await fs.promises.copyFile(filepath, filepath.replace(folders.startDir, folders.newDir))
if (
!isMinified &&
predicate(filepath) &&
filepath.split(/[\\/]+/).reverse()[1] !== "js"
) {
await compile(
filepath,
path.join(filepath.replace(folders.startDir, folders.newDir)),
".."
);
} else {
if (["js", "css"].includes(type)) {
if (!includeSourcesMaps) {
console.log(
`We don't include sourcemap for this build. Skipping ${file.name}.`
);
return await fs.promises.copyFile(
filepath,
filepath.replace(folders.startDir, folders.newDir)
);
}
let fileContent = (await fs.promises.readFile(filepath, "utf8"))
let sourceMap = fileContent.split(/[\n\r]+/g).pop()
if(!sourceMap || !sourceMap.startsWith("//# sourceMappingURL=")){
console.log(`This file doesn't have sourcemap. ${file.name}.`)
await fs.promises.copyFile(filepath, filepath.replace(folders.startDir, folders.newDir))
continue
let fileContent = await fs.promises.readFile(filepath, "utf8");
let sourceMap = fileContent.split(/[\n\r]+/g).pop();
if (!sourceMap || !sourceMap.startsWith("//# sourceMappingURL=")) {
console.log(`This file doesn't have sourcemap. ${file.name}.`);
await fs.promises.copyFile(
filepath,
filepath.replace(folders.startDir, folders.newDir)
);
continue;
}
let sourceMapContent
if(sourceMap.slice(21).startsWith("data:")){
console.log(`Extracting sourcemap from data uri. From file ${file.name}.`)
sourceMapContent = Buffer.from(sourceMap.split("=").slice(1).join("="), "base64").toString("utf-8")
}else{
console.log(`Extracting sourcemap from file ${file.name}.map.`)
await fs.promises.copyFile(filepath, filepath.replace(folders.startDir, folders.newDir))
sourceMapContent = await fs.promises.readFile(path.join(folder, sourceMap.slice(21)), "utf8")
let sourceMapContent;
if (sourceMap.slice(21).startsWith("data:")) {
console.log(
`Extracting sourcemap from data uri. From file ${file.name}.`
);
sourceMapContent = Buffer.from(
sourceMap.split("=").slice(1).join("="),
"base64"
).toString("utf-8");
} else {
console.log(`Extracting sourcemap from file ${file.name}.map.`);
await fs.promises.copyFile(
filepath,
filepath.replace(folders.startDir, folders.newDir)
);
sourceMapContent = await fs.promises.readFile(
path.join(folder, sourceMap.slice(21)),
"utf8"
);
}
sourceMapContent = JSON.parse(sourceMapContent)
sourceMapContent.sourcesContent = []
let sourceMapPath = filepath + ".map"
sourceMapContent = JSON.parse(sourceMapContent);
sourceMapContent.sourcesContent = [];
let sourceMapPath = filepath + ".map";
fileContent = fileContent
// source map
.replace(sourceMap, "//# sourceMappingURL="+filepath.split(/[\\\/]+/g).pop()+".map")
await fs.promises.writeFile(filepath.replace(folders.startDir, folders.newDir), fileContent)
await fs.promises.writeFile(filepath.replace(folders.startDir, folders.newDir)+".map", JSON.stringify(sourceMapContent))
}else{
await fs.promises.copyFile(filepath, filepath.replace(folders.startDir, folders.newDir))
.replace(
sourceMap,
"//# sourceMappingURL=" +
filepath.split(/[\\\/]+/g).pop() +
".map"
);
await fs.promises.writeFile(
filepath.replace(folders.startDir, folders.newDir),
fileContent
);
await fs.promises.writeFile(
filepath.replace(folders.startDir, folders.newDir) + ".map",
JSON.stringify(sourceMapContent)
);
} else {
await fs.promises.copyFile(
filepath,
filepath.replace(folders.startDir, folders.newDir)
);
}
}
}else if(file.isDirectory()){
if(ignoreModules && file.name === "node_modules")continue
if(folders.exclude && folders.exclude.test(path.join(folder, file.name)))continue
await fs.promises.mkdir(path.join(folder, file.name).replace(folders.startDir, folders.newDir), {recursive: true})
await processNextDir(path.join(folder, file.name), ...Array.from(arguments).slice(1))
} else if (file.isDirectory()) {
if (ignoreModules && file.name === "node_modules") continue;
if (folders.exclude && folders.exclude.test(path.join(folder, file.name)))
continue;
await fs.promises.mkdir(
path.join(folder, file.name).replace(folders.startDir, folders.newDir),
{ recursive: true }
);
await processNextDir(
path.join(folder, file.name),
...Array.from(arguments).slice(1)
);
}
}
}
async function main(){
let startTimestamp = Date.now()
console.info("Starting build")
async function main() {
let startTimestamp = Date.now();
console.info("Starting build");
console.info("Reseting existent directory...")
try{
await fs.promises.rm("./distApp", {"recursive": true})
console.info("Reseting existent directory...");
try {
await fs.promises.rm("./distApp", { recursive: true });
} catch (error) {
console.error(error);
}
await fs.promises.mkdir(PROJECT_DIR+"/distApp/dist", {"recursive": true})
await fs.promises.mkdir(PROJECT_DIR + "/distApp/dist", { recursive: true });
console.info("Executing command `yarn compile`")
console.info("Executing command `yarn compile`");
child_process.execSync("yarn compile", {
encoding: "binary",
stdio: "inherit"
})
stdio: "inherit",
});
let startDir = path.join(PROJECT_DIR, "./dist")
let newDir = path.join(PROJECT_DIR, "./distApp/dist")
console.info("No error detected. Copying files from "+startDir+".")
await fs.promises.mkdir(startDir, {recursive: true})
let startDir = path.join(PROJECT_DIR, "./dist");
let newDir = path.join(PROJECT_DIR, "./distApp/dist");
console.info("No error detected. Copying files from " + startDir + ".");
await fs.promises.mkdir(startDir, { recursive: true });
await processNextDir(startDir, {
await processNextDir(
startDir,
newDir
}, ((filepath) => filepath.endsWith(".js")), async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`)
{
startDir,
newDir,
},
(filepath) => filepath.endsWith(".js"),
async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`);
if(filepath.endsWith("git.js")){
await fs.promises.writeFile(newpath, terser.minify(fs.readFileSync(filepath, "utf8").replace(/"{commit}"/g, `"${commit}"`)).code, "utf8")
}else{
await fs.promises.writeFile(newpath, terser.minify(await fs.promises.readFile(filepath, "utf8")).code, "utf8")
if (filepath.endsWith("git.js")) {
await fs.promises.writeFile(
newpath,
(
await terser.minify(
fs
.readFileSync(filepath, "utf8")
.replace(/"{commit}"/g, `"${commit}"`)
)
).code,
"utf8"
);
} else {
await fs.promises.writeFile(
newpath,
(
await terser.minify(await fs.promises.readFile(filepath, "utf8"))
).code,
"utf8"
);
}
}, true).then(() => {
console.info(`Copied files and minified them from ${startDir}.`)
})
},
true
).then(() => {
console.info(`Copied files and minified them from ${startDir}.`);
});
await processNextDir(path.join(PROJECT_DIR, "modules"), {
await processNextDir(
path.join(PROJECT_DIR, "modules"),
{
startDir: path.join(PROJECT_DIR, "modules"),
newDir: path.join(PROJECT_DIR, "distApp", "modules"),
exclude: /discord_spellcheck/g
}, ((filepath) => filepath.endsWith(".js")), async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`)
await fs.promises.writeFile(newpath, terser.minify(await fs.promises.readFile(filepath, "utf8")).code, "utf8")
}, true).then(() => {
console.info(`Copied files and minified them from ${path.join(PROJECT_DIR, "modules")}.`)
})
exclude: /discord_spellcheck/g,
},
(filepath) => filepath.endsWith(".js"),
async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`);
await fs.promises.writeFile(
newpath,
(await terser.minify(await fs.promises.readFile(filepath, "utf8"))).code,
"utf8"
);
},
true
).then(() => {
console.info(
`Copied files and minified them from ${path.join(
PROJECT_DIR,
"modules"
)}.`
);
});
await Promise.all((await fs.promises.readdir(path.join(PROJECT_DIR, "distApp", "modules"))).map(async mdl => {
let dir = path.join(PROJECT_DIR, "distApp", "modules", mdl)
await Promise.all(
(
await fs.promises.readdir(path.join(PROJECT_DIR, "distApp", "modules"))
).map(async (mdl) => {
let dir = path.join(PROJECT_DIR, "distApp", "modules", mdl);
if(!fs.existsSync(path.join(dir, "package.json"))){
if(mdl === "discord_desktop_core"){
dir = path.join(dir, "core")
}else{
return
if (!fs.existsSync(path.join(dir, "package.json"))) {
if (mdl === "discord_desktop_core") {
dir = path.join(dir, "core");
} else {
return;
}
}
console.info(`Installing modules for ${mdl}`)
console.info(`Installing modules for ${mdl}`);
child_process.execSync("yarn --production", {
encoding: "binary",
cwd: dir,
stdio: "inherit"
stdio: "inherit",
});
})
}))
);
await fs.promises.mkdir(path.join(PROJECT_DIR, "distApp", "modules", "discord_spellcheck"), {recursive: true})
await processNextDir(path.join(PROJECT_DIR, "modules", "discord_spellcheck"), {
await fs.promises.mkdir(
path.join(PROJECT_DIR, "distApp", "modules", "discord_spellcheck"),
{ recursive: true }
);
await processNextDir(
path.join(PROJECT_DIR, "modules", "discord_spellcheck"),
{
startDir: path.join(PROJECT_DIR, "modules", "discord_spellcheck"),
newDir: path.join(PROJECT_DIR, "distApp", "modules", "discord_spellcheck")
}, ((filepath) => filepath.endsWith(".js")), async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`)
await fs.promises.writeFile(newpath, terser.minify(await fs.promises.readFile(filepath, "utf8")).code, "utf8")
}, false).then(() => {
console.info(`Copied files and minified them from ${path.join(PROJECT_DIR, "modules")}.`)
})
newDir: path.join(
PROJECT_DIR,
"distApp",
"modules",
"discord_spellcheck"
),
},
(filepath) => filepath.endsWith(".js"),
async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`);
await fs.promises.writeFile(
newpath,
(await terser.minify(await fs.promises.readFile(filepath, "utf8"))).code,
"utf8"
);
},
false
).then(() => {
console.info(
`Copied files and minified them from ${path.join(
PROJECT_DIR,
"modules"
)}.`
);
});
await processNextDir(path.join(PROJECT_DIR, "LightcordApi"), {
await processNextDir(
path.join(PROJECT_DIR, "LightcordApi"),
{
startDir: path.join(PROJECT_DIR, "LightcordApi"),
newDir: path.join(PROJECT_DIR, "distApp", "LightcordApi"),
exclude: /(src|webpack\.config\.js|tsconfig\.json|dist|docs)/g
}, ((filepath) => filepath.endsWith(".js") && (!production ? !filepath.includes("node_modules") : true)), async (filepath, newpath) => {
await fs.promises.copyFile(filepath, newpath)
}, true).then(() => {
console.info(`Copied files and minified them from ${path.join(PROJECT_DIR, "LightcordApi")}.`)
})
exclude: /(src|webpack\.config\.js|tsconfig\.json|dist|docs)/g,
},
(filepath) =>
filepath.endsWith(".js") &&
(!production ? !filepath.includes("node_modules") : true),
async (filepath, newpath) => {
await fs.promises.copyFile(filepath, newpath);
},
true
).then(() => {
console.info(
`Copied files and minified them from ${path.join(
PROJECT_DIR,
"LightcordApi"
)}.`
);
});
child_process.execSync("yarn --production", {
encoding: "binary",
cwd: path.join(PROJECT_DIR, "distApp", "LightcordApi"),
stdio: "inherit"
})
stdio: "inherit",
});
function processDJS(dir){
fs.mkdirSync(path.join(PROJECT_DIR, "distApp", "DiscordJS", dir), {recursive: true})
return processNextDir(path.join(PROJECT_DIR, "DiscordJS", dir), {
function processDJS(dir) {
fs.mkdirSync(path.join(PROJECT_DIR, "distApp", "DiscordJS", dir), {
recursive: true,
});
return processNextDir(
path.join(PROJECT_DIR, "DiscordJS", dir),
{
startDir: path.join(PROJECT_DIR, "DiscordJS", dir),
newDir: path.join(PROJECT_DIR, "distApp", "DiscordJS", dir),
exclude: /node_modules/g
}, ((filepath) => filepath.endsWith(".js")), async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`)
await fs.promises.writeFile(newpath, terser.minify(await fs.promises.readFile(filepath, "utf8")).code, "utf8")
}).then(() => {
console.info(`Copied files and minified them from ${path.join(PROJECT_DIR, "DiscordJS", dir)}.`)
})
exclude: /node_modules/g,
},
(filepath) => filepath.endsWith(".js"),
async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`);
await fs.promises.writeFile(
newpath,
(await terser.minify(await fs.promises.readFile(filepath, "utf8"))).code,
"utf8"
);
}
async function copyFileDJS(file){
await fs.promises.writeFile(path.join(PROJECT_DIR, "distApp", "DiscordJS", file), await fs.promises.readFile(path.join(PROJECT_DIR, "DiscordJS", file)))
).then(() => {
console.info(
`Copied files and minified them from ${path.join(
PROJECT_DIR,
"DiscordJS",
dir
)}.`
);
});
}
async function copyFileDJS(file) {
await fs.promises.writeFile(
path.join(PROJECT_DIR, "distApp", "DiscordJS", file),
await fs.promises.readFile(path.join(PROJECT_DIR, "DiscordJS", file))
);
}
await processDJS("dist")
await copyFileDJS("package.json")
await processDJS("dist");
await copyFileDJS("package.json");
child_process.execSync("yarn --production", {
encoding: "binary",
cwd: path.join(PROJECT_DIR, "distApp", "DiscordJS"),
stdio: "inherit"
})
stdio: "inherit",
});
fs.mkdirSync(path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "dist"), {recursive: true})
const BDPackageJSON = require("../BetterDiscordApp/package.json")
fs.writeFileSync(path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "package.json"), JSON.stringify(BDPackageJSON), "utf8")
const files = [
"index.min.js",
"style.min.css"
]
files.forEach(e => {
files.push(e + ".map")
})
files.forEach(e => {
const pth = path.join(PROJECT_DIR, "BetterDiscordApp", "dist", e)
if(!fs.existsSync(pth))return console.error(`\x1b[31mFile ${pth} from betterdiscord does not exist.\x1b[0m`)
if(e.endsWith(".map")){
const data = JSON.parse(fs.readFileSync(pth, "utf8"))
data.sourcesContent = []
fs.writeFileSync(path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "dist", e), JSON.stringify(data))
}else{
fs.copyFileSync(pth, path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "dist", e))
fs.mkdirSync(path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "dist"), {
recursive: true,
});
const BDPackageJSON = require("../BetterDiscordApp/package.json");
fs.writeFileSync(
path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "package.json"),
JSON.stringify(BDPackageJSON),
"utf8"
);
const files = ["index.min.js", "style.min.css"];
files.forEach((e) => {
files.push(e + ".map");
});
files.forEach((e) => {
const pth = path.join(PROJECT_DIR, "BetterDiscordApp", "dist", e);
if (!fs.existsSync(pth))
return console.error(
`\x1b[31mFile ${pth} from betterdiscord does not exist.\x1b[0m`
);
if (e.endsWith(".map")) {
const data = JSON.parse(fs.readFileSync(pth, "utf8"));
data.sourcesContent = [];
fs.writeFileSync(
path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "dist", e),
JSON.stringify(data)
);
} else {
fs.copyFileSync(
pth,
path.join(PROJECT_DIR, "distApp", "BetterDiscordApp", "dist", e)
);
}
})
});
await fs.promises.mkdir(path.join(PROJECT_DIR, "distApp", "splash", "videos"), {recursive: true})
await processNextDir(path.join(PROJECT_DIR, "splash"), {
await fs.promises.mkdir(
path.join(PROJECT_DIR, "distApp", "splash", "videos"),
{ recursive: true }
);
await processNextDir(
path.join(PROJECT_DIR, "splash"),
{
startDir: path.join(PROJECT_DIR, "splash"),
newDir: path.join(PROJECT_DIR, "distApp", "splash"),
exclude: /node_modules/g
}, (filepath) => {
if(filepath.endsWith(".js"))return true
return false
}, async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`)
await fs.promises.writeFile(newpath, terser.minify(await fs.promises.readFile(filepath, "utf8")).code, "utf8")
}).then(() => {
console.info(`Copied files and minified them from ${path.join(PROJECT_DIR, "splash")}.`)
})
fs.writeFileSync(path.join(PROJECT_DIR, "distApp", "LICENSE"), fs.readFileSync(path.join(PROJECT_DIR, "LICENSE")))
exclude: /node_modules/g,
},
(filepath) => {
if (filepath.endsWith(".js")) return true;
return false;
},
async (filepath, newpath) => {
console.info(`Minifying ${filepath} to ${newpath}`);
await fs.promises.writeFile(
newpath,
(await terser.minify(await fs.promises.readFile(filepath, "utf8"))).code,
"utf8"
);
}
).then(() => {
console.info(
`Copied files and minified them from ${path.join(PROJECT_DIR, "splash")}.`
);
});
fs.writeFileSync(
path.join(PROJECT_DIR, "distApp", "LICENSE"),
fs.readFileSync(path.join(PROJECT_DIR, "LICENSE"))
);
let packageJSON = require("../package.json")
packageJSON.scripts["build:electron_linux"] = packageJSON.scripts["build:electron_linux"].replace("./distApp", ".")
packageJSON.scripts["build:electron_win"] = packageJSON.scripts["build:electron_win"].replace("./distApp", ".")
packageJSON.scripts["build:electron_mac"] = packageJSON.scripts["build:electron_mac"].replace("./distApp", ".")
let packageJSON = require("../package.json");
packageJSON.scripts["build:electron_linux"] = packageJSON.scripts[
"build:electron_linux"
].replace("./distApp", ".");
packageJSON.scripts["build:electron_win"] = packageJSON.scripts[
"build:electron_win"
].replace("./distApp", ".");
packageJSON.scripts["build:electron_mac"] = packageJSON.scripts[
"build:electron_mac"
].replace("./distApp", ".");
fs.writeFileSync(path.join(PROJECT_DIR, "distApp", "package.json"), JSON.stringify(packageJSON), "utf8")
fs.writeFileSync(
path.join(PROJECT_DIR, "distApp", "package.json"),
JSON.stringify(packageJSON),
"utf8"
);
console.info(`Installing ${Object.keys(packageJSON.dependencies).length} packages...`)
console.info(
`Installing ${Object.keys(packageJSON.dependencies).length} packages...`
);
child_process.execSync("yarn --production", {
encoding: "binary",
cwd: path.join(PROJECT_DIR, "distApp"),
stdio: "inherit"
})
console.info("Build took "+(Date.now() - startTimestamp) +"ms.")
stdio: "inherit",
});
console.info("Build took " + (Date.now() - startTimestamp) + "ms.");
}
main()
.catch(err => {
console.error(err)
process.exit(1)
})
main().catch((err) => {
console.error(err);
process.exit(1);
});