|
|
@@ -0,0 +1,327 @@
|
|
|
+/**
|
|
|
+ * 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.
|
|
|
+ *
|
|
|
+ *
|
|
|
+ * TODO why are logs not working??????
|
|
|
+ */
|
|
|
+const AdmZip = require('adm-zip')
|
|
|
+const ag = require('./assetguard.js')
|
|
|
+const child_process = require('child_process')
|
|
|
+const fs = require('fs')
|
|
|
+const mkpath = require('mkdirp')
|
|
|
+const path = require('path')
|
|
|
+
|
|
|
+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(){
|
|
|
+ const mods = this.resolveDefaultMods()
|
|
|
+ this.constructFMLModList(mods, true)
|
|
|
+ const args = this.constructJVMArguments(mods)
|
|
|
+
|
|
|
+ //console.log(args)
|
|
|
+
|
|
|
+ const child = child_process.spawn('C:\\Program Files\\Java\\jdk1.8.0_152\\bin\\javaw.exe', args)
|
|
|
+
|
|
|
+ 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)
|
|
|
+ })
|
|
|
+
|
|
|
+ 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.
|
|
|
+ *
|
|
|
+ * @param {Array.<Object>} mods - An array of enabled mods which will be launched with this process.
|
|
|
+ * @returns {Array.<String>} - An array containing the full JVM arguments for this process.
|
|
|
+ */
|
|
|
+ constructJVMArguments(mods){
|
|
|
+
|
|
|
+ let args = ['-Xmx4G',
|
|
|
+ '-XX:+UseConcMarkSweepGC',
|
|
|
+ '-XX:+CMSIncrementalMode',
|
|
|
+ '-XX:-UseAdaptiveSizePolicy',
|
|
|
+ '-Xmn128M',
|
|
|
+ '-Djava.library.path=' + path.join(this.dir, 'natives'),
|
|
|
+ '-cp',
|
|
|
+ this.classpathArg(mods).join(';'),
|
|
|
+ this.forgeData.mainClass]
|
|
|
+
|
|
|
+ args = args.concat(this._resolveForgeArgs())
|
|
|
+
|
|
|
+ return args
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Resolve the arguments required by forge.
|
|
|
+ *
|
|
|
+ * @returns {Array.<String>} - An array containing the arguments required by forge.
|
|
|
+ */
|
|
|
+ _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':
|
|
|
+ val = this.authUser.selectedProfile.name
|
|
|
+ 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':
|
|
|
+ val = this.authUser.selectedProfile.id
|
|
|
+ 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)
|
|
|
+ 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
|
|
|
+ *
|
|
|
+ * @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 required by this process.
|
|
|
+ */
|
|
|
+ classpathArg(mods){
|
|
|
+ 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.
|
|
|
+ const mojangLibs = this._resolveMojangLibraries()
|
|
|
+ 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
|
|
|
+ *
|
|
|
+ * @returns {Array.<String>} - An array containing the paths of each library mojang declares.
|
|
|
+ */
|
|
|
+ _resolveMojangLibraries(){
|
|
|
+ const libs = []
|
|
|
+
|
|
|
+ const libArr = this.versionData.libraries
|
|
|
+ const nativePath = path.join(this.dir, 'natives')
|
|
|
+ for(let i=0; i<libArr.length; i++){
|
|
|
+ const lib = libArr[i]
|
|
|
+ if(ag.Library.validateRules(lib.rules)){
|
|
|
+ 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
|
|
|
+ const opSys = ag.Library.mojangFriendlyOS()
|
|
|
+ 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){
|
|
|
+ if(exclusion.indexOf(fileName) > -1){
|
|
|
+ shouldExclude = true
|
|
|
+ }
|
|
|
+ })
|
|
|
+
|
|
|
+ // Extract the file.
|
|
|
+ if(!shouldExclude){
|
|
|
+ mkpath.sync(path.join(nativePath, fileName, '..'))
|
|
|
+ fs.writeFile(path.join(nativePath, fileName), zipEntries[i].getData())
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ libs.push(to)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ 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.
|
|
|
+ *
|
|
|
+ * @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.
|
|
|
+ */
|
|
|
+ _resolveServerLibraries(mods){
|
|
|
+ const mdles = this.server.modules
|
|
|
+ let libs = []
|
|
|
+
|
|
|
+ // Locate Forge Libraries
|
|
|
+ for(let i=0; i<mdles.length; i++){
|
|
|
+ if(mdles[i].type != null && mdles[i].type === 'forge-hosted'){
|
|
|
+ let forge = mdles[i]
|
|
|
+ libs.push(path.join(this.libPath, forge.artifact.path == null ? ag._resolvePath(forge.id, forge.artifact.extension) : forge.artifact.path))
|
|
|
+ const res = this._resolveModuleLibraries(forge)
|
|
|
+ if(res.length > 0){
|
|
|
+ libs = libs.concat(res)
|
|
|
+ }
|
|
|
+ break
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ //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.
|
|
|
+ *
|
|
|
+ * @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.
|
|
|
+ */
|
|
|
+ _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'){
|
|
|
+ libs.push(path.join(this.libPath, sm.artifact.path == null ? ag._resolvePath(sm.id, sm.artifact.extension) : sm.artifact.path))
|
|
|
+ }
|
|
|
+ // 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
|