Compare commits
1 commit
Author | SHA1 | Date | |
---|---|---|---|
|
e01fbc6f39 |
29 changed files with 2506 additions and 1 deletions
5
android/Makefile
Normal file
5
android/Makefile
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
build:
|
||||||
|
python setup.py apk
|
||||||
|
adb install -r lbry-0.1-service-debug.apk
|
||||||
|
|
||||||
|
.PHONY: build
|
49
android/bootstrap/__init__.py
Normal file
49
android/bootstrap/__init__.py
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
import inspect
|
||||||
|
from os.path import dirname, join
|
||||||
|
|
||||||
|
import sh
|
||||||
|
from pythonforandroid.util import ensure_dir
|
||||||
|
from pythonforandroid.toolchain import (
|
||||||
|
Bootstrap, current_directory, info, info_main, shprint
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class LBRYServiceBootstrap(Bootstrap):
|
||||||
|
name = 'lbry-service'
|
||||||
|
recipe_depends = ['genericndkbuild', 'python3']
|
||||||
|
bootstrap_dir = dirname(__file__)
|
||||||
|
|
||||||
|
def get_common_dir(self):
|
||||||
|
return join(dirname(inspect.getfile(Bootstrap)), 'bootstraps', 'common')
|
||||||
|
|
||||||
|
def run_distribute(self):
|
||||||
|
info_main('# Creating Android project from build and {} bootstrap'.format(
|
||||||
|
self.name))
|
||||||
|
|
||||||
|
info('This currently just copies the build stuff straight from the build dir.')
|
||||||
|
shprint(sh.rm, '-rf', self.dist_dir)
|
||||||
|
shprint(sh.cp, '-r', self.build_dir, self.dist_dir)
|
||||||
|
with current_directory(self.dist_dir):
|
||||||
|
with open('local.properties', 'w') as fileh:
|
||||||
|
fileh.write('sdk.dir={}'.format(self.ctx.sdk_dir))
|
||||||
|
|
||||||
|
arch = self.ctx.archs[0]
|
||||||
|
if len(self.ctx.archs) > 1:
|
||||||
|
raise ValueError('built for more than one arch, but bootstrap cannot handle that yet')
|
||||||
|
info('Bootstrap running with arch {}'.format(arch))
|
||||||
|
|
||||||
|
with current_directory(self.dist_dir):
|
||||||
|
info('Copying python distribution')
|
||||||
|
|
||||||
|
self.distribute_libs(arch, [self.ctx.get_libs_dir(arch.arch)])
|
||||||
|
self.distribute_aars(arch)
|
||||||
|
self.distribute_javaclasses(self.ctx.javaclass_dir)
|
||||||
|
|
||||||
|
python_bundle_dir = join('_python_bundle', '_python_bundle')
|
||||||
|
ensure_dir(python_bundle_dir)
|
||||||
|
site_packages_dir = self.ctx.python_recipe.create_python_bundle(
|
||||||
|
join(self.dist_dir, python_bundle_dir), arch)
|
||||||
|
|
||||||
|
self.strip_libraries(arch)
|
||||||
|
self.fry_eggs(site_packages_dir)
|
||||||
|
super().run_distribute()
|
89
android/bootstrap/build/blacklist.txt
Normal file
89
android/bootstrap/build/blacklist.txt
Normal file
|
@ -0,0 +1,89 @@
|
||||||
|
# prevent user to include invalid extensions
|
||||||
|
*.apk
|
||||||
|
*.pxd
|
||||||
|
|
||||||
|
# eggs
|
||||||
|
*.egg-info
|
||||||
|
|
||||||
|
# unit test
|
||||||
|
unittest/*
|
||||||
|
|
||||||
|
# python config
|
||||||
|
config/makesetup
|
||||||
|
|
||||||
|
# unused kivy files (platform specific)
|
||||||
|
kivy/input/providers/wm_*
|
||||||
|
kivy/input/providers/mactouch*
|
||||||
|
kivy/input/providers/probesysfs*
|
||||||
|
kivy/input/providers/mtdev*
|
||||||
|
kivy/input/providers/hidinput*
|
||||||
|
kivy/core/camera/camera_videocapture*
|
||||||
|
kivy/core/spelling/*osx*
|
||||||
|
kivy/core/video/video_pyglet*
|
||||||
|
kivy/tools
|
||||||
|
kivy/tests/*
|
||||||
|
kivy/*/*.h
|
||||||
|
kivy/*/*.pxi
|
||||||
|
|
||||||
|
# unused encodings
|
||||||
|
lib-dynload/*codec*
|
||||||
|
encodings/cp*.pyo
|
||||||
|
encodings/tis*
|
||||||
|
encodings/shift*
|
||||||
|
encodings/bz2*
|
||||||
|
encodings/iso*
|
||||||
|
encodings/undefined*
|
||||||
|
encodings/johab*
|
||||||
|
encodings/p*
|
||||||
|
encodings/m*
|
||||||
|
encodings/euc*
|
||||||
|
encodings/k*
|
||||||
|
encodings/unicode_internal*
|
||||||
|
encodings/quo*
|
||||||
|
encodings/gb*
|
||||||
|
encodings/big5*
|
||||||
|
encodings/hp*
|
||||||
|
encodings/hz*
|
||||||
|
|
||||||
|
# unused python modules
|
||||||
|
bsddb/*
|
||||||
|
wsgiref/*
|
||||||
|
hotshot/*
|
||||||
|
pydoc_data/*
|
||||||
|
tty.pyo
|
||||||
|
anydbm.pyo
|
||||||
|
nturl2path.pyo
|
||||||
|
LICENCE.txt
|
||||||
|
macurl2path.pyo
|
||||||
|
dummy_threading.pyo
|
||||||
|
audiodev.pyo
|
||||||
|
antigravity.pyo
|
||||||
|
dumbdbm.pyo
|
||||||
|
sndhdr.pyo
|
||||||
|
__phello__.foo.pyo
|
||||||
|
sunaudio.pyo
|
||||||
|
os2emxpath.pyo
|
||||||
|
multiprocessing/dummy*
|
||||||
|
|
||||||
|
# unused binaries python modules
|
||||||
|
lib-dynload/termios.so
|
||||||
|
lib-dynload/_lsprof.so
|
||||||
|
lib-dynload/*audioop.so
|
||||||
|
lib-dynload/_hotshot.so
|
||||||
|
lib-dynload/_heapq.so
|
||||||
|
lib-dynload/_json.so
|
||||||
|
lib-dynload/grp.so
|
||||||
|
lib-dynload/resource.so
|
||||||
|
lib-dynload/pyexpat.so
|
||||||
|
lib-dynload/_ctypes_test.so
|
||||||
|
lib-dynload/_testcapi.so
|
||||||
|
|
||||||
|
# odd files
|
||||||
|
plat-linux3/regen
|
||||||
|
|
||||||
|
#>sqlite3
|
||||||
|
# conditionnal include depending if some recipes are included or not.
|
||||||
|
sqlite3/*
|
||||||
|
lib-dynload/_sqlite3.so
|
||||||
|
#<sqlite3
|
||||||
|
|
1
android/bootstrap/build/jni/Android.mk
Normal file
1
android/bootstrap/build/jni/Android.mk
Normal file
|
@ -0,0 +1 @@
|
||||||
|
include $(call all-subdir-makefiles)
|
7
android/bootstrap/build/jni/Application.mk
Normal file
7
android/bootstrap/build/jni/Application.mk
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
|
||||||
|
# Uncomment this if you're using STL in your project
|
||||||
|
# See CPLUSPLUS-SUPPORT.html in the NDK documentation for more information
|
||||||
|
# APP_STL := stlport_static
|
||||||
|
|
||||||
|
# APP_ABI := armeabi armeabi-v7a x86
|
||||||
|
APP_ABI := $(ARCH)
|
22
android/bootstrap/build/jni/application/src/Android.mk
Normal file
22
android/bootstrap/build/jni/application/src/Android.mk
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
LOCAL_PATH := $(call my-dir)
|
||||||
|
|
||||||
|
include $(CLEAR_VARS)
|
||||||
|
|
||||||
|
LOCAL_MODULE := main
|
||||||
|
|
||||||
|
# Add your application source files here...
|
||||||
|
LOCAL_SRC_FILES := start.c pyjniusjni.c
|
||||||
|
|
||||||
|
LOCAL_CFLAGS += -I$(LOCAL_PATH)/../../../../../other_builds/$(MK_PYTHON_INCLUDE_ROOT) $(EXTRA_CFLAGS)
|
||||||
|
|
||||||
|
LOCAL_SHARED_LIBRARIES := python_shared
|
||||||
|
|
||||||
|
LOCAL_LDLIBS := -llog $(EXTRA_LDLIBS)
|
||||||
|
|
||||||
|
LOCAL_LDFLAGS += -L$(LOCAL_PATH)/../../../../../other_builds/$(MK_PYTHON_LINK_ROOT) $(APPLICATION_ADDITIONAL_LDFLAGS)
|
||||||
|
|
||||||
|
include $(BUILD_SHARED_LIBRARY)
|
||||||
|
|
||||||
|
ifdef CRYSTAX_PYTHON_VERSION
|
||||||
|
$(call import-module,python/$(CRYSTAX_PYTHON_VERSION))
|
||||||
|
endif
|
|
@ -0,0 +1,10 @@
|
||||||
|
LOCAL_PATH := $(call my-dir)
|
||||||
|
|
||||||
|
include $(CLEAR_VARS)
|
||||||
|
|
||||||
|
LOCAL_MODULE := main
|
||||||
|
|
||||||
|
LOCAL_SRC_FILES := YourSourceHere.c
|
||||||
|
|
||||||
|
include $(BUILD_SHARED_LIBRARY)
|
||||||
|
$(call import-module,SDL)LOCAL_PATH := $(call my-dir)
|
|
@ -0,0 +1,6 @@
|
||||||
|
|
||||||
|
#define BOOTSTRAP_NAME_SERVICEONLY
|
||||||
|
#define BOOTSTRAP_USES_NO_SDL_HEADERS
|
||||||
|
|
||||||
|
const char bootstrap_name[] = "service_only";
|
||||||
|
|
103
android/bootstrap/build/jni/application/src/pyjniusjni.c
Normal file
103
android/bootstrap/build/jni/application/src/pyjniusjni.c
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
|
||||||
|
#include <pthread.h>
|
||||||
|
#include <jni.h>
|
||||||
|
|
||||||
|
#define LOGI(...) do {} while (0)
|
||||||
|
#define LOGE(...) do {} while (0)
|
||||||
|
|
||||||
|
#include "android/log.h"
|
||||||
|
|
||||||
|
/* These JNI management functions are taken from SDL2, but modified to refer to pyjnius */
|
||||||
|
|
||||||
|
/* #define LOG(n, x) __android_log_write(ANDROID_LOG_INFO, (n), (x)) */
|
||||||
|
/* #define LOGP(x) LOG("python", (x)) */
|
||||||
|
#define LOG_TAG "Python_android"
|
||||||
|
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__)
|
||||||
|
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)
|
||||||
|
|
||||||
|
|
||||||
|
/* Function headers */
|
||||||
|
JNIEnv* Android_JNI_GetEnv(void);
|
||||||
|
static void Android_JNI_ThreadDestroyed(void*);
|
||||||
|
|
||||||
|
static pthread_key_t mThreadKey;
|
||||||
|
static JavaVM* mJavaVM;
|
||||||
|
|
||||||
|
int Android_JNI_SetupThread(void)
|
||||||
|
{
|
||||||
|
Android_JNI_GetEnv();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Library init */
|
||||||
|
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved)
|
||||||
|
{
|
||||||
|
JNIEnv *env;
|
||||||
|
mJavaVM = vm;
|
||||||
|
LOGI("JNI_OnLoad called");
|
||||||
|
if ((*mJavaVM)->GetEnv(mJavaVM, (void**) &env, JNI_VERSION_1_4) != JNI_OK) {
|
||||||
|
LOGE("Failed to get the environment using GetEnv()");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
* Create mThreadKey so we can keep track of the JNIEnv assigned to each thread
|
||||||
|
* Refer to http://developer.android.com/guide/practices/design/jni.html for the rationale behind this
|
||||||
|
*/
|
||||||
|
if (pthread_key_create(&mThreadKey, Android_JNI_ThreadDestroyed) != 0) {
|
||||||
|
|
||||||
|
__android_log_print(ANDROID_LOG_ERROR, "pyjniusjni", "Error initializing pthread key");
|
||||||
|
}
|
||||||
|
Android_JNI_SetupThread();
|
||||||
|
|
||||||
|
return JNI_VERSION_1_4;
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEnv* Android_JNI_GetEnv(void)
|
||||||
|
{
|
||||||
|
/* From http://developer.android.com/guide/practices/jni.html
|
||||||
|
* All threads are Linux threads, scheduled by the kernel.
|
||||||
|
* They're usually started from managed code (using Thread.start), but they can also be created elsewhere and then
|
||||||
|
* attached to the JavaVM. For example, a thread started with pthread_create can be attached with the
|
||||||
|
* JNI AttachCurrentThread or AttachCurrentThreadAsDaemon functions. Until a thread is attached, it has no JNIEnv,
|
||||||
|
* and cannot make JNI calls.
|
||||||
|
* Attaching a natively-created thread causes a java.lang.Thread object to be constructed and added to the "main"
|
||||||
|
* ThreadGroup, making it visible to the debugger. Calling AttachCurrentThread on an already-attached thread
|
||||||
|
* is a no-op.
|
||||||
|
* Note: You can call this function any number of times for the same thread, there's no harm in it
|
||||||
|
*/
|
||||||
|
|
||||||
|
JNIEnv *env;
|
||||||
|
int status = (*mJavaVM)->AttachCurrentThread(mJavaVM, &env, NULL);
|
||||||
|
if(status < 0) {
|
||||||
|
LOGE("failed to attach current thread");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* From http://developer.android.com/guide/practices/jni.html
|
||||||
|
* Threads attached through JNI must call DetachCurrentThread before they exit. If coding this directly is awkward,
|
||||||
|
* in Android 2.0 (Eclair) and higher you can use pthread_key_create to define a destructor function that will be
|
||||||
|
* called before the thread exits, and call DetachCurrentThread from there. (Use that key with pthread_setspecific
|
||||||
|
* to store the JNIEnv in thread-local-storage; that way it'll be passed into your destructor as the argument.)
|
||||||
|
* Note: The destructor is not called unless the stored value is != NULL
|
||||||
|
* Note: You can call this function any number of times for the same thread, there's no harm in it
|
||||||
|
* (except for some lost CPU cycles)
|
||||||
|
*/
|
||||||
|
pthread_setspecific(mThreadKey, (void*) env);
|
||||||
|
|
||||||
|
return env;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void Android_JNI_ThreadDestroyed(void* value)
|
||||||
|
{
|
||||||
|
/* The thread is being destroyed, detach it from the Java VM and set the mThreadKey value to NULL as required */
|
||||||
|
JNIEnv *env = (JNIEnv*) value;
|
||||||
|
if (env != NULL) {
|
||||||
|
(*mJavaVM)->DetachCurrentThread(mJavaVM);
|
||||||
|
pthread_setspecific(mThreadKey, NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void *WebView_AndroidGetJNIEnv()
|
||||||
|
{
|
||||||
|
return Android_JNI_GetEnv();
|
||||||
|
}
|
0
android/bootstrap/build/src/main/assets/.gitkeep
Normal file
0
android/bootstrap/build/src/main/assets/.gitkeep
Normal file
141
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/Octal.java
Executable file
141
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/Octal.java
Executable file
|
@ -0,0 +1,141 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2012 Kamran Zafar
|
||||||
|
*
|
||||||
|
* 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.kamranzafar.jtar;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Kamran Zafar
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class Octal {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse an octal string from a header buffer. This is used for the file
|
||||||
|
* permission mode value.
|
||||||
|
*
|
||||||
|
* @param header
|
||||||
|
* The header buffer from which to parse.
|
||||||
|
* @param offset
|
||||||
|
* The offset into the buffer from which to parse.
|
||||||
|
* @param length
|
||||||
|
* The number of header bytes to parse.
|
||||||
|
*
|
||||||
|
* @return The long value of the octal string.
|
||||||
|
*/
|
||||||
|
public static long parseOctal(byte[] header, int offset, int length) {
|
||||||
|
long result = 0;
|
||||||
|
boolean stillPadding = true;
|
||||||
|
|
||||||
|
int end = offset + length;
|
||||||
|
for (int i = offset; i < end; ++i) {
|
||||||
|
if (header[i] == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (header[i] == (byte) ' ' || header[i] == '0') {
|
||||||
|
if (stillPadding)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (header[i] == (byte) ' ')
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
stillPadding = false;
|
||||||
|
|
||||||
|
result = ( result << 3 ) + ( header[i] - '0' );
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse an octal integer from a header buffer.
|
||||||
|
*
|
||||||
|
* @param value
|
||||||
|
* @param buf
|
||||||
|
* The header buffer from which to parse.
|
||||||
|
* @param offset
|
||||||
|
* The offset into the buffer from which to parse.
|
||||||
|
* @param length
|
||||||
|
* The number of header bytes to parse.
|
||||||
|
*
|
||||||
|
* @return The integer value of the octal bytes.
|
||||||
|
*/
|
||||||
|
public static int getOctalBytes(long value, byte[] buf, int offset, int length) {
|
||||||
|
int idx = length - 1;
|
||||||
|
|
||||||
|
buf[offset + idx] = 0;
|
||||||
|
--idx;
|
||||||
|
buf[offset + idx] = (byte) ' ';
|
||||||
|
--idx;
|
||||||
|
|
||||||
|
if (value == 0) {
|
||||||
|
buf[offset + idx] = (byte) '0';
|
||||||
|
--idx;
|
||||||
|
} else {
|
||||||
|
for (long val = value; idx >= 0 && val > 0; --idx) {
|
||||||
|
buf[offset + idx] = (byte) ( (byte) '0' + (byte) ( val & 7 ) );
|
||||||
|
val = val >> 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (; idx >= 0; --idx) {
|
||||||
|
buf[offset + idx] = (byte) ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
return offset + length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse the checksum octal integer from a header buffer.
|
||||||
|
*
|
||||||
|
* @param value
|
||||||
|
* @param buf
|
||||||
|
* The header buffer from which to parse.
|
||||||
|
* @param offset
|
||||||
|
* The offset into the buffer from which to parse.
|
||||||
|
* @param length
|
||||||
|
* The number of header bytes to parse.
|
||||||
|
* @return The integer value of the entry's checksum.
|
||||||
|
*/
|
||||||
|
public static int getCheckSumOctalBytes(long value, byte[] buf, int offset, int length) {
|
||||||
|
getOctalBytes( value, buf, offset, length );
|
||||||
|
buf[offset + length - 1] = (byte) ' ';
|
||||||
|
buf[offset + length - 2] = 0;
|
||||||
|
return offset + length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse an octal long integer from a header buffer.
|
||||||
|
*
|
||||||
|
* @param value
|
||||||
|
* @param buf
|
||||||
|
* The header buffer from which to parse.
|
||||||
|
* @param offset
|
||||||
|
* The offset into the buffer from which to parse.
|
||||||
|
* @param length
|
||||||
|
* The number of header bytes to parse.
|
||||||
|
*
|
||||||
|
* @return The long value of the octal bytes.
|
||||||
|
*/
|
||||||
|
public static int getLongOctalBytes(long value, byte[] buf, int offset, int length) {
|
||||||
|
byte[] temp = new byte[length + 1];
|
||||||
|
getOctalBytes( value, temp, 0, length + 1 );
|
||||||
|
System.arraycopy( temp, 0, buf, offset, length );
|
||||||
|
return offset + length;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
28
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarConstants.java
Executable file
28
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarConstants.java
Executable file
|
@ -0,0 +1,28 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2012 Kamran Zafar
|
||||||
|
*
|
||||||
|
* 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.kamranzafar.jtar;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Kamran Zafar
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class TarConstants {
|
||||||
|
public static final int EOF_BLOCK = 1024;
|
||||||
|
public static final int DATA_BLOCK = 512;
|
||||||
|
public static final int HEADER_BLOCK = 512;
|
||||||
|
}
|
284
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarEntry.java
Executable file
284
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarEntry.java
Executable file
|
@ -0,0 +1,284 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2012 Kamran Zafar
|
||||||
|
*
|
||||||
|
* 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.kamranzafar.jtar;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Kamran Zafar
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class TarEntry {
|
||||||
|
protected File file;
|
||||||
|
protected TarHeader header;
|
||||||
|
|
||||||
|
private TarEntry() {
|
||||||
|
this.file = null;
|
||||||
|
header = new TarHeader();
|
||||||
|
}
|
||||||
|
|
||||||
|
public TarEntry(File file, String entryName) {
|
||||||
|
this();
|
||||||
|
this.file = file;
|
||||||
|
this.extractTarHeader(entryName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TarEntry(byte[] headerBuf) {
|
||||||
|
this();
|
||||||
|
this.parseTarHeader(headerBuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor to create an entry from an existing TarHeader object.
|
||||||
|
*
|
||||||
|
* This method is useful to add new entries programmatically (e.g. for
|
||||||
|
* adding files or directories that do not exist in the file system).
|
||||||
|
*
|
||||||
|
* @param header
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public TarEntry(TarHeader header) {
|
||||||
|
this.file = null;
|
||||||
|
this.header = header;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean equals(TarEntry it) {
|
||||||
|
return header.name.toString().equals(it.header.name.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isDescendent(TarEntry desc) {
|
||||||
|
return desc.header.name.toString().startsWith(header.name.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public TarHeader getHeader() {
|
||||||
|
return header;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
String name = header.name.toString();
|
||||||
|
if (header.namePrefix != null && !header.namePrefix.toString().equals("")) {
|
||||||
|
name = header.namePrefix.toString() + "/" + name;
|
||||||
|
}
|
||||||
|
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(String name) {
|
||||||
|
header.name = new StringBuffer(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getUserId() {
|
||||||
|
return header.userId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setUserId(int userId) {
|
||||||
|
header.userId = userId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getGroupId() {
|
||||||
|
return header.groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGroupId(int groupId) {
|
||||||
|
header.groupId = groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getUserName() {
|
||||||
|
return header.userName.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setUserName(String userName) {
|
||||||
|
header.userName = new StringBuffer(userName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getGroupName() {
|
||||||
|
return header.groupName.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGroupName(String groupName) {
|
||||||
|
header.groupName = new StringBuffer(groupName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setIds(int userId, int groupId) {
|
||||||
|
this.setUserId(userId);
|
||||||
|
this.setGroupId(groupId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setModTime(long time) {
|
||||||
|
header.modTime = time / 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setModTime(Date time) {
|
||||||
|
header.modTime = time.getTime() / 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date getModTime() {
|
||||||
|
return new Date(header.modTime * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
public File getFile() {
|
||||||
|
return this.file;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getSize() {
|
||||||
|
return header.size;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSize(long size) {
|
||||||
|
header.size = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the org.kamrazafar.jtar entry is a directory
|
||||||
|
*
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public boolean isDirectory() {
|
||||||
|
if (this.file != null)
|
||||||
|
return this.file.isDirectory();
|
||||||
|
|
||||||
|
if (header != null) {
|
||||||
|
if (header.linkFlag == TarHeader.LF_DIR)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
if (header.name.toString().endsWith("/"))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract header from File
|
||||||
|
*
|
||||||
|
* @param entryName
|
||||||
|
*/
|
||||||
|
public void extractTarHeader(String entryName) {
|
||||||
|
header = TarHeader.createHeader(entryName, file.length(), file.lastModified() / 1000, file.isDirectory());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calculate checksum
|
||||||
|
*
|
||||||
|
* @param buf
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public long computeCheckSum(byte[] buf) {
|
||||||
|
long sum = 0;
|
||||||
|
|
||||||
|
for (int i = 0; i < buf.length; ++i) {
|
||||||
|
sum += 255 & buf[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes the header to the byte buffer
|
||||||
|
*
|
||||||
|
* @param outbuf
|
||||||
|
*/
|
||||||
|
public void writeEntryHeader(byte[] outbuf) {
|
||||||
|
int offset = 0;
|
||||||
|
|
||||||
|
offset = TarHeader.getNameBytes(header.name, outbuf, offset, TarHeader.NAMELEN);
|
||||||
|
offset = Octal.getOctalBytes(header.mode, outbuf, offset, TarHeader.MODELEN);
|
||||||
|
offset = Octal.getOctalBytes(header.userId, outbuf, offset, TarHeader.UIDLEN);
|
||||||
|
offset = Octal.getOctalBytes(header.groupId, outbuf, offset, TarHeader.GIDLEN);
|
||||||
|
|
||||||
|
long size = header.size;
|
||||||
|
|
||||||
|
offset = Octal.getLongOctalBytes(size, outbuf, offset, TarHeader.SIZELEN);
|
||||||
|
offset = Octal.getLongOctalBytes(header.modTime, outbuf, offset, TarHeader.MODTIMELEN);
|
||||||
|
|
||||||
|
int csOffset = offset;
|
||||||
|
for (int c = 0; c < TarHeader.CHKSUMLEN; ++c)
|
||||||
|
outbuf[offset++] = (byte) ' ';
|
||||||
|
|
||||||
|
outbuf[offset++] = header.linkFlag;
|
||||||
|
|
||||||
|
offset = TarHeader.getNameBytes(header.linkName, outbuf, offset, TarHeader.NAMELEN);
|
||||||
|
offset = TarHeader.getNameBytes(header.magic, outbuf, offset, TarHeader.USTAR_MAGICLEN);
|
||||||
|
offset = TarHeader.getNameBytes(header.userName, outbuf, offset, TarHeader.USTAR_USER_NAMELEN);
|
||||||
|
offset = TarHeader.getNameBytes(header.groupName, outbuf, offset, TarHeader.USTAR_GROUP_NAMELEN);
|
||||||
|
offset = Octal.getOctalBytes(header.devMajor, outbuf, offset, TarHeader.USTAR_DEVLEN);
|
||||||
|
offset = Octal.getOctalBytes(header.devMinor, outbuf, offset, TarHeader.USTAR_DEVLEN);
|
||||||
|
offset = TarHeader.getNameBytes(header.namePrefix, outbuf, offset, TarHeader.USTAR_FILENAME_PREFIX);
|
||||||
|
|
||||||
|
for (; offset < outbuf.length;)
|
||||||
|
outbuf[offset++] = 0;
|
||||||
|
|
||||||
|
long checkSum = this.computeCheckSum(outbuf);
|
||||||
|
|
||||||
|
Octal.getCheckSumOctalBytes(checkSum, outbuf, csOffset, TarHeader.CHKSUMLEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses the tar header to the byte buffer
|
||||||
|
*
|
||||||
|
* @param header
|
||||||
|
* @param bh
|
||||||
|
*/
|
||||||
|
public void parseTarHeader(byte[] bh) {
|
||||||
|
int offset = 0;
|
||||||
|
|
||||||
|
header.name = TarHeader.parseName(bh, offset, TarHeader.NAMELEN);
|
||||||
|
offset += TarHeader.NAMELEN;
|
||||||
|
|
||||||
|
header.mode = (int) Octal.parseOctal(bh, offset, TarHeader.MODELEN);
|
||||||
|
offset += TarHeader.MODELEN;
|
||||||
|
|
||||||
|
header.userId = (int) Octal.parseOctal(bh, offset, TarHeader.UIDLEN);
|
||||||
|
offset += TarHeader.UIDLEN;
|
||||||
|
|
||||||
|
header.groupId = (int) Octal.parseOctal(bh, offset, TarHeader.GIDLEN);
|
||||||
|
offset += TarHeader.GIDLEN;
|
||||||
|
|
||||||
|
header.size = Octal.parseOctal(bh, offset, TarHeader.SIZELEN);
|
||||||
|
offset += TarHeader.SIZELEN;
|
||||||
|
|
||||||
|
header.modTime = Octal.parseOctal(bh, offset, TarHeader.MODTIMELEN);
|
||||||
|
offset += TarHeader.MODTIMELEN;
|
||||||
|
|
||||||
|
header.checkSum = (int) Octal.parseOctal(bh, offset, TarHeader.CHKSUMLEN);
|
||||||
|
offset += TarHeader.CHKSUMLEN;
|
||||||
|
|
||||||
|
header.linkFlag = bh[offset++];
|
||||||
|
|
||||||
|
header.linkName = TarHeader.parseName(bh, offset, TarHeader.NAMELEN);
|
||||||
|
offset += TarHeader.NAMELEN;
|
||||||
|
|
||||||
|
header.magic = TarHeader.parseName(bh, offset, TarHeader.USTAR_MAGICLEN);
|
||||||
|
offset += TarHeader.USTAR_MAGICLEN;
|
||||||
|
|
||||||
|
header.userName = TarHeader.parseName(bh, offset, TarHeader.USTAR_USER_NAMELEN);
|
||||||
|
offset += TarHeader.USTAR_USER_NAMELEN;
|
||||||
|
|
||||||
|
header.groupName = TarHeader.parseName(bh, offset, TarHeader.USTAR_GROUP_NAMELEN);
|
||||||
|
offset += TarHeader.USTAR_GROUP_NAMELEN;
|
||||||
|
|
||||||
|
header.devMajor = (int) Octal.parseOctal(bh, offset, TarHeader.USTAR_DEVLEN);
|
||||||
|
offset += TarHeader.USTAR_DEVLEN;
|
||||||
|
|
||||||
|
header.devMinor = (int) Octal.parseOctal(bh, offset, TarHeader.USTAR_DEVLEN);
|
||||||
|
offset += TarHeader.USTAR_DEVLEN;
|
||||||
|
|
||||||
|
header.namePrefix = TarHeader.parseName(bh, offset, TarHeader.USTAR_FILENAME_PREFIX);
|
||||||
|
}
|
||||||
|
}
|
243
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarHeader.java
Executable file
243
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarHeader.java
Executable file
|
@ -0,0 +1,243 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2012 Kamran Zafar
|
||||||
|
*
|
||||||
|
* 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.kamranzafar.jtar;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Header
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* Offset Size Field
|
||||||
|
* 0 100 File name
|
||||||
|
* 100 8 File mode
|
||||||
|
* 108 8 Owner's numeric user ID
|
||||||
|
* 116 8 Group's numeric user ID
|
||||||
|
* 124 12 File size in bytes
|
||||||
|
* 136 12 Last modification time in numeric Unix time format
|
||||||
|
* 148 8 Checksum for header block
|
||||||
|
* 156 1 Link indicator (file type)
|
||||||
|
* 157 100 Name of linked file
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* File Types
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* Value Meaning
|
||||||
|
* '0' Normal file
|
||||||
|
* (ASCII NUL) Normal file (now obsolete)
|
||||||
|
* '1' Hard link
|
||||||
|
* '2' Symbolic link
|
||||||
|
* '3' Character special
|
||||||
|
* '4' Block special
|
||||||
|
* '5' Directory
|
||||||
|
* '6' FIFO
|
||||||
|
* '7' Contigous
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* Ustar header
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* Offset Size Field
|
||||||
|
* 257 6 UStar indicator "ustar"
|
||||||
|
* 263 2 UStar version "00"
|
||||||
|
* 265 32 Owner user name
|
||||||
|
* 297 32 Owner group name
|
||||||
|
* 329 8 Device major number
|
||||||
|
* 337 8 Device minor number
|
||||||
|
* 345 155 Filename prefix
|
||||||
|
* </pre>
|
||||||
|
*/
|
||||||
|
|
||||||
|
public class TarHeader {
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Header
|
||||||
|
*/
|
||||||
|
public static final int NAMELEN = 100;
|
||||||
|
public static final int MODELEN = 8;
|
||||||
|
public static final int UIDLEN = 8;
|
||||||
|
public static final int GIDLEN = 8;
|
||||||
|
public static final int SIZELEN = 12;
|
||||||
|
public static final int MODTIMELEN = 12;
|
||||||
|
public static final int CHKSUMLEN = 8;
|
||||||
|
public static final byte LF_OLDNORM = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* File Types
|
||||||
|
*/
|
||||||
|
public static final byte LF_NORMAL = (byte) '0';
|
||||||
|
public static final byte LF_LINK = (byte) '1';
|
||||||
|
public static final byte LF_SYMLINK = (byte) '2';
|
||||||
|
public static final byte LF_CHR = (byte) '3';
|
||||||
|
public static final byte LF_BLK = (byte) '4';
|
||||||
|
public static final byte LF_DIR = (byte) '5';
|
||||||
|
public static final byte LF_FIFO = (byte) '6';
|
||||||
|
public static final byte LF_CONTIG = (byte) '7';
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Ustar header
|
||||||
|
*/
|
||||||
|
|
||||||
|
public static final String USTAR_MAGIC = "ustar"; // POSIX
|
||||||
|
|
||||||
|
public static final int USTAR_MAGICLEN = 8;
|
||||||
|
public static final int USTAR_USER_NAMELEN = 32;
|
||||||
|
public static final int USTAR_GROUP_NAMELEN = 32;
|
||||||
|
public static final int USTAR_DEVLEN = 8;
|
||||||
|
public static final int USTAR_FILENAME_PREFIX = 155;
|
||||||
|
|
||||||
|
// Header values
|
||||||
|
public StringBuffer name;
|
||||||
|
public int mode;
|
||||||
|
public int userId;
|
||||||
|
public int groupId;
|
||||||
|
public long size;
|
||||||
|
public long modTime;
|
||||||
|
public int checkSum;
|
||||||
|
public byte linkFlag;
|
||||||
|
public StringBuffer linkName;
|
||||||
|
public StringBuffer magic; // ustar indicator and version
|
||||||
|
public StringBuffer userName;
|
||||||
|
public StringBuffer groupName;
|
||||||
|
public int devMajor;
|
||||||
|
public int devMinor;
|
||||||
|
public StringBuffer namePrefix;
|
||||||
|
|
||||||
|
public TarHeader() {
|
||||||
|
this.magic = new StringBuffer(TarHeader.USTAR_MAGIC);
|
||||||
|
|
||||||
|
this.name = new StringBuffer();
|
||||||
|
this.linkName = new StringBuffer();
|
||||||
|
|
||||||
|
String user = System.getProperty("user.name", "");
|
||||||
|
|
||||||
|
if (user.length() > 31)
|
||||||
|
user = user.substring(0, 31);
|
||||||
|
|
||||||
|
this.userId = 0;
|
||||||
|
this.groupId = 0;
|
||||||
|
this.userName = new StringBuffer(user);
|
||||||
|
this.groupName = new StringBuffer("");
|
||||||
|
this.namePrefix = new StringBuffer();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse an entry name from a header buffer.
|
||||||
|
*
|
||||||
|
* @param name
|
||||||
|
* @param header
|
||||||
|
* The header buffer from which to parse.
|
||||||
|
* @param offset
|
||||||
|
* The offset into the buffer from which to parse.
|
||||||
|
* @param length
|
||||||
|
* The number of header bytes to parse.
|
||||||
|
* @return The header's entry name.
|
||||||
|
*/
|
||||||
|
public static StringBuffer parseName(byte[] header, int offset, int length) {
|
||||||
|
StringBuffer result = new StringBuffer(length);
|
||||||
|
|
||||||
|
int end = offset + length;
|
||||||
|
for (int i = offset; i < end; ++i) {
|
||||||
|
if (header[i] == 0)
|
||||||
|
break;
|
||||||
|
result.append((char) header[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine the number of bytes in an entry name.
|
||||||
|
*
|
||||||
|
* @param name
|
||||||
|
* @param header
|
||||||
|
* The header buffer from which to parse.
|
||||||
|
* @param offset
|
||||||
|
* The offset into the buffer from which to parse.
|
||||||
|
* @param length
|
||||||
|
* The number of header bytes to parse.
|
||||||
|
* @return The number of bytes in a header's entry name.
|
||||||
|
*/
|
||||||
|
public static int getNameBytes(StringBuffer name, byte[] buf, int offset, int length) {
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < length && i < name.length(); ++i) {
|
||||||
|
buf[offset + i] = (byte) name.charAt(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (; i < length; ++i) {
|
||||||
|
buf[offset + i] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return offset + length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new header for a file/directory entry.
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* @param name
|
||||||
|
* File name
|
||||||
|
* @param size
|
||||||
|
* File size in bytes
|
||||||
|
* @param modTime
|
||||||
|
* Last modification time in numeric Unix time format
|
||||||
|
* @param dir
|
||||||
|
* Is directory
|
||||||
|
*
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public static TarHeader createHeader(String entryName, long size, long modTime, boolean dir) {
|
||||||
|
String name = entryName;
|
||||||
|
name = TarUtils.trim(name.replace(File.separatorChar, '/'), '/');
|
||||||
|
|
||||||
|
TarHeader header = new TarHeader();
|
||||||
|
header.linkName = new StringBuffer("");
|
||||||
|
|
||||||
|
if (name.length() > 100) {
|
||||||
|
header.namePrefix = new StringBuffer(name.substring(0, name.lastIndexOf('/')));
|
||||||
|
header.name = new StringBuffer(name.substring(name.lastIndexOf('/') + 1));
|
||||||
|
} else {
|
||||||
|
header.name = new StringBuffer(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dir) {
|
||||||
|
header.mode = 040755;
|
||||||
|
header.linkFlag = TarHeader.LF_DIR;
|
||||||
|
if (header.name.charAt(header.name.length() - 1) != '/') {
|
||||||
|
header.name.append("/");
|
||||||
|
}
|
||||||
|
header.size = 0;
|
||||||
|
} else {
|
||||||
|
header.mode = 0100644;
|
||||||
|
header.linkFlag = TarHeader.LF_NORMAL;
|
||||||
|
header.size = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
header.modTime = modTime;
|
||||||
|
header.checkSum = 0;
|
||||||
|
header.devMajor = 0;
|
||||||
|
header.devMinor = 0;
|
||||||
|
|
||||||
|
return header;
|
||||||
|
}
|
||||||
|
}
|
249
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarInputStream.java
Executable file
249
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarInputStream.java
Executable file
|
@ -0,0 +1,249 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2012 Kamran Zafar
|
||||||
|
*
|
||||||
|
* 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.kamranzafar.jtar;
|
||||||
|
|
||||||
|
import java.io.FilterInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Kamran Zafar
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class TarInputStream extends FilterInputStream {
|
||||||
|
|
||||||
|
private static final int SKIP_BUFFER_SIZE = 2048;
|
||||||
|
private TarEntry currentEntry;
|
||||||
|
private long currentFileSize;
|
||||||
|
private long bytesRead;
|
||||||
|
private boolean defaultSkip = false;
|
||||||
|
|
||||||
|
public TarInputStream(InputStream in) {
|
||||||
|
super(in);
|
||||||
|
currentFileSize = 0;
|
||||||
|
bytesRead = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean markSupported() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Not supported
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public synchronized void mark(int readlimit) {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Not supported
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public synchronized void reset() throws IOException {
|
||||||
|
throw new IOException("mark/reset not supported");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read a byte
|
||||||
|
*
|
||||||
|
* @see java.io.FilterInputStream#read()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int read() throws IOException {
|
||||||
|
byte[] buf = new byte[1];
|
||||||
|
|
||||||
|
int res = this.read(buf, 0, 1);
|
||||||
|
|
||||||
|
if (res != -1) {
|
||||||
|
return 0xFF & buf[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the bytes being read exceed the entry size and adjusts the byte
|
||||||
|
* array length. Updates the byte counters
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* @see java.io.FilterInputStream#read(byte[], int, int)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int read(byte[] b, int off, int len) throws IOException {
|
||||||
|
if (currentEntry != null) {
|
||||||
|
if (currentFileSize == currentEntry.getSize()) {
|
||||||
|
return -1;
|
||||||
|
} else if ((currentEntry.getSize() - currentFileSize) < len) {
|
||||||
|
len = (int) (currentEntry.getSize() - currentFileSize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int br = super.read(b, off, len);
|
||||||
|
|
||||||
|
if (br != -1) {
|
||||||
|
if (currentEntry != null) {
|
||||||
|
currentFileSize += br;
|
||||||
|
}
|
||||||
|
|
||||||
|
bytesRead += br;
|
||||||
|
}
|
||||||
|
|
||||||
|
return br;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the next entry in the tar file
|
||||||
|
*
|
||||||
|
* @return TarEntry
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
public TarEntry getNextEntry() throws IOException {
|
||||||
|
closeCurrentEntry();
|
||||||
|
|
||||||
|
byte[] header = new byte[TarConstants.HEADER_BLOCK];
|
||||||
|
byte[] theader = new byte[TarConstants.HEADER_BLOCK];
|
||||||
|
int tr = 0;
|
||||||
|
|
||||||
|
// Read full header
|
||||||
|
while (tr < TarConstants.HEADER_BLOCK) {
|
||||||
|
int res = read(theader, 0, TarConstants.HEADER_BLOCK - tr);
|
||||||
|
|
||||||
|
if (res < 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
System.arraycopy(theader, 0, header, tr, res);
|
||||||
|
tr += res;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if record is null
|
||||||
|
boolean eof = true;
|
||||||
|
for (byte b : header) {
|
||||||
|
if (b != 0) {
|
||||||
|
eof = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!eof) {
|
||||||
|
currentEntry = new TarEntry(header);
|
||||||
|
}
|
||||||
|
|
||||||
|
return currentEntry;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the current offset (in bytes) from the beginning of the stream.
|
||||||
|
* This can be used to find out at which point in a tar file an entry's content begins, for instance.
|
||||||
|
*/
|
||||||
|
public long getCurrentOffset() {
|
||||||
|
return bytesRead;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closes the current tar entry
|
||||||
|
*
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
protected void closeCurrentEntry() throws IOException {
|
||||||
|
if (currentEntry != null) {
|
||||||
|
if (currentEntry.getSize() > currentFileSize) {
|
||||||
|
// Not fully read, skip rest of the bytes
|
||||||
|
long bs = 0;
|
||||||
|
while (bs < currentEntry.getSize() - currentFileSize) {
|
||||||
|
long res = skip(currentEntry.getSize() - currentFileSize - bs);
|
||||||
|
|
||||||
|
if (res == 0 && currentEntry.getSize() - currentFileSize > 0) {
|
||||||
|
// I suspect file corruption
|
||||||
|
throw new IOException("Possible tar file corruption");
|
||||||
|
}
|
||||||
|
|
||||||
|
bs += res;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
currentEntry = null;
|
||||||
|
currentFileSize = 0L;
|
||||||
|
skipPad();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Skips the pad at the end of each tar entry file content
|
||||||
|
*
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
protected void skipPad() throws IOException {
|
||||||
|
if (bytesRead > 0) {
|
||||||
|
int extra = (int) (bytesRead % TarConstants.DATA_BLOCK);
|
||||||
|
|
||||||
|
if (extra > 0) {
|
||||||
|
long bs = 0;
|
||||||
|
while (bs < TarConstants.DATA_BLOCK - extra) {
|
||||||
|
long res = skip(TarConstants.DATA_BLOCK - extra - bs);
|
||||||
|
bs += res;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Skips 'n' bytes on the InputStream<br>
|
||||||
|
* Overrides default implementation of skip
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public long skip(long n) throws IOException {
|
||||||
|
if (defaultSkip) {
|
||||||
|
// use skip method of parent stream
|
||||||
|
// may not work if skip not implemented by parent
|
||||||
|
long bs = super.skip(n);
|
||||||
|
bytesRead += bs;
|
||||||
|
|
||||||
|
return bs;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (n <= 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
long left = n;
|
||||||
|
byte[] sBuff = new byte[SKIP_BUFFER_SIZE];
|
||||||
|
|
||||||
|
while (left > 0) {
|
||||||
|
int res = read(sBuff, 0, (int) (left < SKIP_BUFFER_SIZE ? left : SKIP_BUFFER_SIZE));
|
||||||
|
if (res < 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
left -= res;
|
||||||
|
}
|
||||||
|
|
||||||
|
return n - left;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isDefaultSkip() {
|
||||||
|
return defaultSkip;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDefaultSkip(boolean defaultSkip) {
|
||||||
|
this.defaultSkip = defaultSkip;
|
||||||
|
}
|
||||||
|
}
|
163
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarOutputStream.java
Executable file
163
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarOutputStream.java
Executable file
|
@ -0,0 +1,163 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2012 Kamran Zafar
|
||||||
|
*
|
||||||
|
* 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.kamranzafar.jtar;
|
||||||
|
|
||||||
|
import java.io.BufferedOutputStream;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.io.RandomAccessFile;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Kamran Zafar
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class TarOutputStream extends OutputStream {
|
||||||
|
private final OutputStream out;
|
||||||
|
private long bytesWritten;
|
||||||
|
private long currentFileSize;
|
||||||
|
private TarEntry currentEntry;
|
||||||
|
|
||||||
|
public TarOutputStream(OutputStream out) {
|
||||||
|
this.out = out;
|
||||||
|
bytesWritten = 0;
|
||||||
|
currentFileSize = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TarOutputStream(final File fout) throws FileNotFoundException {
|
||||||
|
this.out = new BufferedOutputStream(new FileOutputStream(fout));
|
||||||
|
bytesWritten = 0;
|
||||||
|
currentFileSize = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opens a file for writing.
|
||||||
|
*/
|
||||||
|
public TarOutputStream(final File fout, final boolean append) throws IOException {
|
||||||
|
@SuppressWarnings("resource")
|
||||||
|
RandomAccessFile raf = new RandomAccessFile(fout, "rw");
|
||||||
|
final long fileSize = fout.length();
|
||||||
|
if (append && fileSize > TarConstants.EOF_BLOCK) {
|
||||||
|
raf.seek(fileSize - TarConstants.EOF_BLOCK);
|
||||||
|
}
|
||||||
|
out = new BufferedOutputStream(new FileOutputStream(raf.getFD()));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Appends the EOF record and closes the stream
|
||||||
|
*
|
||||||
|
* @see java.io.FilterOutputStream#close()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void close() throws IOException {
|
||||||
|
closeCurrentEntry();
|
||||||
|
write( new byte[TarConstants.EOF_BLOCK] );
|
||||||
|
out.close();
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Writes a byte to the stream and updates byte counters
|
||||||
|
*
|
||||||
|
* @see java.io.FilterOutputStream#write(int)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void write(int b) throws IOException {
|
||||||
|
out.write( b );
|
||||||
|
bytesWritten += 1;
|
||||||
|
|
||||||
|
if (currentEntry != null) {
|
||||||
|
currentFileSize += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the bytes being written exceed the current entry size.
|
||||||
|
*
|
||||||
|
* @see java.io.FilterOutputStream#write(byte[], int, int)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void write(byte[] b, int off, int len) throws IOException {
|
||||||
|
if (currentEntry != null && !currentEntry.isDirectory()) {
|
||||||
|
if (currentEntry.getSize() < currentFileSize + len) {
|
||||||
|
throw new IOException( "The current entry[" + currentEntry.getName() + "] size["
|
||||||
|
+ currentEntry.getSize() + "] is smaller than the bytes[" + ( currentFileSize + len )
|
||||||
|
+ "] being written." );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
out.write( b, off, len );
|
||||||
|
|
||||||
|
bytesWritten += len;
|
||||||
|
|
||||||
|
if (currentEntry != null) {
|
||||||
|
currentFileSize += len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes the next tar entry header on the stream
|
||||||
|
*
|
||||||
|
* @param entry
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
public void putNextEntry(TarEntry entry) throws IOException {
|
||||||
|
closeCurrentEntry();
|
||||||
|
|
||||||
|
byte[] header = new byte[TarConstants.HEADER_BLOCK];
|
||||||
|
entry.writeEntryHeader( header );
|
||||||
|
|
||||||
|
write( header );
|
||||||
|
|
||||||
|
currentEntry = entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closes the current tar entry
|
||||||
|
*
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
protected void closeCurrentEntry() throws IOException {
|
||||||
|
if (currentEntry != null) {
|
||||||
|
if (currentEntry.getSize() > currentFileSize) {
|
||||||
|
throw new IOException( "The current entry[" + currentEntry.getName() + "] of size["
|
||||||
|
+ currentEntry.getSize() + "] has not been fully written." );
|
||||||
|
}
|
||||||
|
|
||||||
|
currentEntry = null;
|
||||||
|
currentFileSize = 0;
|
||||||
|
|
||||||
|
pad();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pads the last content block
|
||||||
|
*
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
protected void pad() throws IOException {
|
||||||
|
if (bytesWritten > 0) {
|
||||||
|
int extra = (int) ( bytesWritten % TarConstants.DATA_BLOCK );
|
||||||
|
|
||||||
|
if (extra > 0) {
|
||||||
|
write( new byte[TarConstants.DATA_BLOCK - extra] );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
96
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarUtils.java
Executable file
96
android/bootstrap/build/src/main/java/org/kamranzafar/jtar/TarUtils.java
Executable file
|
@ -0,0 +1,96 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2012 Kamran Zafar
|
||||||
|
*
|
||||||
|
* 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.kamranzafar.jtar;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Kamran
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class TarUtils {
|
||||||
|
/**
|
||||||
|
* Determines the tar file size of the given folder/file path
|
||||||
|
*
|
||||||
|
* @param path
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public static long calculateTarSize(File path) {
|
||||||
|
return tarSize(path) + TarConstants.EOF_BLOCK;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static long tarSize(File dir) {
|
||||||
|
long size = 0;
|
||||||
|
|
||||||
|
if (dir.isFile()) {
|
||||||
|
return entrySize(dir.length());
|
||||||
|
} else {
|
||||||
|
File[] subFiles = dir.listFiles();
|
||||||
|
|
||||||
|
if (subFiles != null && subFiles.length > 0) {
|
||||||
|
for (File file : subFiles) {
|
||||||
|
if (file.isFile()) {
|
||||||
|
size += entrySize(file.length());
|
||||||
|
} else {
|
||||||
|
size += tarSize(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Empty folder header
|
||||||
|
return TarConstants.HEADER_BLOCK;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static long entrySize(long fileSize) {
|
||||||
|
long size = 0;
|
||||||
|
size += TarConstants.HEADER_BLOCK; // Header
|
||||||
|
size += fileSize; // File size
|
||||||
|
|
||||||
|
long extra = size % TarConstants.DATA_BLOCK;
|
||||||
|
|
||||||
|
if (extra > 0) {
|
||||||
|
size += (TarConstants.DATA_BLOCK - extra); // pad
|
||||||
|
}
|
||||||
|
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String trim(String s, char c) {
|
||||||
|
StringBuffer tmp = new StringBuffer(s);
|
||||||
|
for (int i = 0; i < tmp.length(); i++) {
|
||||||
|
if (tmp.charAt(i) != c) {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
tmp.deleteCharAt(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = tmp.length() - 1; i >= 0; i--) {
|
||||||
|
if (tmp.charAt(i) != c) {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
tmp.deleteCharAt(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return tmp.toString();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,400 @@
|
||||||
|
|
||||||
|
package org.kivy.android;
|
||||||
|
|
||||||
|
import java.net.Socket;
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
|
|
||||||
|
import android.os.SystemClock;
|
||||||
|
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
|
import java.io.FileWriter;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
|
import android.app.*;
|
||||||
|
import android.content.*;
|
||||||
|
import android.view.*;
|
||||||
|
import android.view.SurfaceView;
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.util.Log;
|
||||||
|
import android.widget.Toast;
|
||||||
|
import android.os.AsyncTask;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.os.PowerManager;
|
||||||
|
import android.graphics.PixelFormat;
|
||||||
|
import android.view.SurfaceHolder;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.pm.ActivityInfo;
|
||||||
|
import android.content.pm.PackageManager;
|
||||||
|
import android.content.pm.ApplicationInfo;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.widget.ImageView;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import android.graphics.Bitmap;
|
||||||
|
import android.graphics.BitmapFactory;
|
||||||
|
import android.graphics.Color;
|
||||||
|
|
||||||
|
import android.widget.AbsoluteLayout;
|
||||||
|
|
||||||
|
import android.webkit.WebViewClient;
|
||||||
|
import android.webkit.WebView;
|
||||||
|
|
||||||
|
import org.kivy.android.PythonUtil;
|
||||||
|
|
||||||
|
import org.renpy.android.ResourceManager;
|
||||||
|
import org.renpy.android.AssetExtract;
|
||||||
|
|
||||||
|
public class PythonActivity extends Activity {
|
||||||
|
// This activity is modified from a mixture of the SDLActivity and
|
||||||
|
// PythonActivity in the SDL2 bootstrap, but removing all the SDL2
|
||||||
|
// specifics.
|
||||||
|
|
||||||
|
private static final String TAG = "PythonActivity";
|
||||||
|
|
||||||
|
public static PythonActivity mActivity = null;
|
||||||
|
|
||||||
|
/** If shared libraries (e.g. the native application) could not be loaded. */
|
||||||
|
public static boolean mBrokenLibraries;
|
||||||
|
|
||||||
|
protected static Thread mPythonThread;
|
||||||
|
|
||||||
|
private ResourceManager resourceManager = null;
|
||||||
|
private Bundle mMetaData = null;
|
||||||
|
private PowerManager.WakeLock mWakeLock = null;
|
||||||
|
|
||||||
|
public String getAppRoot() {
|
||||||
|
String app_root = getFilesDir().getAbsolutePath() + "/app";
|
||||||
|
return app_root;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void initialize() {
|
||||||
|
// The static nature of the singleton and Android quirkiness force us to initialize everything here
|
||||||
|
// Otherwise, when exiting the app and returning to it, these variables *keep* their pre exit values
|
||||||
|
mBrokenLibraries = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
|
Log.v(TAG, "My oncreate running");
|
||||||
|
resourceManager = new ResourceManager(this);
|
||||||
|
|
||||||
|
Log.v(TAG, "Ready to unpack");
|
||||||
|
File app_root_file = new File(getAppRoot());
|
||||||
|
unpackData("private", app_root_file);
|
||||||
|
|
||||||
|
Log.v(TAG, "About to do super onCreate");
|
||||||
|
super.onCreate(savedInstanceState);
|
||||||
|
Log.v(TAG, "Did super onCreate");
|
||||||
|
|
||||||
|
this.mActivity = this;
|
||||||
|
//this.showLoadingScreen();
|
||||||
|
Log.v("Python", "Device: " + android.os.Build.DEVICE);
|
||||||
|
Log.v("Python", "Model: " + android.os.Build.MODEL);
|
||||||
|
|
||||||
|
//Log.v(TAG, "Ready to unpack");
|
||||||
|
//new UnpackFilesTask().execute(getAppRoot());
|
||||||
|
|
||||||
|
PythonActivity.initialize();
|
||||||
|
|
||||||
|
// Load shared libraries
|
||||||
|
String errorMsgBrokenLib = "";
|
||||||
|
try {
|
||||||
|
loadLibraries();
|
||||||
|
} catch(UnsatisfiedLinkError e) {
|
||||||
|
System.err.println(e.getMessage());
|
||||||
|
mBrokenLibraries = true;
|
||||||
|
errorMsgBrokenLib = e.getMessage();
|
||||||
|
} catch(Exception e) {
|
||||||
|
System.err.println(e.getMessage());
|
||||||
|
mBrokenLibraries = true;
|
||||||
|
errorMsgBrokenLib = e.getMessage();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mBrokenLibraries)
|
||||||
|
{
|
||||||
|
AlertDialog.Builder dlgAlert = new AlertDialog.Builder(this);
|
||||||
|
dlgAlert.setMessage("An error occurred while trying to load the application libraries. Please try again and/or reinstall."
|
||||||
|
+ System.getProperty("line.separator")
|
||||||
|
+ System.getProperty("line.separator")
|
||||||
|
+ "Error: " + errorMsgBrokenLib);
|
||||||
|
dlgAlert.setTitle("Python Error");
|
||||||
|
dlgAlert.setPositiveButton("Exit",
|
||||||
|
new DialogInterface.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(DialogInterface dialog,int id) {
|
||||||
|
// if this button is clicked, close current activity
|
||||||
|
PythonActivity.mActivity.finish();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
dlgAlert.setCancelable(false);
|
||||||
|
dlgAlert.create().show();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up the Python environment
|
||||||
|
String app_root_dir = getAppRoot();
|
||||||
|
String mFilesDirectory = mActivity.getFilesDir().getAbsolutePath();
|
||||||
|
|
||||||
|
Log.v(TAG, "Setting env vars for start.c and Python to use");
|
||||||
|
PythonActivity.nativeSetEnv("ANDROID_ENTRYPOINT", "main.pyo");
|
||||||
|
PythonActivity.nativeSetEnv("ANDROID_ARGUMENT", app_root_dir);
|
||||||
|
PythonActivity.nativeSetEnv("ANDROID_APP_PATH", app_root_dir);
|
||||||
|
PythonActivity.nativeSetEnv("ANDROID_PRIVATE", mFilesDirectory);
|
||||||
|
PythonActivity.nativeSetEnv("ANDROID_UNPACK", app_root_dir);
|
||||||
|
PythonActivity.nativeSetEnv("PYTHONHOME", app_root_dir);
|
||||||
|
PythonActivity.nativeSetEnv("PYTHONPATH", app_root_dir + ":" + app_root_dir + "/lib");
|
||||||
|
PythonActivity.nativeSetEnv("PYTHONOPTIMIZE", "2");
|
||||||
|
|
||||||
|
try {
|
||||||
|
Log.v(TAG, "Access to our meta-data...");
|
||||||
|
mActivity.mMetaData = mActivity.getPackageManager().getApplicationInfo(
|
||||||
|
mActivity.getPackageName(), PackageManager.GET_META_DATA).metaData;
|
||||||
|
|
||||||
|
PowerManager pm = (PowerManager) mActivity.getSystemService(Context.POWER_SERVICE);
|
||||||
|
if ( mActivity.mMetaData.getInt("wakelock") == 1 ) {
|
||||||
|
mActivity.mWakeLock = pm.newWakeLock(PowerManager.SCREEN_BRIGHT_WAKE_LOCK, "Screen On");
|
||||||
|
mActivity.mWakeLock.acquire();
|
||||||
|
}
|
||||||
|
} catch (PackageManager.NameNotFoundException e) {
|
||||||
|
}
|
||||||
|
|
||||||
|
final Thread pythonThread = new Thread(new PythonMain(), "PythonThread");
|
||||||
|
PythonActivity.mPythonThread = pythonThread;
|
||||||
|
pythonThread.start();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDestroy() {
|
||||||
|
Log.i("Destroy", "end of app");
|
||||||
|
super.onDestroy();
|
||||||
|
|
||||||
|
// make sure all child threads (python_thread) are stopped
|
||||||
|
android.os.Process.killProcess(android.os.Process.myPid());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void loadLibraries() {
|
||||||
|
String app_root = new String(getAppRoot());
|
||||||
|
File app_root_file = new File(app_root);
|
||||||
|
PythonUtil.loadLibraries(app_root_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void recursiveDelete(File f) {
|
||||||
|
if (f.isDirectory()) {
|
||||||
|
for (File r : f.listFiles()) {
|
||||||
|
recursiveDelete(r);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
f.delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show an error using a toast. (Only makes sense from non-UI
|
||||||
|
* threads.)
|
||||||
|
*/
|
||||||
|
public void toastError(final String msg) {
|
||||||
|
|
||||||
|
final Activity thisActivity = this;
|
||||||
|
|
||||||
|
runOnUiThread(new Runnable () {
|
||||||
|
public void run() {
|
||||||
|
Toast.makeText(thisActivity, msg, Toast.LENGTH_LONG).show();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Wait to show the error.
|
||||||
|
synchronized (this) {
|
||||||
|
try {
|
||||||
|
this.wait(1000);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void unpackData(final String resource, File target) {
|
||||||
|
|
||||||
|
Log.v(TAG, "UNPACKING!!! " + resource + " " + target.getName());
|
||||||
|
|
||||||
|
// The version of data in memory and on disk.
|
||||||
|
String data_version = resourceManager.getString(resource + "_version");
|
||||||
|
String disk_version = null;
|
||||||
|
|
||||||
|
Log.v(TAG, "Data version is " + data_version);
|
||||||
|
|
||||||
|
// If no version, no unpacking is necessary.
|
||||||
|
if (data_version == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the current disk version, if any.
|
||||||
|
String filesDir = target.getAbsolutePath();
|
||||||
|
String disk_version_fn = filesDir + "/" + resource + ".version";
|
||||||
|
|
||||||
|
try {
|
||||||
|
byte buf[] = new byte[64];
|
||||||
|
InputStream is = new FileInputStream(disk_version_fn);
|
||||||
|
int len = is.read(buf);
|
||||||
|
disk_version = new String(buf, 0, len);
|
||||||
|
is.close();
|
||||||
|
} catch (Exception e) {
|
||||||
|
disk_version = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the disk data is out of date, extract it and write the
|
||||||
|
// version file.
|
||||||
|
// if (! data_version.equals(disk_version)) {
|
||||||
|
if (! data_version.equals(disk_version)) {
|
||||||
|
Log.v(TAG, "Extracting " + resource + " assets.");
|
||||||
|
|
||||||
|
recursiveDelete(target);
|
||||||
|
target.mkdirs();
|
||||||
|
|
||||||
|
AssetExtract ae = new AssetExtract(this);
|
||||||
|
if (!ae.extractTar(resource + ".mp3", target.getAbsolutePath())) {
|
||||||
|
toastError("Could not extract " + resource + " data.");
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Write .nomedia.
|
||||||
|
new File(target, ".nomedia").createNewFile();
|
||||||
|
|
||||||
|
// Write version file.
|
||||||
|
FileOutputStream os = new FileOutputStream(disk_version_fn);
|
||||||
|
os.write(data_version.getBytes());
|
||||||
|
os.close();
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.w("python", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
long lastBackClick = SystemClock.elapsedRealtime();
|
||||||
|
@Override
|
||||||
|
public boolean onKeyDown(int keyCode, KeyEvent event) {
|
||||||
|
// If it wasn't the Back key or there's no web page history, bubble up to the default
|
||||||
|
// system behavior (probably exit the activity)
|
||||||
|
if (SystemClock.elapsedRealtime() - lastBackClick > 2000){
|
||||||
|
lastBackClick = SystemClock.elapsedRealtime();
|
||||||
|
Toast.makeText(this, "Click again to close the app",
|
||||||
|
Toast.LENGTH_LONG).show();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
lastBackClick = SystemClock.elapsedRealtime();
|
||||||
|
return super.onKeyDown(keyCode, event);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
// Listener interface for onNewIntent
|
||||||
|
//
|
||||||
|
|
||||||
|
public interface NewIntentListener {
|
||||||
|
void onNewIntent(Intent intent);
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<NewIntentListener> newIntentListeners = null;
|
||||||
|
|
||||||
|
public void registerNewIntentListener(NewIntentListener listener) {
|
||||||
|
if ( this.newIntentListeners == null )
|
||||||
|
this.newIntentListeners = Collections.synchronizedList(new ArrayList<NewIntentListener>());
|
||||||
|
this.newIntentListeners.add(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void unregisterNewIntentListener(NewIntentListener listener) {
|
||||||
|
if ( this.newIntentListeners == null )
|
||||||
|
return;
|
||||||
|
this.newIntentListeners.remove(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onNewIntent(Intent intent) {
|
||||||
|
if ( this.newIntentListeners == null )
|
||||||
|
return;
|
||||||
|
this.onResume();
|
||||||
|
synchronized ( this.newIntentListeners ) {
|
||||||
|
Iterator<NewIntentListener> iterator = this.newIntentListeners.iterator();
|
||||||
|
while ( iterator.hasNext() ) {
|
||||||
|
(iterator.next()).onNewIntent(intent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
// Listener interface for onActivityResult
|
||||||
|
//
|
||||||
|
|
||||||
|
public interface ActivityResultListener {
|
||||||
|
void onActivityResult(int requestCode, int resultCode, Intent data);
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<ActivityResultListener> activityResultListeners = null;
|
||||||
|
|
||||||
|
public void registerActivityResultListener(ActivityResultListener listener) {
|
||||||
|
if ( this.activityResultListeners == null )
|
||||||
|
this.activityResultListeners = Collections.synchronizedList(new ArrayList<ActivityResultListener>());
|
||||||
|
this.activityResultListeners.add(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void unregisterActivityResultListener(ActivityResultListener listener) {
|
||||||
|
if ( this.activityResultListeners == null )
|
||||||
|
return;
|
||||||
|
this.activityResultListeners.remove(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
|
||||||
|
if ( this.activityResultListeners == null )
|
||||||
|
return;
|
||||||
|
this.onResume();
|
||||||
|
synchronized ( this.activityResultListeners ) {
|
||||||
|
Iterator<ActivityResultListener> iterator = this.activityResultListeners.iterator();
|
||||||
|
while ( iterator.hasNext() )
|
||||||
|
(iterator.next()).onActivityResult(requestCode, resultCode, intent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void start_service(String serviceTitle, String serviceDescription,
|
||||||
|
String pythonServiceArgument) {
|
||||||
|
Intent serviceIntent = new Intent(PythonActivity.mActivity, PythonService.class);
|
||||||
|
String argument = PythonActivity.mActivity.getFilesDir().getAbsolutePath();
|
||||||
|
String filesDirectory = argument;
|
||||||
|
String app_root_dir = PythonActivity.mActivity.getAppRoot();
|
||||||
|
serviceIntent.putExtra("androidPrivate", argument);
|
||||||
|
serviceIntent.putExtra("androidArgument", app_root_dir);
|
||||||
|
serviceIntent.putExtra("serviceEntrypoint", "service/main.pyo");
|
||||||
|
serviceIntent.putExtra("pythonName", "python");
|
||||||
|
serviceIntent.putExtra("pythonHome", app_root_dir);
|
||||||
|
serviceIntent.putExtra("pythonPath", app_root_dir + ":" + app_root_dir + "/lib");
|
||||||
|
serviceIntent.putExtra("serviceTitle", serviceTitle);
|
||||||
|
serviceIntent.putExtra("serviceDescription", serviceDescription);
|
||||||
|
serviceIntent.putExtra("pythonServiceArgument", pythonServiceArgument);
|
||||||
|
PythonActivity.mActivity.startService(serviceIntent);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void stop_service() {
|
||||||
|
Intent serviceIntent = new Intent(PythonActivity.mActivity, PythonService.class);
|
||||||
|
PythonActivity.mActivity.stopService(serviceIntent);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static native void nativeSetEnv(String j_name, String j_value);
|
||||||
|
public static native int nativeInit(Object arguments);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class PythonMain implements Runnable {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
PythonActivity.nativeInit(new String[0]);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,45 @@
|
||||||
|
package org.kivy.android.concurrency;
|
||||||
|
|
||||||
|
import java.util.concurrent.locks.Condition;
|
||||||
|
import java.util.concurrent.locks.Lock;
|
||||||
|
import java.util.concurrent.locks.ReentrantLock;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by ryan on 3/28/14.
|
||||||
|
*/
|
||||||
|
public class PythonEvent {
|
||||||
|
private final Lock lock = new ReentrantLock();
|
||||||
|
private final Condition cond = lock.newCondition();
|
||||||
|
private boolean flag = false;
|
||||||
|
|
||||||
|
public void set() {
|
||||||
|
lock.lock();
|
||||||
|
try {
|
||||||
|
flag = true;
|
||||||
|
cond.signalAll();
|
||||||
|
} finally {
|
||||||
|
lock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void wait_() throws InterruptedException {
|
||||||
|
lock.lock();
|
||||||
|
try {
|
||||||
|
while (!flag) {
|
||||||
|
cond.await();
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
lock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clear() {
|
||||||
|
lock.lock();
|
||||||
|
try {
|
||||||
|
flag = false;
|
||||||
|
cond.signalAll();
|
||||||
|
} finally {
|
||||||
|
lock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
package org.kivy.android.concurrency;
|
||||||
|
|
||||||
|
import java.util.concurrent.locks.Lock;
|
||||||
|
import java.util.concurrent.locks.ReentrantLock;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by ryan on 3/28/14.
|
||||||
|
*/
|
||||||
|
public class PythonLock {
|
||||||
|
private final Lock lock = new ReentrantLock();
|
||||||
|
|
||||||
|
public void acquire() {
|
||||||
|
lock.lock();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void release() {
|
||||||
|
lock.unlock();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,257 @@
|
||||||
|
package org.renpy.android;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import android.content.BroadcastReceiver;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.IntentFilter;
|
||||||
|
import android.hardware.Sensor;
|
||||||
|
import android.hardware.SensorEvent;
|
||||||
|
import android.hardware.SensorEventListener;
|
||||||
|
import android.hardware.SensorManager;
|
||||||
|
import android.net.ConnectivityManager;
|
||||||
|
import android.net.NetworkInfo;
|
||||||
|
import android.net.wifi.ScanResult;
|
||||||
|
import android.net.wifi.WifiManager;
|
||||||
|
import android.os.Vibrator;
|
||||||
|
import android.view.View;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Methods that are expected to be called via JNI, to access the device's
|
||||||
|
* non-screen hardware. (For example, the vibration and accelerometer.)
|
||||||
|
*/
|
||||||
|
public class Hardware {
|
||||||
|
|
||||||
|
// The context.
|
||||||
|
static Context context;
|
||||||
|
static View view;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Vibrate for s seconds.
|
||||||
|
*/
|
||||||
|
public static void vibrate(double s) {
|
||||||
|
Vibrator v = (Vibrator) context
|
||||||
|
.getSystemService(Context.VIBRATOR_SERVICE);
|
||||||
|
if (v != null) {
|
||||||
|
v.vibrate((int) (1000 * s));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an Overview of all Hardware Sensors of an Android Device
|
||||||
|
*/
|
||||||
|
public static String getHardwareSensors() {
|
||||||
|
SensorManager sm = (SensorManager) context
|
||||||
|
.getSystemService(Context.SENSOR_SERVICE);
|
||||||
|
List<Sensor> allSensors = sm.getSensorList(Sensor.TYPE_ALL);
|
||||||
|
|
||||||
|
if (allSensors != null) {
|
||||||
|
String resultString = "";
|
||||||
|
for (Sensor s : allSensors) {
|
||||||
|
resultString += String.format("Name=" + s.getName());
|
||||||
|
resultString += String.format(",Vendor=" + s.getVendor());
|
||||||
|
resultString += String.format(",Version=" + s.getVersion());
|
||||||
|
resultString += String.format(",MaximumRange="
|
||||||
|
+ s.getMaximumRange());
|
||||||
|
// XXX MinDelay is not in the 2.2
|
||||||
|
// resultString += String.format(",MinDelay=" +
|
||||||
|
// s.getMinDelay());
|
||||||
|
resultString += String.format(",Power=" + s.getPower());
|
||||||
|
resultString += String.format(",Type=" + s.getType() + "\n");
|
||||||
|
}
|
||||||
|
return resultString;
|
||||||
|
}
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get Access to 3 Axis Hardware Sensors Accelerometer, Orientation and
|
||||||
|
* Magnetic Field Sensors
|
||||||
|
*/
|
||||||
|
public static class generic3AxisSensor implements SensorEventListener {
|
||||||
|
private final SensorManager sSensorManager;
|
||||||
|
private final Sensor sSensor;
|
||||||
|
private final int sSensorType;
|
||||||
|
SensorEvent sSensorEvent;
|
||||||
|
|
||||||
|
public generic3AxisSensor(int sensorType) {
|
||||||
|
sSensorType = sensorType;
|
||||||
|
sSensorManager = (SensorManager) context
|
||||||
|
.getSystemService(Context.SENSOR_SERVICE);
|
||||||
|
sSensor = sSensorManager.getDefaultSensor(sSensorType);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void onAccuracyChanged(Sensor sensor, int accuracy) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void onSensorChanged(SensorEvent event) {
|
||||||
|
sSensorEvent = event;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enable or disable the Sensor by registering/unregistering
|
||||||
|
*/
|
||||||
|
public void changeStatus(boolean enable) {
|
||||||
|
if (enable) {
|
||||||
|
sSensorManager.registerListener(this, sSensor,
|
||||||
|
SensorManager.SENSOR_DELAY_NORMAL);
|
||||||
|
} else {
|
||||||
|
sSensorManager.unregisterListener(this, sSensor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read the Sensor
|
||||||
|
*/
|
||||||
|
public float[] readSensor() {
|
||||||
|
if (sSensorEvent != null) {
|
||||||
|
return sSensorEvent.values;
|
||||||
|
} else {
|
||||||
|
float rv[] = { 0f, 0f, 0f };
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static generic3AxisSensor accelerometerSensor = null;
|
||||||
|
public static generic3AxisSensor orientationSensor = null;
|
||||||
|
public static generic3AxisSensor magneticFieldSensor = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* functions for backward compatibility reasons
|
||||||
|
*/
|
||||||
|
|
||||||
|
public static void accelerometerEnable(boolean enable) {
|
||||||
|
if (accelerometerSensor == null)
|
||||||
|
accelerometerSensor = new generic3AxisSensor(
|
||||||
|
Sensor.TYPE_ACCELEROMETER);
|
||||||
|
accelerometerSensor.changeStatus(enable);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static float[] accelerometerReading() {
|
||||||
|
float rv[] = { 0f, 0f, 0f };
|
||||||
|
if (accelerometerSensor == null)
|
||||||
|
return rv;
|
||||||
|
return (float[]) accelerometerSensor.readSensor();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void orientationSensorEnable(boolean enable) {
|
||||||
|
if (orientationSensor == null)
|
||||||
|
orientationSensor = new generic3AxisSensor(Sensor.TYPE_ORIENTATION);
|
||||||
|
orientationSensor.changeStatus(enable);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static float[] orientationSensorReading() {
|
||||||
|
float rv[] = { 0f, 0f, 0f };
|
||||||
|
if (orientationSensor == null)
|
||||||
|
return rv;
|
||||||
|
return (float[]) orientationSensor.readSensor();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void magneticFieldSensorEnable(boolean enable) {
|
||||||
|
if (magneticFieldSensor == null)
|
||||||
|
magneticFieldSensor = new generic3AxisSensor(
|
||||||
|
Sensor.TYPE_MAGNETIC_FIELD);
|
||||||
|
magneticFieldSensor.changeStatus(enable);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static float[] magneticFieldSensorReading() {
|
||||||
|
float rv[] = { 0f, 0f, 0f };
|
||||||
|
if (magneticFieldSensor == null)
|
||||||
|
return rv;
|
||||||
|
return (float[]) magneticFieldSensor.readSensor();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scan WiFi networks
|
||||||
|
*/
|
||||||
|
static List<ScanResult> latestResult;
|
||||||
|
|
||||||
|
public static void enableWifiScanner() {
|
||||||
|
IntentFilter i = new IntentFilter();
|
||||||
|
i.addAction(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION);
|
||||||
|
|
||||||
|
context.registerReceiver(new BroadcastReceiver() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onReceive(Context c, Intent i) {
|
||||||
|
// Code to execute when SCAN_RESULTS_AVAILABLE_ACTION event
|
||||||
|
// occurs
|
||||||
|
WifiManager w = (WifiManager) c
|
||||||
|
.getSystemService(Context.WIFI_SERVICE);
|
||||||
|
latestResult = w.getScanResults(); // Returns a <list> of
|
||||||
|
// scanResults
|
||||||
|
}
|
||||||
|
|
||||||
|
}, i);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String scanWifi() {
|
||||||
|
|
||||||
|
// Now you can call this and it should execute the broadcastReceiver's
|
||||||
|
// onReceive()
|
||||||
|
WifiManager wm = (WifiManager) context
|
||||||
|
.getSystemService(Context.WIFI_SERVICE);
|
||||||
|
boolean a = wm.startScan();
|
||||||
|
|
||||||
|
if (latestResult != null) {
|
||||||
|
|
||||||
|
String latestResultString = "";
|
||||||
|
for (ScanResult result : latestResult) {
|
||||||
|
latestResultString += String.format("%s\t%s\t%d\n",
|
||||||
|
result.SSID, result.BSSID, result.level);
|
||||||
|
}
|
||||||
|
|
||||||
|
return latestResultString;
|
||||||
|
}
|
||||||
|
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* network state
|
||||||
|
*/
|
||||||
|
|
||||||
|
public static boolean network_state = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check network state directly
|
||||||
|
*
|
||||||
|
* (only one connection can be active at a given moment, detects all network
|
||||||
|
* type)
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public static boolean checkNetwork() {
|
||||||
|
boolean state = false;
|
||||||
|
final ConnectivityManager conMgr = (ConnectivityManager) context
|
||||||
|
.getSystemService(Context.CONNECTIVITY_SERVICE);
|
||||||
|
|
||||||
|
final NetworkInfo activeNetwork = conMgr.getActiveNetworkInfo();
|
||||||
|
if (activeNetwork != null && activeNetwork.isConnected()) {
|
||||||
|
state = true;
|
||||||
|
} else {
|
||||||
|
state = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To recieve network state changes
|
||||||
|
*/
|
||||||
|
public static void registerNetworkCheck() {
|
||||||
|
IntentFilter i = new IntentFilter();
|
||||||
|
i.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
|
||||||
|
context.registerReceiver(new BroadcastReceiver() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onReceive(Context c, Intent i) {
|
||||||
|
network_state = checkNetwork();
|
||||||
|
}
|
||||||
|
|
||||||
|
}, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
0
android/bootstrap/build/src/main/jniLibs/.gitkeep
Normal file
0
android/bootstrap/build/src/main/jniLibs/.gitkeep
Normal file
0
android/bootstrap/build/src/main/res/drawable/.gitkeep
Normal file
0
android/bootstrap/build/src/main/res/drawable/.gitkeep
Normal file
93
android/bootstrap/build/templates/AndroidManifest.tmpl.xml
Normal file
93
android/bootstrap/build/templates/AndroidManifest.tmpl.xml
Normal file
|
@ -0,0 +1,93 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
package="{{ args.package }}"
|
||||||
|
android:versionCode="{{ args.numeric_version }}"
|
||||||
|
android:versionName="{{ args.version }}"
|
||||||
|
android:installLocation="auto">
|
||||||
|
|
||||||
|
<supports-screens
|
||||||
|
android:smallScreens="true"
|
||||||
|
android:normalScreens="true"
|
||||||
|
android:largeScreens="true"
|
||||||
|
android:anyDensity="true"
|
||||||
|
{% if args.min_sdk_version >= 9 %}
|
||||||
|
android:xlargeScreens="true"
|
||||||
|
{% endif %}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- Android 2.3.3 -->
|
||||||
|
<uses-sdk android:minSdkVersion="{{ args.min_sdk_version }}" android:targetSdkVersion="{{ android_api }}" />
|
||||||
|
|
||||||
|
<!-- Set permissions -->
|
||||||
|
{% for perm in args.permissions %}
|
||||||
|
{% if '.' in perm %}
|
||||||
|
<uses-permission android:name="{{ perm }}" />
|
||||||
|
{% else %}
|
||||||
|
<uses-permission android:name="android.permission.{{ perm }}" />
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
{% if args.wakelock %}
|
||||||
|
<uses-permission android:name="android.permission.WAKE_LOCK" />
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if args.billing_pubkey %}
|
||||||
|
<uses-permission android:name="com.android.vending.BILLING" />
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<!-- Create a Java class extending SDLActivity and place it in a
|
||||||
|
directory under src matching the package, e.g.
|
||||||
|
src/com/gamemaker/game/MyGame.java
|
||||||
|
|
||||||
|
then replace "SDLActivity" with the name of your class (e.g. "MyGame")
|
||||||
|
in the XML below.
|
||||||
|
|
||||||
|
An example Java class can be found in README-android.txt
|
||||||
|
-->
|
||||||
|
<application android:label="@string/app_name"
|
||||||
|
android:icon="@drawable/icon"
|
||||||
|
android:allowBackup="true"
|
||||||
|
android:theme="@android:style/Theme.NoTitleBar{% if not args.window %}.Fullscreen{% endif %}"
|
||||||
|
android:hardwareAccelerated="true" >
|
||||||
|
|
||||||
|
{% for m in args.meta_data %}
|
||||||
|
<meta-data android:name="{{ m.split('=', 1)[0] }}" android:value="{{ m.split('=', 1)[-1] }}"/>{% endfor %}
|
||||||
|
<meta-data android:name="wakelock" android:value="{% if args.wakelock %}1{% else %}0{% endif %}"/>
|
||||||
|
|
||||||
|
<activity android:name="org.kivy.android.PythonActivity"
|
||||||
|
android:label="@string/app_name"
|
||||||
|
android:configChanges="keyboardHidden|orientation{% if args.min_sdk_version >= 13 %}|screenSize{% endif %}"
|
||||||
|
>
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.MAIN" />
|
||||||
|
<category android:name="android.intent.category.LAUNCHER" />
|
||||||
|
</intent-filter>
|
||||||
|
{%- if args.intent_filters -%}
|
||||||
|
{{- args.intent_filters -}}
|
||||||
|
{%- endif -%}
|
||||||
|
</activity>
|
||||||
|
|
||||||
|
{% if service %}
|
||||||
|
<service android:name="org.kivy.android.PythonService"
|
||||||
|
android:process=":pythonservice" />
|
||||||
|
{% endif %}
|
||||||
|
{% for name in service_names %}
|
||||||
|
<service android:name="{{ args.package }}.Service{{ name|capitalize }}"
|
||||||
|
android:process=":service_{{ name }}" />
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
{% if args.billing_pubkey %}
|
||||||
|
<service android:name="org.kivy.android.billing.BillingReceiver"
|
||||||
|
android:process=":pythonbilling" />
|
||||||
|
<receiver android:name="org.kivy.android.billing.BillingReceiver"
|
||||||
|
android:process=":pythonbillingreceiver">
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="com.android.vending.billing.IN_APP_NOTIFY" />
|
||||||
|
<action android:name="com.android.vending.billing.RESPONSE_CODE" />
|
||||||
|
<action android:name="com.android.vending.billing.PURCHASE_STATE_CHANGED" />
|
||||||
|
</intent-filter>
|
||||||
|
</receiver>
|
||||||
|
{% endif %}
|
||||||
|
</application>
|
||||||
|
|
||||||
|
</manifest>
|
74
android/bootstrap/build/templates/Service.tmpl.java
Normal file
74
android/bootstrap/build/templates/Service.tmpl.java
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
package {{ args.package }};
|
||||||
|
|
||||||
|
import android.os.Binder;
|
||||||
|
import android.os.IBinder;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.Context;
|
||||||
|
import org.kivy.android.PythonService;
|
||||||
|
|
||||||
|
public class Service{{ name|capitalize }} extends PythonService {
|
||||||
|
/**
|
||||||
|
* Binder given to clients
|
||||||
|
*/
|
||||||
|
private final IBinder mBinder = new Service{{ name|capitalize }}Binder();
|
||||||
|
|
||||||
|
{% if sticky %}
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int getStartType() {
|
||||||
|
return START_STICKY;
|
||||||
|
}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if foreground %}
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean getStartForeground() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
public static void start(Context ctx, String pythonServiceArgument) {
|
||||||
|
String argument = ctx.getFilesDir().getAbsolutePath() + "/app";
|
||||||
|
Intent intent = new Intent(ctx, Service{{ name|capitalize }}.class);
|
||||||
|
intent.putExtra("androidPrivate", argument);
|
||||||
|
intent.putExtra("androidArgument", argument);
|
||||||
|
intent.putExtra("serviceEntrypoint", "{{ entrypoint }}");
|
||||||
|
intent.putExtra("serviceTitle", "{{ name|capitalize }}");
|
||||||
|
intent.putExtra("serviceDescription", "");
|
||||||
|
intent.putExtra("pythonName", "{{ name }}");
|
||||||
|
intent.putExtra("pythonHome", argument);
|
||||||
|
intent.putExtra("androidUnpack", argument);
|
||||||
|
intent.putExtra("pythonPath", argument + ":" + argument + "/lib");
|
||||||
|
intent.putExtra("pythonServiceArgument", pythonServiceArgument);
|
||||||
|
ctx.startService(intent);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void stop(Context ctx) {
|
||||||
|
Intent intent = new Intent(ctx, Service{{ name|capitalize }}.class);
|
||||||
|
ctx.stopService(intent);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public IBinder onBind(Intent intent) {
|
||||||
|
return mBinder;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class used for the client Binder. Because we know this service always
|
||||||
|
* runs in the same process as its clients, we don't need to deal with IPC.
|
||||||
|
*/
|
||||||
|
public class Service{{ name|capitalize }}Binder extends Binder {
|
||||||
|
Service{{ name|capitalize }} getService() {
|
||||||
|
// Return this instance of Service{{ name|capitalize }} so clients can call public methods
|
||||||
|
return Service{{ name|capitalize }}.this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
5
android/bootstrap/build/templates/strings.tmpl.xml
Normal file
5
android/bootstrap/build/templates/strings.tmpl.xml
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<resources>
|
||||||
|
<string name="app_name">{{ args.name }}</string>
|
||||||
|
<string name="private_version">{{ private_version }}</string>
|
||||||
|
</resources>
|
68
android/service/main.py
Normal file
68
android/service/main.py
Normal file
|
@ -0,0 +1,68 @@
|
||||||
|
from lbrynet.extras.daemon.Daemon import Daemon
|
||||||
|
from jnius import PythonJavaClass, autoclass, java_method
|
||||||
|
JAVA_NAMESPACE = 'org.kivy.android'
|
||||||
|
JNI_NAMESPACE = 'org/kivy/android'
|
||||||
|
|
||||||
|
_activity = autoclass(JAVA_NAMESPACE + '.PythonActivity').mActivity
|
||||||
|
|
||||||
|
_callbacks = {
|
||||||
|
'on_new_intent': [],
|
||||||
|
'on_activity_result': [],
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class NewIntentListener(PythonJavaClass):
|
||||||
|
__javainterfaces__ = [JNI_NAMESPACE + '/PythonActivity$NewIntentListener']
|
||||||
|
__javacontext__ = 'app'
|
||||||
|
|
||||||
|
def __init__(self, callback, **kwargs):
|
||||||
|
super(NewIntentListener, self).__init__(**kwargs)
|
||||||
|
self.callback = callback
|
||||||
|
|
||||||
|
@java_method('(Landroid/content/Intent;)V')
|
||||||
|
def onNewIntent(self, intent):
|
||||||
|
self.callback(intent)
|
||||||
|
|
||||||
|
|
||||||
|
class ActivityResultListener(PythonJavaClass):
|
||||||
|
__javainterfaces__ = [JNI_NAMESPACE + '/PythonActivity$ActivityResultListener']
|
||||||
|
__javacontext__ = 'app'
|
||||||
|
|
||||||
|
def __init__(self, callback):
|
||||||
|
super(ActivityResultListener, self).__init__()
|
||||||
|
self.callback = callback
|
||||||
|
|
||||||
|
@java_method('(IILandroid/content/Intent;)V')
|
||||||
|
def onActivityResult(self, requestCode, resultCode, intent):
|
||||||
|
self.callback(requestCode, resultCode, intent)
|
||||||
|
|
||||||
|
|
||||||
|
def bind(**kwargs):
|
||||||
|
for event, callback in kwargs.items():
|
||||||
|
if event not in _callbacks:
|
||||||
|
raise Exception('Unknown {!r} event'.format(event))
|
||||||
|
elif event == 'on_new_intent':
|
||||||
|
listener = NewIntentListener(callback)
|
||||||
|
_activity.registerNewIntentListener(listener)
|
||||||
|
_callbacks[event].append(listener)
|
||||||
|
elif event == 'on_activity_result':
|
||||||
|
listener = ActivityResultListener(callback)
|
||||||
|
_activity.registerActivityResultListener(listener)
|
||||||
|
_callbacks[event].append(listener)
|
||||||
|
|
||||||
|
|
||||||
|
def unbind(**kwargs):
|
||||||
|
for event, callback in kwargs.items():
|
||||||
|
if event not in _callbacks:
|
||||||
|
raise Exception('Unknown {!r} event'.format(event))
|
||||||
|
else:
|
||||||
|
for listener in _callbacks[event][:]:
|
||||||
|
if listener.callback == callback:
|
||||||
|
_callbacks[event].remove(listener)
|
||||||
|
if event == 'on_new_intent':
|
||||||
|
_activity.unregisterNewIntentListener(listener)
|
||||||
|
elif event == 'on_activity_result':
|
||||||
|
_activity.unregisterActivityResultListener(listener)
|
||||||
|
|
||||||
|
|
||||||
|
print('YEAH! Done all the things!!! WOOOHOOO!!!!')
|
49
android/setup.py
Normal file
49
android/setup.py
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
from os.path import join, dirname, abspath
|
||||||
|
from pythonforandroid.toolchain import Bootstrap
|
||||||
|
from setuptools import setup, find_packages
|
||||||
|
from bootstrap import LBRYServiceBootstrap
|
||||||
|
|
||||||
|
|
||||||
|
Bootstrap.bootstraps = {
|
||||||
|
'lbry-service': LBRYServiceBootstrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
setup(
|
||||||
|
name='lbryservice',
|
||||||
|
version='0.1',
|
||||||
|
author="LBRY Inc.",
|
||||||
|
author_email="hello@lbry.io",
|
||||||
|
url="https://lbry.io",
|
||||||
|
description="Android Service for LBRY Network.",
|
||||||
|
license='MIT',
|
||||||
|
python_requires='>=3.7',
|
||||||
|
packages=find_packages(),
|
||||||
|
package_data={'service': ['*.py']},
|
||||||
|
options={
|
||||||
|
'apk': {
|
||||||
|
'dist_name': 'lbry-service',
|
||||||
|
'bootstrap': 'lbry-service',
|
||||||
|
'package': 'io.lbry.service',
|
||||||
|
'permissions': ['INTERNET'],
|
||||||
|
'requirements': ','.join([
|
||||||
|
# needed by aiohttp
|
||||||
|
'multidict', 'yarl', 'async_timeout', 'chardet',
|
||||||
|
# minimum needed by torba:
|
||||||
|
'aiohttp', 'coincurve', 'pbkdf2', 'cryptography', 'attrs',
|
||||||
|
abspath(join(dirname(__file__), '..', '..', 'torba')),
|
||||||
|
# minimum needed by lbrynet
|
||||||
|
'aioupnp', 'appdirs', 'distro', 'base58', 'jsonrpc', 'protobuf',
|
||||||
|
'msgpack', 'jsonschema', 'ecdsa', 'pyyaml', 'docopt',
|
||||||
|
abspath(join(dirname(__file__), '..')),
|
||||||
|
'genericndkbuild', 'pyjnius', 'sqlite3', 'python3'
|
||||||
|
]),
|
||||||
|
'android-api': '26',
|
||||||
|
'ndk-api': '21',
|
||||||
|
'ndk-version': 'r17c',
|
||||||
|
'arch': 'armeabi-v7a',
|
||||||
|
'sdk-dir': '/home/lex/projects/android',
|
||||||
|
'ndk-dir': '/home/lex/projects/android/android-ndk-r17c/'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
1
setup.py
1
setup.py
|
@ -37,6 +37,5 @@ setup(
|
||||||
'torba',
|
'torba',
|
||||||
'pyyaml',
|
'pyyaml',
|
||||||
'docopt',
|
'docopt',
|
||||||
'colorama==0.3.7',
|
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
Loading…
Add table
Reference in a new issue