summaryrefslogtreecommitdiff
path: root/java/src
diff options
context:
space:
mode:
authorGrant Limberg <glimberg@gmail.com>2015-04-30 21:50:05 -0700
committerGrant Limberg <glimberg@gmail.com>2015-04-30 21:50:05 -0700
commit75d71370258b102144a1f5941b64f35ddf815a8c (patch)
tree84291d5b0e39b63f6e056cdd4f5579933a2d1827 /java/src
parent52df59c55299419feb16a5daeed01389f4fafc73 (diff)
downloadinfinitytier-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.java93
-rw-r--r--java/src/com/zerotierone/sdk/Node.java20
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";