com.xiaopy.python.android.AndroidPlatform Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of chaquopy_java Show documentation
Show all versions of chaquopy_java Show documentation
The Python SDK for Android
package com.xiaopy.python.android;
import android.app.*;
import android.content.*;
import android.content.res.*;
import android.os.*;
import com.xiaopy.python.*;
import java.io.*;
import java.util.*;
import org.jetbrains.annotations.*;
import org.json.*;
/** Platform for Chaquopy on Android. */
public class AndroidPlatform extends Python.Platform {
/** @deprecated Internal use in importer.py and test_android.py. */
public static String ABI;
// TODO: this list could be eliminated if we simply removed all files or directories
// other than AssetFinder and the bootstrap list.
private static final String[] OBSOLETE_FILES = {
// No longer extracted since 0.6.0
"app.zip",
"requirements.zip",
// Renamed back to .zip in 1.1.0
"chaquopy.mp3",
"stdlib.mp3",
// Renamed to bootstrap.zip in 1.3.0
"chaquopy.zip",
// Split into bootstrap-native and stdlib-native/.zip in 1.3.0
"lib-dynload",
// Renamed to stdlib-common.zip in 6.2.2.
"stdlib.zip",
// Renamed to .imy in 8.0.0.
"bootstrap.zip",
"stdlib-common.zip",
// Removed in 12.1.7.
"ticket.txt",
};
private static final String[] OBSOLETE_CACHE = {
// Moved from cache to files dir in 6.3.0
"AssetFinder"
};
/** @deprecated Internal use in chaquopy_java.pyx. */
public Application mContext;
private SharedPreferences sp;
private JSONObject buildJson;
private AssetManager am;
/** Uses the {@link android.app.Application} context of the given context to initialize
* Python. */
public AndroidPlatform(@NotNull Context context) {
mContext = (Application) context.getApplicationContext();
sp = mContext.getSharedPreferences(Common.ASSET_DIR, Context.MODE_PRIVATE);
am = mContext.getAssets();
// TODO: this complexity is unnecessary if the only ABI we can actually use is
// Build.CPU_ABI, which is the ABI of the current process
// (https://stackoverflow.com/a/53158339). Verify this is true across all API
// levels, and then replace all references to AndroidPlatform.ABI with
// Build.CPU_ABI.
List supportedAbis = new ArrayList<>(); // In order of preference.
if (Build.VERSION.SDK_INT >= 21) {
Collections.addAll(supportedAbis, Build.SUPPORTED_ABIS);
} else {
Collections.addAll(supportedAbis, Build.CPU_ABI, Build.CPU_ABI2);
}
for (String abi : supportedAbis) {
try {
am.open(Common.ASSET_DIR + "/" + Common.assetZip(Common.ASSET_STDLIB, abi));
ABI = abi;
break;
} catch (IOException ignored) {}
}
if (ABI == null) {
throw new RuntimeException("None of this device's ABIs " + supportedAbis +
" are supported by this app.");
}
}
/** Returns the Application context of the context which was passed to the contructor. */
public @NotNull Application getApplication() {
return mContext;
}
@Override
public @NotNull String getPath() {
// These assets will be extracted to separate files and used as the initial PYTHONPATH.
String path = "";
String assetDir = mContext.getFilesDir() + "/" + Common.ASSET_DIR;
List bootstrapAssets = new ArrayList<>(Arrays.asList(
Common.assetZip(Common.ASSET_STDLIB, Common.ABI_COMMON),
Common.assetZip(Common.ASSET_BOOTSTRAP),
Common.ASSET_BOOTSTRAP_NATIVE + "/" + ABI));
for (int i = 0; i < bootstrapAssets.size(); i++) {
path += assetDir + "/" + bootstrapAssets.get(i);
if (i < bootstrapAssets.size() - 1) {
path += ":";
}
}
// Now add some non-Python assets which also need to be pre-extracted.
Collections.addAll(bootstrapAssets, Common.ASSET_CACERT);
try {
deleteObsolete(mContext.getFilesDir(), OBSOLETE_FILES);
deleteObsolete(mContext.getCacheDir(), OBSOLETE_CACHE);
extractAssets(bootstrapAssets);
loadNativeLibs();
} catch (IOException | JSONException e) {
throw new RuntimeException(e);
}
return path;
}
private void deleteObsolete(File baseDir, String[] filenames) {
for (String filename : filenames) {
filename = filename.replace("", ABI);
deleteRecursive(new File(baseDir, Common.ASSET_DIR + "/" + filename));
}
}
@Override
public void onStart(@NotNull Python py) {
// These assets will be added to the start of sys.path using AssetFinder paths,
// so their content will be extracted on demand.
String[] appPath = {
Common.ASSET_APP,
Common.ASSET_REQUIREMENTS,
Common.ASSET_STDLIB + "-" + ABI,
};
py.getModule("java.android").callAttr("initialize", mContext, buildJson, appPath);
}
private void extractAssets(List assets) throws IOException, JSONException {
String buildJsonPath = Common.ASSET_DIR + "/" + Common.ASSET_BUILD_JSON;
buildJson = new JSONObject(streamToString(am.open(buildJsonPath)));
JSONObject assetsJson = buildJson.getJSONObject("assets");
// AssetManager.list() is surprisingly slow (20 ms per call on the API 23 emulator), so
// we'll avoid using it.
Set unextracted = new HashSet<>(assets);
Set directories = new HashSet<>();
SharedPreferences.Editor spe = sp.edit();
for (Iterator i = assetsJson.keys(); i.hasNext(); /**/) {
String path = i.next();
for (String ea : assets) {
if (path.equals(ea) || path.startsWith(ea + "/")) {
extractAsset(assetsJson, spe, path);
unextracted.remove(ea);
if (path.startsWith(ea + "/")) {
directories.add(ea);
}
break;
}
}
}
if (! unextracted.isEmpty()) {
throw new RuntimeException("Failed to find assets: " + unextracted);
}
for (String dir : directories) {
cleanExtractedDir(dir, assetsJson);
}
spe.apply();
}
// TODO #5677: multi-process race conditions.
private void extractAsset(JSONObject assetsJson, SharedPreferences.Editor spe,
String path) throws IOException, JSONException {
String fullPath = Common.ASSET_DIR + "/" + path;
File outFile = new File(mContext.getFilesDir(), fullPath);
// See also similar code in importer.py.
String spKey = "asset." + path;
String newHash = assetsJson.getString(path);
if (outFile.exists() && sp.getString(spKey, "").equals(newHash)) {
return;
}
outFile.delete();
File outDir = outFile.getParentFile();
if (!outDir.exists()) {
outDir.mkdirs();
if (!outDir.isDirectory()) {
throw new IOException("Failed to create " + outDir);
}
}
InputStream inStream = am.open(fullPath);
File tmpFile = new File(outDir, outFile.getName() + ".tmp");
tmpFile.delete();
OutputStream outStream = new FileOutputStream(tmpFile);
try {
transferStream(inStream, outStream);
} finally {
outStream.close();
}
if (!tmpFile.renameTo(outFile)) {
throw new IOException("Failed to create " + outFile);
}
spe.putString(spKey, newHash);
}
private void cleanExtractedDir(String dir, JSONObject assetsJson) {
File outDir = new File(mContext.getFilesDir(), Common.ASSET_DIR + "/" + dir);
for (String name : outDir.list()) {
File outFile = new File(outDir, name);
if (outFile.isDirectory()) {
cleanExtractedDir(dir + "/" + name, assetsJson);
} else if (!assetsJson.has(dir + "/" + name)) {
outFile.delete();
}
}
}
private void deleteRecursive(File file) {
File[] children = file.listFiles();
if (children != null) {
for (File child : children) {
deleteRecursive(child);
}
}
file.delete();
}
private void transferStream(InputStream in, OutputStream out) throws IOException {
byte[] buffer = new byte[1024 * 1024];
int len = in.read(buffer);
while (len != -1) {
out.write(buffer, 0, len);
len = in.read(buffer);
}
}
/** This converts all newlines to "\n", and adds a newline at the end of the stream even if
* none was present, but neither of those things should matter for a text file. */
private String streamToString(InputStream in) throws IOException {
BufferedReader reader = new BufferedReader(new InputStreamReader(in));
StringBuilder out = new StringBuilder();
String line;
while ((line = reader.readLine()) != null) {
out.append(line);
out.append("\n");
}
return out.toString();
}
private void loadNativeLibs() throws JSONException {
// Libraries must be loaded in dependency order before API level 18 (#5323). However,
// even if our minimum API level increases to 18 or higher in the future, we should
// still keep pre-loading the OpenSSL and SQLite libraries, because we can't guarantee
// that our lib directory will always be on the LD_LIBRARY_PATH (#5563).
System.loadLibrary("crypto_chaquopy");
System.loadLibrary("ssl_chaquopy");
System.loadLibrary("sqlite3_chaquopy");
System.loadLibrary("python" + buildJson.getString("python_version"));
System.loadLibrary("chaquopy_java");
}
}