2014-01-09 01:22:34 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2012-2014 MultiMC Contributors
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package org.multimc;
|
|
|
|
|
2014-01-22 02:20:09 +01:00
|
|
|
import java.io.*;
|
2014-11-01 10:39:32 +01:00
|
|
|
import java.io.File;
|
2014-01-09 01:22:34 +01:00
|
|
|
import java.lang.reflect.Field;
|
|
|
|
import java.lang.reflect.Method;
|
|
|
|
import java.lang.reflect.Modifier;
|
|
|
|
import java.net.URL;
|
|
|
|
import java.net.URLClassLoader;
|
2014-11-01 10:39:32 +01:00
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.nio.file.Path;
|
|
|
|
import java.util.*;
|
2014-01-09 01:22:34 +01:00
|
|
|
import java.util.Arrays;
|
2014-01-22 02:20:09 +01:00
|
|
|
import java.util.Enumeration;
|
2014-01-09 01:22:34 +01:00
|
|
|
import java.util.List;
|
2014-01-22 02:20:09 +01:00
|
|
|
import java.util.zip.ZipEntry;
|
|
|
|
import java.util.zip.ZipFile;
|
2014-01-09 01:22:34 +01:00
|
|
|
|
|
|
|
public class Utils
|
|
|
|
{
|
2014-01-14 01:13:35 +01:00
|
|
|
/**
|
|
|
|
* Combine two parts of a path.
|
2014-01-22 02:20:09 +01:00
|
|
|
*
|
2014-01-14 01:13:35 +01:00
|
|
|
* @param path1
|
|
|
|
* @param path2
|
|
|
|
* @return the paths, combined
|
|
|
|
*/
|
2014-01-22 02:20:09 +01:00
|
|
|
public static String combine(String path1, String path2)
|
2014-01-14 01:13:35 +01:00
|
|
|
{
|
|
|
|
File file1 = new File(path1);
|
|
|
|
File file2 = new File(file1, path2);
|
|
|
|
return file2.getPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Join a list of strings into a string using a separator!
|
2014-01-22 02:20:09 +01:00
|
|
|
*
|
|
|
|
* @param strings the string list to join
|
2014-01-14 01:13:35 +01:00
|
|
|
* @param separator the glue
|
|
|
|
* @return the result.
|
|
|
|
*/
|
2014-01-22 02:20:09 +01:00
|
|
|
public static String join(List<String> strings, String separator)
|
2014-01-14 01:13:35 +01:00
|
|
|
{
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
String sep = "";
|
2014-01-22 02:20:09 +01:00
|
|
|
for (String s : strings)
|
2014-01-14 01:13:35 +01:00
|
|
|
{
|
|
|
|
sb.append(sep).append(s);
|
|
|
|
sep = separator;
|
|
|
|
}
|
|
|
|
return sb.toString();
|
|
|
|
}
|
|
|
|
|
2014-01-09 01:22:34 +01:00
|
|
|
/**
|
|
|
|
* Adds the specified library to the classpath
|
|
|
|
*
|
|
|
|
* @param s the path to add
|
|
|
|
* @throws Exception
|
|
|
|
*/
|
|
|
|
public static void addToClassPath(String s) throws Exception
|
|
|
|
{
|
|
|
|
File f = new File(s);
|
|
|
|
URL u = f.toURI().toURL();
|
|
|
|
URLClassLoader urlClassLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
|
|
|
|
Class urlClass = URLClassLoader.class;
|
|
|
|
Method method = urlClass.getDeclaredMethod("addURL", new Class[]{URL.class});
|
|
|
|
method.setAccessible(true);
|
|
|
|
method.invoke(urlClassLoader, new Object[]{u});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds many libraries to the classpath
|
|
|
|
*
|
|
|
|
* @param jars the paths to add
|
|
|
|
*/
|
|
|
|
public static boolean addToClassPath(List<String> jars)
|
|
|
|
{
|
|
|
|
boolean pure = true;
|
|
|
|
// initialize the class path
|
|
|
|
for (String jar : jars)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
Utils.addToClassPath(jar);
|
|
|
|
} catch (Exception e)
|
|
|
|
{
|
|
|
|
System.err.println("Unable to load: " + jar);
|
|
|
|
e.printStackTrace(System.err);
|
|
|
|
pure = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return pure;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds the specified path to the java library path
|
|
|
|
*
|
|
|
|
* @param pathToAdd the path to add
|
|
|
|
* @throws Exception
|
|
|
|
*/
|
2014-01-22 02:20:09 +01:00
|
|
|
@Deprecated
|
|
|
|
public static void addLibraryPath(String pathToAdd) throws Exception
|
2014-01-09 01:22:34 +01:00
|
|
|
{
|
|
|
|
final Field usrPathsField = ClassLoader.class.getDeclaredField("usr_paths");
|
|
|
|
usrPathsField.setAccessible(true);
|
|
|
|
|
|
|
|
//get array of paths
|
|
|
|
final String[] paths = (String[]) usrPathsField.get(null);
|
|
|
|
|
|
|
|
//check if the path to add is already present
|
|
|
|
for (String path : paths)
|
|
|
|
{
|
|
|
|
if (path.equals(pathToAdd))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//add the new path
|
|
|
|
final String[] newPaths = Arrays.copyOf(paths, paths.length + 1);
|
|
|
|
newPaths[newPaths.length - 1] = pathToAdd;
|
|
|
|
usrPathsField.set(null, newPaths);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Finds a field that looks like a Minecraft base folder in a supplied class
|
|
|
|
*
|
|
|
|
* @param mc the class to scan
|
|
|
|
*/
|
|
|
|
public static Field getMCPathField(Class<?> mc)
|
|
|
|
{
|
|
|
|
Field[] fields = mc.getDeclaredFields();
|
|
|
|
|
|
|
|
for (Field f : fields)
|
|
|
|
{
|
|
|
|
if (f.getType() != File.class)
|
|
|
|
{
|
|
|
|
// Has to be File
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (f.getModifiers() != (Modifier.PRIVATE + Modifier.STATIC))
|
|
|
|
{
|
|
|
|
// And Private Static.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
2014-01-17 22:55:10 +01:00
|
|
|
|
2014-01-22 02:20:09 +01:00
|
|
|
/**
|
|
|
|
* Log to the MultiMC console
|
|
|
|
*
|
|
|
|
* @param message A String containing the message
|
|
|
|
* @param level A String containing the level name. See MinecraftProcess::getLevel()
|
|
|
|
*/
|
|
|
|
public static void log(String message, String level)
|
|
|
|
{
|
|
|
|
// Kinda dirty
|
|
|
|
String tag = "!![" + level + "]!";
|
|
|
|
System.out.println(tag + message.replace("\n", "\n" + tag));
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void log(String message)
|
|
|
|
{
|
|
|
|
log(message, "MultiMC");
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void log()
|
|
|
|
{
|
|
|
|
System.out.println();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pushes bytes from in to out. Closes both streams no matter what.
|
|
|
|
* @param in the input stream
|
|
|
|
* @param out the output stream
|
|
|
|
* @throws IOException
|
|
|
|
*/
|
|
|
|
private static void copyStream(InputStream in, OutputStream out) throws IOException
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
byte[] buffer = new byte[4096];
|
|
|
|
int len;
|
|
|
|
|
|
|
|
while((len = in.read(buffer)) >= 0)
|
|
|
|
out.write(buffer, 0, len);
|
|
|
|
} finally
|
|
|
|
{
|
|
|
|
in.close();
|
|
|
|
out.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-11-01 10:39:32 +01:00
|
|
|
* Replace a 'target' string 'suffix' with 'replacement'
|
|
|
|
*/
|
|
|
|
public static String replaceSuffix (String target, String suffix, String replacement)
|
|
|
|
{
|
|
|
|
if (!target.endsWith(suffix))
|
|
|
|
{
|
|
|
|
return target;
|
|
|
|
}
|
|
|
|
String prefix = target.substring(0, target.length() - suffix.length());
|
|
|
|
return prefix + replacement;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unzip zip file with natives 'source' into the folder 'targetFolder'
|
|
|
|
*
|
|
|
|
* Contains a hack for OSX. Yay.
|
2014-01-22 02:20:09 +01:00
|
|
|
* @param source
|
|
|
|
* @param targetFolder
|
|
|
|
* @throws IOException
|
|
|
|
*/
|
2014-11-01 10:39:32 +01:00
|
|
|
public static void unzipNatives(File source, File targetFolder) throws IOException
|
2014-01-22 02:20:09 +01:00
|
|
|
{
|
|
|
|
ZipFile zip = new ZipFile(source);
|
2014-11-01 10:39:32 +01:00
|
|
|
Set <String> toProcess = new HashSet<String>();
|
2014-01-22 02:20:09 +01:00
|
|
|
try
|
|
|
|
{
|
|
|
|
Enumeration entries = zip.entries();
|
|
|
|
|
|
|
|
while (entries.hasMoreElements())
|
|
|
|
{
|
|
|
|
ZipEntry entry = (ZipEntry) entries.nextElement();
|
|
|
|
|
2014-11-01 10:39:32 +01:00
|
|
|
String entryName = entry.getName();
|
|
|
|
File targetFile = new File(targetFolder, entryName);
|
2014-01-22 02:20:09 +01:00
|
|
|
if (targetFile.getParentFile() != null)
|
|
|
|
{
|
|
|
|
targetFile.getParentFile().mkdirs();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entry.isDirectory())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
copyStream(zip.getInputStream(entry), new BufferedOutputStream(new FileOutputStream(targetFile)));
|
2014-11-01 10:39:32 +01:00
|
|
|
toProcess.add(entryName);
|
2014-01-22 02:20:09 +01:00
|
|
|
}
|
|
|
|
} finally
|
|
|
|
{
|
|
|
|
zip.close();
|
|
|
|
}
|
2014-11-01 10:39:32 +01:00
|
|
|
for (String entryName : toProcess)
|
|
|
|
{
|
|
|
|
// check if we need a symlink
|
|
|
|
String suffixFrom = null;
|
|
|
|
String suffixTo = null;
|
|
|
|
if(entryName.endsWith(".dylib"))
|
|
|
|
{
|
|
|
|
suffixFrom = ".dylib";
|
|
|
|
suffixTo = ".jnilib";
|
|
|
|
}
|
|
|
|
else if(entryName.endsWith(".jnilib"))
|
|
|
|
{
|
|
|
|
suffixFrom = ".jnilib";
|
|
|
|
suffixTo = ".dylib";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
String linkName = replaceSuffix(entryName, suffixFrom, suffixTo);
|
|
|
|
File targetFile = new File(targetFolder, entryName);
|
|
|
|
File symlinkFile = new File(targetFolder, linkName);
|
|
|
|
|
|
|
|
// if the link file exists already for whatever reason, do not create symlinks
|
|
|
|
if(symlinkFile.exists())
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// create a symlink. This means we always have .jnilib and .dylib variants of the same libs.
|
|
|
|
Path linkLink = symlinkFile.toPath();
|
|
|
|
Path linkTarget = targetFile.toPath();
|
|
|
|
Files.createSymbolicLink(linkLink, linkTarget);
|
|
|
|
}
|
2014-01-22 02:20:09 +01:00
|
|
|
}
|
2014-01-09 01:22:34 +01:00
|
|
|
}
|
2014-01-22 02:20:09 +01:00
|
|
|
|