All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.xiaopy.python.android.AndroidPlatform Maven / Gradle / Ivy

There is a newer version: 12.1.13
Show newest version
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.6.
        "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");
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy