2017-11-19 03:48:21 -05:00
|
|
|
/**
|
|
|
|
* The initial iteration of this file will not support optional submodules.
|
|
|
|
* Support will be added down the line, only top-level modules will recieve optional support.
|
|
|
|
*/
|
|
|
|
const AdmZip = require('adm-zip')
|
2017-11-30 03:00:06 -05:00
|
|
|
const {AssetGuard, Library} = require('./assetguard.js')
|
2017-11-19 03:48:21 -05:00
|
|
|
const child_process = require('child_process')
|
2017-12-03 00:38:22 -05:00
|
|
|
const ConfigManager = require('./configmanager.js')
|
2018-04-15 00:00:08 -04:00
|
|
|
const crypto = require('crypto')
|
2017-11-19 03:48:21 -05:00
|
|
|
const fs = require('fs')
|
|
|
|
const mkpath = require('mkdirp')
|
2018-04-15 00:00:08 -04:00
|
|
|
const os = require('os')
|
2017-11-19 03:48:21 -05:00
|
|
|
const path = require('path')
|
2018-04-15 00:00:08 -04:00
|
|
|
const rimraf = require('rimraf')
|
2017-11-30 01:40:56 -05:00
|
|
|
const {URL} = require('url')
|
2017-11-19 03:48:21 -05:00
|
|
|
|
|
|
|
class ProcessBuilder {
|
|
|
|
|
|
|
|
constructor(gameDirectory, distroServer, versionData, forgeData, authUser){
|
|
|
|
this.dir = gameDirectory
|
|
|
|
this.server = distroServer
|
|
|
|
this.versionData = versionData
|
|
|
|
this.forgeData = forgeData
|
|
|
|
this.authUser = authUser
|
|
|
|
this.fmlDir = path.join(this.dir, 'versions', this.server.id + '.json')
|
|
|
|
this.libPath = path.join(this.dir, 'libraries')
|
|
|
|
}
|
|
|
|
|
|
|
|
static shouldInclude(mdle){
|
|
|
|
//If the module should be included by default
|
|
|
|
return mdle.required == null || mdle.required.value == null || mdle.required.value === true || (mdle.required.value === false && (mdle.required.def == null || mdle.required.def === true))
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convienence method to run the functions typically used to build a process.
|
|
|
|
*/
|
|
|
|
build(){
|
2018-04-15 00:00:08 -04:00
|
|
|
const tempNativePath = path.join(os.tmpdir(), ConfigManager.getTempNativeFolder(), crypto.pseudoRandomBytes(16).toString('hex'))
|
2017-11-30 01:40:56 -05:00
|
|
|
process.throwDeprecation = true
|
2017-11-19 03:48:21 -05:00
|
|
|
const mods = this.resolveDefaultMods()
|
|
|
|
this.constructFMLModList(mods, true)
|
2018-04-15 00:00:08 -04:00
|
|
|
const args = this.constructJVMArguments(mods, tempNativePath)
|
2017-11-19 03:48:21 -05:00
|
|
|
|
2017-11-30 01:40:56 -05:00
|
|
|
console.log(args)
|
2017-11-19 03:48:21 -05:00
|
|
|
|
2018-03-31 13:05:05 -04:00
|
|
|
const child = child_process.spawn(ConfigManager.getJavaExecutable(), args, {
|
2018-05-10 00:01:38 -04:00
|
|
|
cwd: ConfigManager.getGameDirectory(),
|
|
|
|
detached: ConfigManager.isLaunchDetached()
|
2018-03-31 13:05:05 -04:00
|
|
|
})
|
2017-11-19 03:48:21 -05:00
|
|
|
|
2018-05-10 00:01:38 -04:00
|
|
|
if(ConfigManager.isLaunchDetached()){
|
|
|
|
child.unref()
|
|
|
|
}
|
|
|
|
|
2017-11-19 03:48:21 -05:00
|
|
|
child.stdout.on('data', (data) => {
|
|
|
|
console.log('Minecraft:', data.toString('utf8'))
|
|
|
|
})
|
|
|
|
child.stderr.on('data', (data) => {
|
|
|
|
console.log('Minecraft:', data.toString('utf8'))
|
|
|
|
})
|
|
|
|
child.on('close', (code, signal) => {
|
|
|
|
console.log('Exited with code', code)
|
2018-04-15 00:00:08 -04:00
|
|
|
rimraf(tempNativePath, (err) => {
|
|
|
|
if(err){
|
|
|
|
console.warn('Error while deleting temp dir', err)
|
|
|
|
} else {
|
|
|
|
console.log('Temp dir deleted successfully.')
|
|
|
|
}
|
|
|
|
})
|
2017-11-19 03:48:21 -05:00
|
|
|
})
|
|
|
|
|
|
|
|
return child
|
|
|
|
}
|
|
|
|
|
|
|
|
resolveDefaultMods(options = {type: 'forgemod'}){
|
|
|
|
//Returns array of default forge mods to load.
|
|
|
|
const mods = []
|
|
|
|
const mdles = this.server.modules
|
|
|
|
|
|
|
|
for(let i=0; i<mdles.length; ++i){
|
|
|
|
if(mdles[i].type != null && mdles[i].type === options.type){
|
|
|
|
if(ProcessBuilder.shouldInclude(mdles[i])){
|
|
|
|
mods.push(mdles[i])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mods
|
|
|
|
}
|
|
|
|
|
|
|
|
constructFMLModList(mods, save = false){
|
|
|
|
const modList = {}
|
|
|
|
modList.repositoryRoot = path.join(this.dir, 'modstore')
|
|
|
|
const ids = []
|
|
|
|
for(let i=0; i<mods.length; ++i){
|
|
|
|
ids.push(mods[i].id)
|
|
|
|
}
|
|
|
|
modList.modRef = ids
|
|
|
|
|
|
|
|
if(save){
|
|
|
|
const json = JSON.stringify(modList, null, 4)
|
|
|
|
fs.writeFileSync(this.fmlDir, json, 'UTF-8')
|
|
|
|
}
|
|
|
|
|
|
|
|
return modList
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Construct the argument array that will be passed to the JVM process.
|
|
|
|
*
|
2018-03-28 16:42:10 -04:00
|
|
|
* @param {Array.<Object>} mods An array of enabled mods which will be launched with this process.
|
2018-04-15 00:00:08 -04:00
|
|
|
* @param {string} tempNativePath The path to store the native libraries.
|
2018-03-28 16:42:10 -04:00
|
|
|
* @returns {Array.<string>} An array containing the full JVM arguments for this process.
|
2017-11-19 03:48:21 -05:00
|
|
|
*/
|
2018-04-15 00:00:08 -04:00
|
|
|
constructJVMArguments(mods, tempNativePath){
|
|
|
|
|
2017-12-03 00:38:22 -05:00
|
|
|
let args = ['-Xmx' + ConfigManager.getMaxRAM(),
|
2018-04-15 01:26:40 -04:00
|
|
|
'-Xms' + ConfigManager.getMinRAM(),
|
2018-04-15 00:00:08 -04:00
|
|
|
'-Djava.library.path=' + tempNativePath,
|
2017-11-19 03:48:21 -05:00
|
|
|
'-cp',
|
2018-04-15 00:21:26 -04:00
|
|
|
this.classpathArg(mods, tempNativePath).join(process.platform === 'win32' ? ';' : ':'),
|
2017-11-19 03:48:21 -05:00
|
|
|
this.forgeData.mainClass]
|
|
|
|
|
2018-04-15 00:49:20 -04:00
|
|
|
if(process.platform === 'darwin'){
|
|
|
|
args.unshift('-Xdock:name=WesterosCraft')
|
|
|
|
args.unshift('-Xdock:icon=' + path.join(__dirname, '..', 'images', 'minecraft.icns'))
|
|
|
|
}
|
|
|
|
|
2018-04-15 01:26:40 -04:00
|
|
|
args.splice(2, 0, ...ConfigManager.getJVMOptions())
|
2017-11-30 01:40:56 -05:00
|
|
|
|
2017-11-19 03:48:21 -05:00
|
|
|
args = args.concat(this._resolveForgeArgs())
|
|
|
|
|
|
|
|
return args
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolve the arguments required by forge.
|
|
|
|
*
|
2018-03-28 16:42:10 -04:00
|
|
|
* @returns {Array.<string>} An array containing the arguments required by forge.
|
2017-11-19 03:48:21 -05:00
|
|
|
*/
|
|
|
|
_resolveForgeArgs(){
|
|
|
|
const mcArgs = this.forgeData.minecraftArguments.split(' ')
|
|
|
|
const argDiscovery = /\${*(.*)}/
|
|
|
|
|
|
|
|
// Replace the declared variables with their proper values.
|
|
|
|
for(let i=0; i<mcArgs.length; ++i){
|
|
|
|
if(argDiscovery.test(mcArgs[i])){
|
|
|
|
const identifier = mcArgs[i].match(argDiscovery)[1]
|
|
|
|
let val = null;
|
|
|
|
switch(identifier){
|
|
|
|
case 'auth_player_name':
|
2017-12-03 08:12:55 -05:00
|
|
|
val = this.authUser.displayName
|
2017-11-19 03:48:21 -05:00
|
|
|
break
|
|
|
|
case 'version_name':
|
|
|
|
//val = versionData.id
|
|
|
|
val = this.server.id
|
|
|
|
break
|
|
|
|
case 'game_directory':
|
|
|
|
val = this.dir
|
|
|
|
break
|
|
|
|
case 'assets_root':
|
|
|
|
val = path.join(this.dir, 'assets')
|
|
|
|
break
|
|
|
|
case 'assets_index_name':
|
|
|
|
val = this.versionData.assets
|
|
|
|
break
|
|
|
|
case 'auth_uuid':
|
2017-12-03 08:12:55 -05:00
|
|
|
val = this.authUser.uuid
|
2017-11-19 03:48:21 -05:00
|
|
|
break
|
|
|
|
case 'auth_access_token':
|
|
|
|
val = this.authUser.accessToken
|
|
|
|
break
|
|
|
|
case 'user_type':
|
|
|
|
val = 'MOJANG'
|
|
|
|
break
|
|
|
|
case 'version_type':
|
|
|
|
val = this.versionData.type
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if(val != null){
|
|
|
|
mcArgs[i] = val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mcArgs.push('--modListFile')
|
|
|
|
mcArgs.push('absolute:' + this.fmlDir)
|
2017-11-30 01:40:56 -05:00
|
|
|
|
|
|
|
// Prepare game resolution
|
2017-12-03 00:38:22 -05:00
|
|
|
if(ConfigManager.isFullscreen()){
|
2017-11-30 01:40:56 -05:00
|
|
|
mcArgs.unshift('--fullscreen')
|
|
|
|
} else {
|
2017-12-03 00:38:22 -05:00
|
|
|
mcArgs.unshift(ConfigManager.getGameWidth())
|
2017-11-30 01:40:56 -05:00
|
|
|
mcArgs.unshift('--width')
|
2017-12-03 00:38:22 -05:00
|
|
|
mcArgs.unshift(ConfigManager.getGameHeight())
|
2017-11-30 01:40:56 -05:00
|
|
|
mcArgs.unshift('--height')
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prepare autoconnect
|
2017-12-03 00:38:22 -05:00
|
|
|
if(ConfigManager.isAutoConnect() && this.server.autoconnect){
|
2017-11-30 01:40:56 -05:00
|
|
|
const serverURL = new URL('my://' + this.server.server_ip)
|
|
|
|
mcArgs.unshift(serverURL.hostname)
|
|
|
|
mcArgs.unshift('--server')
|
|
|
|
if(serverURL.port){
|
|
|
|
mcArgs.unshift(serverURL.port)
|
|
|
|
mcArgs.unshift('--port')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-19 03:48:21 -05:00
|
|
|
return mcArgs
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolve the full classpath argument list for this process. This method will resolve all Mojang-declared
|
|
|
|
* libraries as well as the libraries declared by the server. Since mods are permitted to declare libraries,
|
|
|
|
* this method requires all enabled mods as an input
|
|
|
|
*
|
2018-03-28 16:42:10 -04:00
|
|
|
* @param {Array.<Object>} mods An array of enabled mods which will be launched with this process.
|
2018-04-15 00:00:08 -04:00
|
|
|
* @param {string} tempNativePath The path to store the native libraries.
|
2018-03-28 16:42:10 -04:00
|
|
|
* @returns {Array.<string>} An array containing the paths of each library required by this process.
|
2017-11-19 03:48:21 -05:00
|
|
|
*/
|
2018-04-15 00:00:08 -04:00
|
|
|
classpathArg(mods, tempNativePath){
|
2017-11-19 03:48:21 -05:00
|
|
|
let cpArgs = []
|
|
|
|
|
|
|
|
// Add the version.jar to the classpath.
|
|
|
|
const version = this.versionData.id
|
|
|
|
cpArgs.push(path.join(this.dir, 'versions', version, version + '.jar'))
|
|
|
|
|
|
|
|
// Resolve the Mojang declared libraries.
|
2018-04-15 00:00:08 -04:00
|
|
|
const mojangLibs = this._resolveMojangLibraries(tempNativePath)
|
2017-11-19 03:48:21 -05:00
|
|
|
cpArgs = cpArgs.concat(mojangLibs)
|
|
|
|
|
|
|
|
// Resolve the server declared libraries.
|
|
|
|
const servLibs = this._resolveServerLibraries(mods)
|
|
|
|
cpArgs = cpArgs.concat(servLibs)
|
|
|
|
|
|
|
|
return cpArgs
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolve the libraries defined by Mojang's version data. This method will also extract
|
|
|
|
* native libraries and point to the correct location for its classpath.
|
|
|
|
*
|
|
|
|
* TODO - clean up function
|
|
|
|
*
|
2018-04-15 00:00:08 -04:00
|
|
|
* @param {string} tempNativePath The path to store the native libraries.
|
2018-03-28 16:42:10 -04:00
|
|
|
* @returns {Array.<string>} An array containing the paths of each library mojang declares.
|
2017-11-19 03:48:21 -05:00
|
|
|
*/
|
2018-04-15 00:00:08 -04:00
|
|
|
_resolveMojangLibraries(tempNativePath){
|
2017-11-19 03:48:21 -05:00
|
|
|
const libs = []
|
|
|
|
|
|
|
|
const libArr = this.versionData.libraries
|
2018-04-15 00:00:08 -04:00
|
|
|
mkpath.sync(tempNativePath)
|
2017-11-19 03:48:21 -05:00
|
|
|
for(let i=0; i<libArr.length; i++){
|
|
|
|
const lib = libArr[i]
|
2017-11-30 03:00:06 -05:00
|
|
|
if(Library.validateRules(lib.rules)){
|
2017-11-19 03:48:21 -05:00
|
|
|
if(lib.natives == null){
|
|
|
|
const dlInfo = lib.downloads
|
|
|
|
const artifact = dlInfo.artifact
|
|
|
|
const to = path.join(this.libPath, artifact.path)
|
|
|
|
libs.push(to)
|
|
|
|
} else {
|
|
|
|
// Extract the native library.
|
|
|
|
const natives = lib.natives
|
|
|
|
const extractInst = lib.extract
|
|
|
|
const exclusionArr = extractInst.exclude
|
2017-11-30 03:00:06 -05:00
|
|
|
const opSys = Library.mojangFriendlyOS()
|
2017-11-19 03:48:21 -05:00
|
|
|
const indexId = natives[opSys]
|
|
|
|
const dlInfo = lib.downloads
|
|
|
|
const classifiers = dlInfo.classifiers
|
|
|
|
const artifact = classifiers[indexId]
|
|
|
|
|
|
|
|
// Location of native zip.
|
|
|
|
const to = path.join(this.libPath, artifact.path)
|
|
|
|
|
|
|
|
let zip = new AdmZip(to)
|
|
|
|
let zipEntries = zip.getEntries()
|
|
|
|
|
|
|
|
// Unzip the native zip.
|
|
|
|
for(let i=0; i<zipEntries.length; i++){
|
|
|
|
const fileName = zipEntries[i].entryName
|
|
|
|
|
|
|
|
let shouldExclude = false
|
|
|
|
|
|
|
|
// Exclude noted files.
|
|
|
|
exclusionArr.forEach(function(exclusion){
|
2017-11-27 04:31:54 -05:00
|
|
|
if(fileName.indexOf(exclusion) > -1){
|
2017-11-19 03:48:21 -05:00
|
|
|
shouldExclude = true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// Extract the file.
|
|
|
|
if(!shouldExclude){
|
2018-04-15 00:00:08 -04:00
|
|
|
fs.writeFile(path.join(tempNativePath, fileName), zipEntries[i].getData(), (err) => {
|
2017-11-30 01:40:56 -05:00
|
|
|
if(err){
|
|
|
|
console.error('Error while extracting native library:', err)
|
|
|
|
}
|
|
|
|
})
|
2017-11-19 03:48:21 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return libs
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolve the libraries declared by this server in order to add them to the classpath.
|
|
|
|
* This method will also check each enabled mod for libraries, as mods are permitted to
|
|
|
|
* declare libraries.
|
|
|
|
*
|
2018-03-28 16:42:10 -04:00
|
|
|
* @param {Array.<Object>} mods An array of enabled mods which will be launched with this process.
|
|
|
|
* @returns {Array.<string>} An array containing the paths of each library this server requires.
|
2017-11-19 03:48:21 -05:00
|
|
|
*/
|
|
|
|
_resolveServerLibraries(mods){
|
|
|
|
const mdles = this.server.modules
|
|
|
|
let libs = []
|
|
|
|
|
2017-11-19 04:12:41 -05:00
|
|
|
// Locate Forge/Libraries
|
2017-11-19 03:48:21 -05:00
|
|
|
for(let i=0; i<mdles.length; i++){
|
2017-11-19 04:12:41 -05:00
|
|
|
if(mdles[i].type != null && (mdles[i].type === 'forge-hosted' || mdles[i].type === 'library')){
|
|
|
|
let lib = mdles[i]
|
2017-11-30 03:00:06 -05:00
|
|
|
libs.push(path.join(this.libPath, lib.artifact.path == null ? AssetGuard._resolvePath(lib.id, lib.artifact.extension) : lib.artifact.path))
|
2017-11-19 04:12:41 -05:00
|
|
|
if(lib.sub_modules != null){
|
|
|
|
const res = this._resolveModuleLibraries(lib)
|
|
|
|
if(res.length > 0){
|
|
|
|
libs = libs.concat(res)
|
|
|
|
}
|
2017-11-19 03:48:21 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Check for any libraries in our mod list.
|
|
|
|
for(let i=0; i<mods.length; i++){
|
|
|
|
if(mods.sub_modules != null){
|
|
|
|
const res = this._resolveModuleLibraries(mods[i])
|
|
|
|
if(res.length > 0){
|
|
|
|
libs = libs.concat(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return libs
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively resolve the path of each library required by this module.
|
|
|
|
*
|
2018-03-28 16:42:10 -04:00
|
|
|
* @param {Object} mdle A module object from the server distro index.
|
|
|
|
* @returns {Array.<string>} An array containing the paths of each library this module requires.
|
2017-11-19 03:48:21 -05:00
|
|
|
*/
|
|
|
|
_resolveModuleLibraries(mdle){
|
|
|
|
if(mdle.sub_modules == null){
|
|
|
|
return []
|
|
|
|
}
|
|
|
|
let libs = []
|
|
|
|
for(let i=0; i<mdle.sub_modules.length; i++){
|
|
|
|
const sm = mdle.sub_modules[i]
|
|
|
|
if(sm.type != null && sm.type == 'library'){
|
2017-11-30 03:00:06 -05:00
|
|
|
libs.push(path.join(this.libPath, sm.artifact.path == null ? AssetGuard._resolvePath(sm.id, sm.artifact.extension) : sm.artifact.path))
|
2017-11-19 03:48:21 -05:00
|
|
|
}
|
|
|
|
// If this module has submodules, we need to resolve the libraries for those.
|
|
|
|
// To avoid unnecessary recursive calls, base case is checked here.
|
|
|
|
if(mdle.sub_modules != null){
|
|
|
|
const res = this._resolveModuleLibraries(sm)
|
|
|
|
if(res.length > 0){
|
|
|
|
libs = libs.concat(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return libs
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = ProcessBuilder
|