2018-10-29 18:13:58 -04:00
|
|
|
const child = require('child_process');
|
|
|
|
const path = require('path');
|
|
|
|
const handler = require('./handler');
|
2019-05-22 18:28:48 -04:00
|
|
|
const packager = require('./package');
|
2018-10-29 18:13:58 -04:00
|
|
|
const fs = require('fs');
|
2019-05-22 18:28:48 -04:00
|
|
|
const EventEmitter = require('events').EventEmitter;
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
class MCLCore extends EventEmitter {
|
|
|
|
constructor(options) {
|
|
|
|
super();
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
this.options = options;
|
|
|
|
this.handler = new handler(this);
|
2019-05-23 19:40:34 -04:00
|
|
|
this.pid = null;
|
2019-04-28 17:52:37 -04:00
|
|
|
}
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-23 20:40:44 -04:00
|
|
|
async launch(authorization) {
|
|
|
|
if(!authorization) throw Error('No authorization to launch the client with!');
|
|
|
|
|
|
|
|
if({}.toString.call(authorization) === "[object Promise]") {
|
|
|
|
this.options.authorization = await authorization;
|
|
|
|
} else {
|
|
|
|
this.options.authorization = authorization
|
|
|
|
}
|
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
this.options.root = path.resolve(this.options.root);
|
|
|
|
if(!fs.existsSync(this.options.root)) {
|
|
|
|
this.emit('debug', '[MCLC]: Attempting to create root folder');
|
|
|
|
fs.mkdirSync(this.options.root);
|
|
|
|
}
|
2018-12-01 13:26:15 -05:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
if(this.options.clientPackage) {
|
|
|
|
this.emit('debug', `[MCLC]: Extracting client package to ${this.options.root}`);
|
|
|
|
await packager.extractPackage(this.options.root, this.options.clientPackage);
|
|
|
|
}
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
const directory = path.join(this.options.root, 'versions', this.options.version.number);
|
|
|
|
this.options.directory = directory;
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
// Version JSON for the main launcher folder
|
|
|
|
const versionFile = await this.handler.getVersion();
|
|
|
|
const mcPath = this.options.version.custom ? path.join(this.options.root, 'versions', this.options.version.custom , `${this.options.version.custom}.jar`):
|
|
|
|
path.join(directory, `${this.options.version.number}.jar`);
|
|
|
|
const nativePath = await this.handler.getNatives();
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
if (!fs.existsSync(mcPath)) {
|
|
|
|
this.emit('debug', '[MCLC]: Attempting to download Minecraft version jar');
|
|
|
|
await this.handler.getJar();
|
|
|
|
}
|
|
|
|
|
|
|
|
let forge = null;
|
|
|
|
let custom = null;
|
|
|
|
if(this.options.forge) {
|
|
|
|
this.emit('debug', '[MCLC]: Detected Forge in options, getting dependencies');
|
|
|
|
forge = await this.handler.getForgeDependencies();
|
|
|
|
}
|
|
|
|
if(this.options.version.custom) {
|
|
|
|
this.emit('debug', '[MCLC]: Detected custom in options, setting custom version file');
|
|
|
|
custom = require(path.join(this.options.root, 'versions', this.options.version.custom, `${this.options.version.custom}.json`));
|
|
|
|
}
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
const args = [];
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
// Jvm
|
|
|
|
let jvm = [
|
|
|
|
'-XX:-UseAdaptiveSizePolicy',
|
|
|
|
'-XX:-OmitStackTraceInFastThrow',
|
|
|
|
'-Dfml.ignorePatchDiscrepancies=true',
|
|
|
|
'-Dfml.ignoreInvalidMinecraftCertificates=true',
|
|
|
|
`-Djava.library.path=${nativePath}`,
|
|
|
|
`-Xmx${this.options.memory.max}M`,
|
|
|
|
`-Xms${this.options.memory.min}M`
|
|
|
|
];
|
|
|
|
jvm.push(await this.handler.getJVM());
|
|
|
|
if(this.options.customArgs) jvm = jvm.concat(this.options.customArgs);
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
const classes = await this.handler.getClasses();
|
|
|
|
let classPaths = ['-cp'];
|
|
|
|
const separator = this.options.os === "windows" ? ";" : ":";
|
|
|
|
this.emit('debug', `[MCLC]: Using ${separator} to separate class paths`);
|
|
|
|
if(forge) {
|
|
|
|
this.emit('debug', '[MCLC]: Setting Forge class paths');
|
|
|
|
classPaths.push(`${this.options.forge.path || this.options.forge}${separator}${forge.paths.join(separator)}${separator}${classes.join(separator)}${separator}${mcPath}`);
|
|
|
|
classPaths.push(forge.forge.mainClass)
|
|
|
|
} else {
|
|
|
|
const file = custom || versionFile;
|
|
|
|
classPaths.push(`${mcPath}${separator}${classes.join(separator)}`);
|
|
|
|
classPaths.push(file.mainClass);
|
|
|
|
}
|
|
|
|
classPaths = await handler.cleanUp(classPaths);
|
|
|
|
|
|
|
|
// Download version's assets
|
|
|
|
this.emit('debug', '[MCLC]: Attempting to download assets');
|
|
|
|
await this.handler.getAssets();
|
|
|
|
|
|
|
|
// Launch options. Thank you Lyrus for the reformat <3
|
|
|
|
const modification = forge ? forge.forge : null || custom ? custom : null;
|
|
|
|
const launchOptions = await this.handler.getLaunchOptions(modification);
|
|
|
|
|
|
|
|
const launchArguments = args.concat(jvm, classPaths, launchOptions);
|
2019-05-23 19:40:34 -04:00
|
|
|
this.emit('arguments', launchArguments);
|
|
|
|
this.emit('debug', launchArguments.join(' '));
|
2019-05-22 18:28:48 -04:00
|
|
|
|
|
|
|
const minecraft = child.spawn(this.options.javaPath ? this.options.javaPath : 'java', launchArguments);
|
|
|
|
minecraft.stdout.on('data', (data) => this.emit('data', data));
|
|
|
|
minecraft.stderr.on('data', (data) => this.emit('error', data));
|
|
|
|
minecraft.on('close', (code) => this.emit('close', code));
|
2019-05-23 19:40:34 -04:00
|
|
|
|
|
|
|
this.pid = minecraft.pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
async close() {
|
|
|
|
child.exec(`taskkill /PID ${this.pid}`, (err, out, error) => {return {err, out, error}})
|
|
|
|
}
|
|
|
|
|
|
|
|
async restart() {
|
|
|
|
await this.close();
|
|
|
|
await this.launch()
|
2019-05-22 18:28:48 -04:00
|
|
|
}
|
|
|
|
}
|
2018-10-29 18:13:58 -04:00
|
|
|
|
2019-05-22 18:28:48 -04:00
|
|
|
module.exports = MCLCore;
|