diff options
author | Grant Limberg <glimberg@gmail.com> | 2015-04-30 21:50:05 -0700 |
---|---|---|
committer | Grant Limberg <glimberg@gmail.com> | 2015-04-30 21:50:05 -0700 |
commit | 75d71370258b102144a1f5941b64f35ddf815a8c (patch) | |
tree | 84291d5b0e39b63f6e056cdd4f5579933a2d1827 /java/src | |
parent | 52df59c55299419feb16a5daeed01389f4fafc73 (diff) | |
download | infinitytier-75d71370258b102144a1f5941b64f35ddf815a8c.tar.gz infinitytier-75d71370258b102144a1f5941b64f35ddf815a8c.zip |
added a class I found to load JNI code from within JAR files on non-android platforms.
That doesn't come for free, unlike android.
Diffstat (limited to 'java/src')
-rw-r--r-- | java/src/com/zerotierone/sdk/NativeUtils.java | 93 | ||||
-rw-r--r-- | java/src/com/zerotierone/sdk/Node.java | 20 |
2 files changed, 112 insertions, 1 deletions
diff --git a/java/src/com/zerotierone/sdk/NativeUtils.java b/java/src/com/zerotierone/sdk/NativeUtils.java new file mode 100644 index 00000000..d761c68c --- /dev/null +++ b/java/src/com/zerotierone/sdk/NativeUtils.java @@ -0,0 +1,93 @@ +package com.zerotierone.sdk; + +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; + +/** + * Simple library class for working with JNI (Java Native Interface) + * + * @see http://adamheinrich.com/2012/how-to-load-native-jni-library-from-jar + * + * @author Adam Heirnich <adam@adamh.cz>, http://www.adamh.cz + */ +public class NativeUtils { + + /** + * Private constructor - this class will never be instanced + */ + private NativeUtils() { + } + + /** + * Loads library from current JAR archive + * + * The file from JAR is copied into system temporary directory and then loaded. The temporary file is deleted after exiting. + * Method uses String as filename because the pathname is "abstract", not system-dependent. + * + * @param filename The filename inside JAR as absolute path (beginning with '/'), e.g. /package/File.ext + * @throws IOException If temporary file creation or read/write operation fails + * @throws IllegalArgumentException If source file (param path) does not exist + * @throws IllegalArgumentException If the path is not absolute or if the filename is shorter than three characters (restriction of {@see File#createTempFile(java.lang.String, java.lang.String)}). + */ + public static void loadLibraryFromJar(String path) throws IOException { + + if (!path.startsWith("/")) { + throw new IllegalArgumentException("The path has to be absolute (start with '/')."); + } + + // Obtain filename from path + String[] parts = path.split("/"); + String filename = (parts.length > 1) ? parts[parts.length - 1] : null; + + // Split filename to prexif and suffix (extension) + String prefix = ""; + String suffix = null; + if (filename != null) { + parts = filename.split("\\.", 2); + prefix = parts[0]; + suffix = (parts.length > 1) ? "."+parts[parts.length - 1] : null; // Thanks, davs! :-) + } + + // Check if the filename is okay + if (filename == null || prefix.length() < 3) { + throw new IllegalArgumentException("The filename has to be at least 3 characters long."); + } + + // Prepare temporary file + File temp = File.createTempFile(prefix, suffix); + temp.deleteOnExit(); + + if (!temp.exists()) { + throw new FileNotFoundException("File " + temp.getAbsolutePath() + " does not exist."); + } + + // Prepare buffer for data copying + byte[] buffer = new byte[1024]; + int readBytes; + + // Open and check input stream + InputStream is = NativeUtils.class.getResourceAsStream(path); + if (is == null) { + throw new FileNotFoundException("File " + path + " was not found inside JAR."); + } + + // Open output stream and copy data between source file in JAR and the temporary file + OutputStream os = new FileOutputStream(temp); + try { + while ((readBytes = is.read(buffer)) != -1) { + os.write(buffer, 0, readBytes); + } + } finally { + // If read/write fails, close streams safely before throwing an exception + os.close(); + is.close(); + } + + // Finally, load the library + System.load(temp.getAbsolutePath()); + } +}
\ No newline at end of file diff --git a/java/src/com/zerotierone/sdk/Node.java b/java/src/com/zerotierone/sdk/Node.java index 911e83d3..6b4a8729 100644 --- a/java/src/com/zerotierone/sdk/Node.java +++ b/java/src/com/zerotierone/sdk/Node.java @@ -31,13 +31,31 @@ import java.nio.ByteBuffer; import java.lang.Long; import java.net.InetSocketAddress; import java.util.ArrayList; +import java.io.IOException; /** * A ZeroTier One node */ public class Node { static { - System.loadLibrary("ZeroTierOneJNI"); + try { + System.loadLibrary("ZeroTierOneJNI"); + } catch (UnsatisfiedLinkError e) { + try { + if(System.getProperty("os.name").startsWith("Windows")) { + System.out.println("Arch: " + System.getProperty("sun.arch.data.model")); + if(System.getProperty("sun.arch.data.model").equals("64")) { + NativeUtils.loadLibraryFromJar("/lib/ZeroTierOneJNI_win64.dll"); + } else { + NativeUtils.loadLibraryFromJar("/lib/ZeroTierOneJNI_win32.dll"); + } + } else { + // TODO: Mac + } + } catch (IOException ioe) { + ioe.printStackTrace(); + } + } } private static final String TAG = "NODE"; |