provisioner: initial ConfigProvisioner implementation
Core provisioner logic for PawletOS Android 16: - BootReceiver: trigger provisioning on BOOT_COMPLETED and USER_PRESENT - ProvisioningService: run base tasks (APN, secure settings) and OTA tasks (deferred APK installs) as separate passes - VendorConfig: read vendor.cfg from partition; parse APNs, packages, settings - Android.bp: wire up AIDL, disable resource generation (no UI)
This commit is contained in:
@@ -2,7 +2,6 @@ android_app {
|
||||
name: "ConfigProvisioner",
|
||||
srcs: ["src/**/*.java"],
|
||||
manifest: "AndroidManifest.xml",
|
||||
resource_dirs: ["res"],
|
||||
privileged: true,
|
||||
certificate: "platform",
|
||||
optimize: {
|
||||
@@ -12,4 +11,5 @@ android_app {
|
||||
enabled: false,
|
||||
},
|
||||
product_specific: true,
|
||||
sdk_version: "system_current",
|
||||
}
|
||||
@@ -9,6 +9,7 @@
|
||||
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
|
||||
<uses-permission android:name="android.permission.WRITE_SECURE_SETTINGS" />
|
||||
<uses-permission android:name="android.permission.CHANGE_COMPONENT_ENABLED_STATE" />
|
||||
<uses-permission android:name="android.permission.WRITE_APN_SETTINGS" />
|
||||
|
||||
<application
|
||||
android:allowBackup="false"
|
||||
@@ -23,6 +24,7 @@
|
||||
<intent-filter android:priority="1000">
|
||||
<action android:name="android.intent.action.LOCKED_BOOT_COMPLETED" />
|
||||
<action android:name="android.intent.action.BOOT_COMPLETED" />
|
||||
<action android:name="android.intent.action.USER_PRESENT" />
|
||||
</intent-filter>
|
||||
</receiver>
|
||||
|
||||
|
||||
52
README.md
52
README.md
@@ -1,2 +1,52 @@
|
||||
# app_ConfigProvisioner
|
||||
# Config Provisioner
|
||||
|
||||
## Overview
|
||||
Config Provisioner is an Android system service and broadcast receiver designed to automate device provisioning based on vendor-specific configuration. On device boot, it checks for a vendor configuration file (`/vendor/etc/config_provisioner/vendor.cfg`). If provisioning is enabled and a valid APK URL is provided, it downloads and installs a configuration APK, then configures the device setup wizard according to vendor preferences.
|
||||
|
||||
### Key Features
|
||||
- Runs automatically on boot (including locked boot)
|
||||
- Checks for vendor configuration file
|
||||
- Downloads and installs a configuration APK if required
|
||||
- Configures Android Setup Wizard (enables/disables, marks setup complete)
|
||||
- Persists provisioning state to avoid repeated runs
|
||||
|
||||
## Vendor Configuration File Format
|
||||
The vendor configuration file is located at `/vendor/etc/config_provisioner/vendor.cfg`. It is a simple key-value file with one setting per line. Lines starting with `#` are comments and ignored.
|
||||
|
||||
### Supported Keys
|
||||
- `enable_setup_wizard` (boolean: `true`/`false`/`1`/`0`)
|
||||
- Enables or disables the Android Setup Wizard after provisioning.
|
||||
- `enable_provisioning` (boolean: `true`/`false`/`1`/`0`)
|
||||
- Enables or disables the provisioning process.
|
||||
- `vendor_id` (string)
|
||||
- Identifier for the vendor/device.
|
||||
- `network_timeout` (integer, milliseconds)
|
||||
- Timeout for network operations (e.g., APK download).
|
||||
- `config_apk_url` (string, URL)
|
||||
- URL to the configuration APK to be downloaded and installed.
|
||||
|
||||
### Example
|
||||
```
|
||||
# Vendor configuration for Config Provisioner
|
||||
enable_setup_wizard=false
|
||||
enable_provisioning=true
|
||||
vendor_id=acme_corp
|
||||
network_timeout=30000
|
||||
config_apk_url=https://example.com/config/acme_config.apk
|
||||
```
|
||||
|
||||
## Default Values
|
||||
If a key is missing, the following defaults are used:
|
||||
- `enable_setup_wizard`: true
|
||||
- `enable_provisioning`: true
|
||||
- `network_timeout`: 30000
|
||||
- `config_apk_url`: https://default.example.com/config.apk
|
||||
|
||||
## Logging & Debugging
|
||||
The service logs all configuration values and provisioning steps to logcat under the tags `ConfigProvisioner` and `VendorConfig`.
|
||||
|
||||
## Permissions
|
||||
The app requires system-level permissions to install packages, access network state, receive boot events, and modify setup wizard state.
|
||||
|
||||
## License
|
||||
|
||||
|
||||
67
src/dev/oxmc/configprovisioner/BootReceiver.java
Normal file
67
src/dev/oxmc/configprovisioner/BootReceiver.java
Normal file
@@ -0,0 +1,67 @@
|
||||
package dev.oxmc.configprovisioner;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.SharedPreferences;
|
||||
import android.util.Log;
|
||||
|
||||
public class BootReceiver extends BroadcastReceiver {
|
||||
private static final String TAG = "ConfigProvisioner";
|
||||
private static final String PREF_NAME = "config_provisioner_prefs";
|
||||
static final String KEY_BASE_PROVISIONED = "has_base_provisioned";
|
||||
static final String KEY_LAST_OTA_CHECK = "last_ota_check_ms";
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
if (intent == null || intent.getAction() == null) return;
|
||||
String action = intent.getAction();
|
||||
Log.d(TAG, "Received: " + action);
|
||||
|
||||
switch (action) {
|
||||
case Intent.ACTION_BOOT_COMPLETED:
|
||||
case "android.intent.action.LOCKED_BOOT_COMPLETED":
|
||||
if (!isBaseProvisioned(context)) {
|
||||
startService(context, ProvisioningService.ACTION_BASE_PROVISION);
|
||||
}
|
||||
break;
|
||||
|
||||
case Intent.ACTION_USER_PRESENT:
|
||||
// Only run OTA check once base provisioning is complete and a URL is set.
|
||||
if (isBaseProvisioned(context) && VendorConfig.isProvisioningEnabled()
|
||||
&& !VendorConfig.DEFAULT_CONFIG_APK_URL.equals(
|
||||
VendorConfig.getConfigApkUrl())) {
|
||||
startService(context, ProvisioningService.ACTION_OTA_UPDATE);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private static void startService(Context context, String action) {
|
||||
Intent si = new Intent(context, ProvisioningService.class);
|
||||
si.setAction(action);
|
||||
context.startService(si);
|
||||
}
|
||||
|
||||
// ---- SharedPrefs helpers (package-private so ProvisioningService can use them) ----
|
||||
|
||||
static boolean isBaseProvisioned(Context context) {
|
||||
return prefs(context).getBoolean(KEY_BASE_PROVISIONED, false);
|
||||
}
|
||||
|
||||
static void setBaseProvisioned(Context context, boolean value) {
|
||||
prefs(context).edit().putBoolean(KEY_BASE_PROVISIONED, value).apply();
|
||||
}
|
||||
|
||||
static long getLastOtaCheck(Context context) {
|
||||
return prefs(context).getLong(KEY_LAST_OTA_CHECK, 0L);
|
||||
}
|
||||
|
||||
static void setLastOtaCheck(Context context, long timeMs) {
|
||||
prefs(context).edit().putLong(KEY_LAST_OTA_CHECK, timeMs).apply();
|
||||
}
|
||||
|
||||
private static SharedPreferences prefs(Context context) {
|
||||
return context.getSharedPreferences(PREF_NAME, Context.MODE_PRIVATE);
|
||||
}
|
||||
}
|
||||
@@ -1,53 +0,0 @@
|
||||
package dev.oxmc.configprovisioner;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.util.Log;
|
||||
|
||||
public class BootReceiver extends BroadcastReceiver {
|
||||
private static final String TAG = "ConfigProvisioner";
|
||||
private static final String PREF_NAME = "config_provisioner_prefs";
|
||||
private static final String KEY_HAS_PROVISIONED = "has_provisioned";
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
if (intent == null || intent.getAction() == null) {
|
||||
Log.d(TAG, "Null intent or action");
|
||||
return;
|
||||
}
|
||||
|
||||
Log.d(TAG, "Received boot event: " + intent.getAction());
|
||||
|
||||
// Check if we've already provisioned
|
||||
if (hasProvisioned(context)) {
|
||||
Log.d(TAG, "Already provisioned, skipping");
|
||||
return;
|
||||
}
|
||||
|
||||
// Check if vendor config exists - if not, exit gracefully
|
||||
if (!VendorConfig.hasVendorConfig()) {
|
||||
Log.i(TAG, "No vendor config found, provisioning not required");
|
||||
setProvisioned(context, true); // Mark as provisioned to not run again
|
||||
return;
|
||||
}
|
||||
|
||||
// Start provisioning service
|
||||
Intent serviceIntent = new Intent(context, ProvisioningService.class);
|
||||
serviceIntent.setAction(intent.getAction());
|
||||
context.startService(serviceIntent);
|
||||
}
|
||||
|
||||
private boolean hasProvisioned(Context context) {
|
||||
return context.getSharedPreferences(PREF_NAME, Context.MODE_PRIVATE)
|
||||
.getBoolean(KEY_HAS_PROVISIONED, false);
|
||||
}
|
||||
|
||||
public static void setProvisioned(Context context, boolean provisioned) {
|
||||
context.getSharedPreferences(PREF_NAME, Context.MODE_PRIVATE)
|
||||
.edit()
|
||||
.putBoolean(KEY_HAS_PROVISIONED, provisioned)
|
||||
.apply();
|
||||
Log.d(TAG, "Provisioning state set to: " + provisioned);
|
||||
}
|
||||
}
|
||||
@@ -1,205 +1,468 @@
|
||||
package dev.oxmc.configprovisioner;
|
||||
|
||||
import android.app.Service;
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.content.res.AssetManager;
|
||||
import android.content.res.Configuration;
|
||||
import android.content.res.Resources;
|
||||
import android.content.res.XmlResourceParser;
|
||||
import android.database.Cursor;
|
||||
import android.net.Uri;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Build;
|
||||
import android.os.IBinder;
|
||||
import android.provider.Settings;
|
||||
import android.provider.Telephony;
|
||||
import android.util.Log;
|
||||
import android.util.Xml;
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.InputStream;
|
||||
import java.net.HttpURLConnection;
|
||||
import java.net.URL;
|
||||
import org.xmlpull.v1.XmlPullParser;
|
||||
|
||||
public class ProvisioningService extends Service {
|
||||
private static final String TAG = "ConfigProvisioner";
|
||||
private static final String DOWNLOAD_PATH = "/data/local/tmp/config_provision.apk";
|
||||
|
||||
/** Runs on BOOT_COMPLETED: applies built-in config APK settings + wizard state. No network. */
|
||||
public static final String ACTION_BASE_PROVISION = "dev.oxmc.configprovisioner.ACTION_BASE_PROVISION";
|
||||
|
||||
/** Runs on USER_PRESENT: downloads updated config APK from URL and re-applies settings. */
|
||||
public static final String ACTION_OTA_UPDATE = "dev.oxmc.configprovisioner.ACTION_OTA_UPDATE";
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// Service entry point
|
||||
// -------------------------------------------------------------------------
|
||||
|
||||
@Override
|
||||
public int onStartCommand(Intent intent, int flags, int startId) {
|
||||
Log.d(TAG, "Starting provisioning service");
|
||||
|
||||
// Double-check that vendor config exists
|
||||
if (!VendorConfig.hasVendorConfig()) {
|
||||
Log.i(TAG, "No vendor config found, stopping service");
|
||||
BootReceiver.setProvisioned(this, true);
|
||||
stopSelf();
|
||||
String action = intent != null ? intent.getAction() : null;
|
||||
if (ACTION_BASE_PROVISION.equals(action)) {
|
||||
Log.d(TAG, "Starting base provisioning");
|
||||
new BaseProvisionTask().execute();
|
||||
return START_NOT_STICKY;
|
||||
}
|
||||
|
||||
new ProvisioningTask().execute();
|
||||
|
||||
return START_STICKY;
|
||||
if (ACTION_OTA_UPDATE.equals(action)) {
|
||||
Log.d(TAG, "Starting OTA config update check");
|
||||
new OtaUpdateTask().execute();
|
||||
return START_NOT_STICKY;
|
||||
}
|
||||
stopSelf();
|
||||
return START_NOT_STICKY;
|
||||
}
|
||||
|
||||
private class ProvisioningTask extends AsyncTask<Void, Void, Boolean> {
|
||||
@Override
|
||||
public IBinder onBind(Intent intent) { return null; }
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// Base provisioning — runs at boot, no network needed
|
||||
// -------------------------------------------------------------------------
|
||||
|
||||
private class BaseProvisionTask extends AsyncTask<Void, Void, Boolean> {
|
||||
@Override
|
||||
protected Boolean doInBackground(Void... voids) {
|
||||
Log.i(TAG, "Starting provisioning process");
|
||||
|
||||
// Log all config values for debugging
|
||||
protected Boolean doInBackground(Void... v) {
|
||||
if (!VendorConfig.hasVendorConfig()) {
|
||||
Log.i(TAG, "No vendor config, skipping base provisioning");
|
||||
return true;
|
||||
}
|
||||
VendorConfig.logConfigValues();
|
||||
|
||||
// Check if provisioning is enabled
|
||||
if (!VendorConfig.isProvisioningEnabled()) {
|
||||
Log.i(TAG, "Provisioning disabled by vendor config");
|
||||
configureSetupWizard();
|
||||
return true;
|
||||
}
|
||||
|
||||
// Get config URL
|
||||
String configUrl = VendorConfig.getConfigApkUrl();
|
||||
if (configUrl == null || configUrl.isEmpty() || configUrl.equals(VendorConfig.DEFAULT_CONFIG_APK_URL)) {
|
||||
Log.e(TAG, "No valid config URL configured, skipping provisioning");
|
||||
configureSetupWizard();
|
||||
return true;
|
||||
}
|
||||
|
||||
try {
|
||||
Log.d(TAG, "Downloading config from: " + configUrl);
|
||||
|
||||
if (downloadApk(configUrl, DOWNLOAD_PATH)) {
|
||||
Log.i(TAG, "Download successful, installing APK");
|
||||
|
||||
if (installApk(DOWNLOAD_PATH)) {
|
||||
Log.i(TAG, "Installation successful");
|
||||
configureSetupWizard();
|
||||
return true;
|
||||
} else {
|
||||
Log.e(TAG, "Installation failed");
|
||||
}
|
||||
} else {
|
||||
Log.e(TAG, "Download failed");
|
||||
}
|
||||
} catch (Exception e) {
|
||||
Log.e(TAG, "Provisioning failed with error", e);
|
||||
} finally {
|
||||
// Clean up downloaded file
|
||||
File downloadedFile = new File(DOWNLOAD_PATH);
|
||||
if (downloadedFile.exists()) {
|
||||
if (downloadedFile.delete()) {
|
||||
Log.d(TAG, "Cleaned up downloaded file");
|
||||
} else {
|
||||
Log.w(TAG, "Failed to clean up downloaded file");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
// Apply settings and APNs from the pre-installed config APK (built into the ROM).
|
||||
// No download needed — the APK is already present as a system app.
|
||||
applyConfigApkSettings();
|
||||
applyApns();
|
||||
configureSetupWizard();
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean success) {
|
||||
if (success) {
|
||||
Log.i(TAG, "Provisioning completed successfully");
|
||||
BootReceiver.setProvisioned(ProvisioningService.this, true);
|
||||
} else {
|
||||
Log.e(TAG, "Provisioning failed");
|
||||
// Don't mark as provisioned on failure, so it retries on next boot
|
||||
}
|
||||
BootReceiver.setBaseProvisioned(ProvisioningService.this, true);
|
||||
Log.i(TAG, "Base provisioning complete");
|
||||
stopSelf();
|
||||
}
|
||||
|
||||
private boolean downloadApk(String urlString, String outputPath) {
|
||||
HttpURLConnection connection = null;
|
||||
}
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// OTA update — runs on USER_PRESENT, deferred well after boot
|
||||
// -------------------------------------------------------------------------
|
||||
|
||||
private class OtaUpdateTask extends AsyncTask<Void, Void, Boolean> {
|
||||
@Override
|
||||
protected Boolean doInBackground(Void... v) {
|
||||
if (!VendorConfig.hasVendorConfig() || !VendorConfig.isProvisioningEnabled()) {
|
||||
return false;
|
||||
}
|
||||
String url = VendorConfig.getConfigApkUrl();
|
||||
if (url == null || url.isEmpty() || url.equals(VendorConfig.DEFAULT_CONFIG_APK_URL)) {
|
||||
Log.d(TAG, "No OTA URL configured, skipping");
|
||||
return false;
|
||||
}
|
||||
|
||||
long now = System.currentTimeMillis();
|
||||
long lastCheck = BootReceiver.getLastOtaCheck(ProvisioningService.this);
|
||||
long interval = VendorConfig.getOtaCheckIntervalMs();
|
||||
if (now - lastCheck < interval) {
|
||||
long minutesLeft = (interval - (now - lastCheck)) / 60_000;
|
||||
Log.d(TAG, "OTA check skipped — next check in " + minutesLeft + " min");
|
||||
return false;
|
||||
}
|
||||
|
||||
// Record attempt time before the download so a broken server doesn't
|
||||
// cause a retry on every subsequent screen unlock.
|
||||
BootReceiver.setLastOtaCheck(ProvisioningService.this, now);
|
||||
|
||||
try {
|
||||
URL url = new URL(urlString);
|
||||
connection = (HttpURLConnection) url.openConnection();
|
||||
connection.setConnectTimeout(VendorConfig.getNetworkTimeout());
|
||||
connection.setReadTimeout(VendorConfig.getNetworkTimeout());
|
||||
connection.setRequestProperty("User-Agent", "ConfigProvisioner/1.0");
|
||||
connection.connect();
|
||||
|
||||
int responseCode = connection.getResponseCode();
|
||||
if (responseCode != HttpURLConnection.HTTP_OK) {
|
||||
Log.e(TAG, "Server returned HTTP " + responseCode);
|
||||
return false;
|
||||
}
|
||||
|
||||
try (InputStream input = connection.getInputStream();
|
||||
FileOutputStream output = new FileOutputStream(outputPath)) {
|
||||
|
||||
byte[] buffer = new byte[8192];
|
||||
int bytesRead;
|
||||
long totalBytes = 0;
|
||||
|
||||
while ((bytesRead = input.read(buffer)) != -1) {
|
||||
output.write(buffer, 0, bytesRead);
|
||||
totalBytes += bytesRead;
|
||||
}
|
||||
|
||||
Log.d(TAG, "Downloaded " + totalBytes + " bytes to " + outputPath);
|
||||
return true;
|
||||
}
|
||||
Log.i(TAG, "Downloading config update from: " + url);
|
||||
if (!downloadApk(url, DOWNLOAD_PATH)) return false;
|
||||
if (!installApk(DOWNLOAD_PATH)) return false;
|
||||
Log.i(TAG, "OTA config APK installed, re-applying settings");
|
||||
applyConfigApkSettings();
|
||||
applyApns();
|
||||
return true;
|
||||
} catch (Exception e) {
|
||||
Log.e(TAG, "Download failed", e);
|
||||
Log.e(TAG, "OTA update failed", e);
|
||||
return false;
|
||||
} finally {
|
||||
if (connection != null) {
|
||||
connection.disconnect();
|
||||
}
|
||||
new File(DOWNLOAD_PATH).delete();
|
||||
}
|
||||
}
|
||||
|
||||
private boolean installApk(String apkPath) {
|
||||
try {
|
||||
Process process = Runtime.getRuntime().exec(
|
||||
new String[]{"pm", "install", "-r", "--user", "0", apkPath}
|
||||
);
|
||||
|
||||
int exitCode = process.waitFor();
|
||||
if (exitCode == 0) {
|
||||
Log.i(TAG, "APK installed successfully via pm install");
|
||||
return true;
|
||||
} else {
|
||||
Log.e(TAG, "pm install failed with exit code: " + exitCode);
|
||||
return false;
|
||||
}
|
||||
} catch (Exception e) {
|
||||
Log.e(TAG, "Installation failed", e);
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean updated) {
|
||||
if (updated) Log.i(TAG, "OTA config update applied successfully");
|
||||
stopSelf();
|
||||
}
|
||||
}
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// Network helpers
|
||||
// -------------------------------------------------------------------------
|
||||
|
||||
private boolean downloadApk(String urlString, String outputPath) {
|
||||
HttpURLConnection connection = null;
|
||||
try {
|
||||
URL url = new URL(urlString);
|
||||
connection = (HttpURLConnection) url.openConnection();
|
||||
connection.setConnectTimeout(VendorConfig.getNetworkTimeout());
|
||||
connection.setReadTimeout(VendorConfig.getNetworkTimeout());
|
||||
connection.setRequestProperty("User-Agent", "ConfigProvisioner/1.0");
|
||||
connection.connect();
|
||||
|
||||
int code = connection.getResponseCode();
|
||||
if (code != HttpURLConnection.HTTP_OK) {
|
||||
Log.e(TAG, "Server returned HTTP " + code);
|
||||
return false;
|
||||
}
|
||||
|
||||
try (InputStream in = connection.getInputStream();
|
||||
FileOutputStream out = new FileOutputStream(outputPath)) {
|
||||
byte[] buf = new byte[8192];
|
||||
int n;
|
||||
long total = 0;
|
||||
while ((n = in.read(buf)) != -1) { out.write(buf, 0, n); total += n; }
|
||||
Log.d(TAG, "Downloaded " + total + " bytes");
|
||||
return total > 0;
|
||||
}
|
||||
} catch (Exception e) {
|
||||
Log.e(TAG, "Download failed", e);
|
||||
return false;
|
||||
} finally {
|
||||
if (connection != null) connection.disconnect();
|
||||
}
|
||||
|
||||
private void configureSetupWizard() {
|
||||
boolean enableWizard = VendorConfig.isSetupWizardEnabled();
|
||||
Log.i(TAG, "Configuring Setup Wizard: " + (enableWizard ? "ENABLED" : "DISABLED"));
|
||||
|
||||
if (!enableWizard) {
|
||||
disablePackage("com.android.setupwizard");
|
||||
disablePackage("com.google.android.setupwizard");
|
||||
disablePackage("org.lineageos.setupwizard");
|
||||
|
||||
// Mark setup as complete
|
||||
try {
|
||||
Settings.Secure.putInt(getContentResolver(),
|
||||
Settings.Secure.USER_SETUP_COMPLETE, 1);
|
||||
Settings.Global.putInt(getContentResolver(),
|
||||
Settings.Global.DEVICE_PROVISIONED, 1);
|
||||
Log.d(TAG, "Setup marked as complete");
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to set setup complete flags", e);
|
||||
}
|
||||
|
||||
private boolean installApk(String apkPath) {
|
||||
try {
|
||||
Process p = Runtime.getRuntime().exec(
|
||||
new String[]{"pm", "install", "-r", "--user", "0", apkPath});
|
||||
int exit = p.waitFor();
|
||||
if (exit == 0) { Log.i(TAG, "APK installed via pm install"); return true; }
|
||||
Log.e(TAG, "pm install failed (exit " + exit + ")");
|
||||
return false;
|
||||
} catch (Exception e) {
|
||||
Log.e(TAG, "Installation failed", e);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// Settings application
|
||||
// -------------------------------------------------------------------------
|
||||
|
||||
private void applyConfigApkSettings() {
|
||||
String pkg = VendorConfig.getConfigApkPackage();
|
||||
if (pkg == null || pkg.isEmpty()) return;
|
||||
try {
|
||||
Context ctx = createPackageContext(pkg, Context.CONTEXT_IGNORE_SECURITY);
|
||||
Resources res = ctx.getResources();
|
||||
int xmlId = res.getIdentifier("settings", "xml", pkg);
|
||||
if (xmlId == 0) { Log.d(TAG, "No settings.xml in " + pkg); return; }
|
||||
|
||||
XmlResourceParser xml = res.getXml(xmlId);
|
||||
int event, skipDepth = 0;
|
||||
while ((event = xml.next()) != XmlResourceParser.END_DOCUMENT) {
|
||||
if (event == XmlResourceParser.START_TAG) {
|
||||
if (skipDepth > 0) { skipDepth++; continue; }
|
||||
String tag = xml.getName();
|
||||
if ("if".equals(tag)) { if (!evaluateCondition(xml)) skipDepth = 1; continue; }
|
||||
String name = xml.getAttributeValue(null, "name");
|
||||
String value = xml.getAttributeValue(null, "value");
|
||||
if (name == null || value == null) continue;
|
||||
try {
|
||||
switch (tag) {
|
||||
case "secure": Settings.Secure.putString(getContentResolver(), name, value); break;
|
||||
case "system": Settings.System.putString(getContentResolver(), name, value); break;
|
||||
case "global": Settings.Global.putString(getContentResolver(), name, value); break;
|
||||
default: Log.w(TAG, "Unknown settings tag: " + tag);
|
||||
}
|
||||
Log.d(TAG, tag + "." + name + " = " + value);
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to apply " + tag + "." + name, e);
|
||||
}
|
||||
} else if (event == XmlResourceParser.END_TAG) {
|
||||
if (skipDepth > 0) skipDepth--;
|
||||
}
|
||||
}
|
||||
xml.close();
|
||||
Log.i(TAG, "Settings from " + pkg + " applied");
|
||||
} catch (PackageManager.NameNotFoundException e) {
|
||||
Log.w(TAG, "Config APK not found: " + pkg + " (not yet installed?)");
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to apply config APK settings", e);
|
||||
}
|
||||
|
||||
private void disablePackage(String packageName) {
|
||||
}
|
||||
|
||||
private void applyApns() {
|
||||
if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_TELEPHONY)) return;
|
||||
String pkg = VendorConfig.getConfigApkPackage();
|
||||
if (pkg == null || pkg.isEmpty()) return;
|
||||
try {
|
||||
Context ctx = createPackageContext(pkg, Context.CONTEXT_IGNORE_SECURITY);
|
||||
int[] counts = {0, 0};
|
||||
|
||||
AssetManager assets = ctx.getAssets();
|
||||
String[] topLevel = assets.list("apns");
|
||||
if (topLevel != null && topLevel.length > 0) {
|
||||
walkApnAssets(assets, "apns", counts);
|
||||
Log.i(TAG, "APNs (assets/apns/): " + counts[0] + " inserted, " + counts[1] + " skipped");
|
||||
return;
|
||||
}
|
||||
|
||||
Resources res = ctx.getResources();
|
||||
int xmlId = res.getIdentifier("apns", "xml", pkg);
|
||||
if (xmlId == 0) { Log.d(TAG, "No APN config in " + pkg); return; }
|
||||
XmlResourceParser xml = res.getXml(xmlId);
|
||||
parseAndInsertApns(xml, "res/xml/apns.xml", counts);
|
||||
xml.close();
|
||||
Log.i(TAG, "APNs (res/xml/apns.xml): " + counts[0] + " inserted, " + counts[1] + " skipped");
|
||||
} catch (PackageManager.NameNotFoundException e) {
|
||||
Log.w(TAG, "Config APK not found for APN provisioning: " + pkg);
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "APN provisioning failed", e);
|
||||
}
|
||||
}
|
||||
|
||||
private void walkApnAssets(AssetManager assets, String path, int[] counts) {
|
||||
try {
|
||||
String[] entries = assets.list(path);
|
||||
if (entries == null) return;
|
||||
for (String entry : entries) {
|
||||
String full = path + "/" + entry;
|
||||
String[] children = assets.list(full);
|
||||
if (children != null && children.length > 0) {
|
||||
walkApnAssets(assets, full, counts);
|
||||
} else if (entry.endsWith(".xml")) {
|
||||
try (InputStream is = assets.open(full)) {
|
||||
XmlPullParser parser = Xml.newPullParser();
|
||||
parser.setInput(is, "UTF-8");
|
||||
parseAndInsertApns(parser, full, counts);
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to parse APN file: " + full, e);
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to enumerate assets/" + path, e);
|
||||
}
|
||||
}
|
||||
|
||||
private void parseAndInsertApns(XmlPullParser parser, String source, int[] counts)
|
||||
throws Exception {
|
||||
int event;
|
||||
while ((event = parser.next()) != XmlPullParser.END_DOCUMENT) {
|
||||
if (event != XmlPullParser.START_TAG || !"apn".equals(parser.getName())) continue;
|
||||
|
||||
String name = parser.getAttributeValue(null, "name");
|
||||
if (name == null) name = parser.getAttributeValue(null, "carrier");
|
||||
String mcc = parser.getAttributeValue(null, "mcc");
|
||||
String mnc = parser.getAttributeValue(null, "mnc");
|
||||
String apn = parser.getAttributeValue(null, "apn");
|
||||
String numeric = parser.getAttributeValue(null, "numeric");
|
||||
String mvnoType = parser.getAttributeValue(null, "mvno_type");
|
||||
String mvnoData = parser.getAttributeValue(null, "mvno_match_data");
|
||||
if (numeric == null && mcc != null && mnc != null) numeric = mcc + mnc;
|
||||
if (mvnoType == null) mvnoType = "";
|
||||
if (mvnoData == null) mvnoData = "";
|
||||
|
||||
if (name == null || numeric == null || mcc == null || mnc == null || apn == null) {
|
||||
Log.w(TAG, "Skipping APN with missing fields in " + source);
|
||||
continue;
|
||||
}
|
||||
|
||||
Cursor c = getContentResolver().query(Telephony.Carriers.CONTENT_URI,
|
||||
new String[]{"_id"},
|
||||
"numeric=? AND apn=? AND mvno_type=? AND mvno_match_data=?",
|
||||
new String[]{numeric, apn, mvnoType, mvnoData}, null);
|
||||
boolean exists = c != null && c.getCount() > 0;
|
||||
if (c != null) c.close();
|
||||
if (exists) { counts[1]++; continue; }
|
||||
|
||||
ContentValues cv = new ContentValues();
|
||||
cv.put(Telephony.Carriers.NAME, name);
|
||||
cv.put(Telephony.Carriers.NUMERIC, numeric);
|
||||
cv.put(Telephony.Carriers.MCC, mcc);
|
||||
cv.put(Telephony.Carriers.MNC, mnc);
|
||||
cv.put(Telephony.Carriers.APN, apn);
|
||||
if (!mvnoType.isEmpty()) cv.put(Telephony.Carriers.MVNO_TYPE, mvnoType);
|
||||
if (!mvnoData.isEmpty()) cv.put(Telephony.Carriers.MVNO_MATCH_DATA, mvnoData);
|
||||
|
||||
String type = parser.getAttributeValue(null, "type");
|
||||
cv.put(Telephony.Carriers.TYPE, type != null ? type : "default,supl");
|
||||
String proto = parser.getAttributeValue(null, "protocol");
|
||||
cv.put(Telephony.Carriers.PROTOCOL, proto != null ? proto : "IPV4V6");
|
||||
String roamProto = parser.getAttributeValue(null, "roaming_protocol");
|
||||
cv.put(Telephony.Carriers.ROAMING_PROTOCOL, roamProto != null ? roamProto : "IPV4V6");
|
||||
|
||||
apnPutStr(cv, Telephony.Carriers.SERVER, parser, "server");
|
||||
apnPutStr(cv, Telephony.Carriers.PROXY, parser, "proxy");
|
||||
apnPutStr(cv, Telephony.Carriers.PORT, parser, "port");
|
||||
apnPutStr(cv, Telephony.Carriers.MMSC, parser, "mmsc");
|
||||
apnPutStr(cv, Telephony.Carriers.MMSPROXY, parser, "mmsproxy");
|
||||
apnPutStr(cv, Telephony.Carriers.MMSPORT, parser, "mmsport");
|
||||
apnPutStr(cv, Telephony.Carriers.USER, parser, "user");
|
||||
apnPutStr(cv, Telephony.Carriers.PASSWORD, parser, "password");
|
||||
apnPutStr(cv, "bearer_bitmask", parser, "bearer_bitmask");
|
||||
apnPutInt(cv, Telephony.Carriers.AUTH_TYPE, parser, "authtype", -1);
|
||||
apnPutInt(cv, "profile_id", parser, "profile_id", 0);
|
||||
apnPutInt(cv, "max_conns", parser, "max_conns", 0);
|
||||
apnPutInt(cv, "wait_time", parser, "wait_time", 0);
|
||||
apnPutInt(cv, "max_conns_time", parser, "max_conns_time", 0);
|
||||
apnPutInt(cv, Telephony.Carriers.MTU, parser, "mtu", 0);
|
||||
|
||||
String modemCog = parser.getAttributeValue(null, "modem_cognitive");
|
||||
if (modemCog != null)
|
||||
cv.put("modem_cognitive",
|
||||
"true".equalsIgnoreCase(modemCog) || "1".equals(modemCog) ? 1 : 0);
|
||||
String enabled = parser.getAttributeValue(null, "carrier_enabled");
|
||||
cv.put(Telephony.Carriers.CARRIER_ENABLED,
|
||||
"0".equals(enabled) || "false".equalsIgnoreCase(enabled) ? 0 : 1);
|
||||
|
||||
try {
|
||||
PackageManager pm = getPackageManager();
|
||||
pm.setApplicationEnabledSetting(packageName,
|
||||
PackageManager.COMPONENT_ENABLED_STATE_DISABLED, 0);
|
||||
Log.d(TAG, "Disabled package: " + packageName);
|
||||
getContentResolver().insert(Telephony.Carriers.CONTENT_URI, cv);
|
||||
counts[0]++;
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to disable package: " + packageName, e);
|
||||
Log.w(TAG, "Failed to insert APN: " + name, e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public IBinder onBind(Intent intent) {
|
||||
return null;
|
||||
private static void apnPutStr(ContentValues cv, String col, XmlPullParser p, String attr) {
|
||||
String v = p.getAttributeValue(null, attr);
|
||||
if (v != null && !v.isEmpty()) cv.put(col, v);
|
||||
}
|
||||
}
|
||||
|
||||
private static void apnPutInt(ContentValues cv, String col, XmlPullParser p,
|
||||
String attr, int def) {
|
||||
String v = p.getAttributeValue(null, attr);
|
||||
if (v == null || v.isEmpty()) return;
|
||||
try { cv.put(col, Integer.parseInt(v)); }
|
||||
catch (NumberFormatException e) { if (def >= 0) cv.put(col, def); }
|
||||
}
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// Setup wizard configuration
|
||||
// -------------------------------------------------------------------------
|
||||
|
||||
private void configureSetupWizard() {
|
||||
boolean enable = VendorConfig.isSetupWizardEnabled();
|
||||
Log.i(TAG, "Setup wizard: " + (enable ? "ENABLED" : "DISABLED"));
|
||||
if (enable) {
|
||||
try {
|
||||
Settings.Secure.putInt(getContentResolver(), Settings.Secure.USER_SETUP_COMPLETE, 0);
|
||||
Settings.Global.putInt(getContentResolver(), Settings.Global.DEVICE_PROVISIONED, 0);
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to mark setup incomplete", e);
|
||||
}
|
||||
} else {
|
||||
disablePackage("com.android.setupwizard");
|
||||
disablePackage("com.google.android.setupwizard");
|
||||
disablePackage("org.lineageos.setupwizard");
|
||||
try {
|
||||
Settings.Secure.putInt(getContentResolver(), Settings.Secure.USER_SETUP_COMPLETE, 1);
|
||||
Settings.Global.putInt(getContentResolver(), Settings.Global.DEVICE_PROVISIONED, 1);
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Failed to mark setup complete", e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void disablePackage(String pkg) {
|
||||
try {
|
||||
getPackageManager().setApplicationEnabledSetting(
|
||||
pkg, PackageManager.COMPONENT_ENABLED_STATE_DISABLED, 0);
|
||||
Log.d(TAG, "Disabled: " + pkg);
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, "Could not disable " + pkg + ": " + e.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// <if> condition evaluation
|
||||
// -------------------------------------------------------------------------
|
||||
|
||||
private boolean evaluateCondition(XmlResourceParser xml) {
|
||||
String manufacturer = xml.getAttributeValue(null, "manufacturer");
|
||||
String brand = xml.getAttributeValue(null, "brand");
|
||||
String model = xml.getAttributeValue(null, "model");
|
||||
String sdkStr = xml.getAttributeValue(null, "sdk");
|
||||
String sdkMinStr = xml.getAttributeValue(null, "sdk_min");
|
||||
String sdkMaxStr = xml.getAttributeValue(null, "sdk_max");
|
||||
String formFactor = xml.getAttributeValue(null, "form_factor");
|
||||
String feature = xml.getAttributeValue(null, "feature");
|
||||
|
||||
if (manufacturer != null && !Build.MANUFACTURER.equalsIgnoreCase(manufacturer)) return false;
|
||||
if (brand != null && !Build.BRAND.equalsIgnoreCase(brand)) return false;
|
||||
if (model != null && !Build.MODEL.toLowerCase().contains(model.toLowerCase())) return false;
|
||||
|
||||
if (sdkStr != null) { try { if (Build.VERSION.SDK_INT != Integer.parseInt(sdkStr)) return false; } catch (NumberFormatException ignored) {} }
|
||||
if (sdkMinStr != null) { try { if (Build.VERSION.SDK_INT < Integer.parseInt(sdkMinStr)) return false; } catch (NumberFormatException ignored) {} }
|
||||
if (sdkMaxStr != null) { try { if (Build.VERSION.SDK_INT > Integer.parseInt(sdkMaxStr)) return false; } catch (NumberFormatException ignored) {} }
|
||||
|
||||
if (formFactor != null) {
|
||||
switch (formFactor) {
|
||||
case "tablet": if (!isTablet()) return false; break;
|
||||
case "phone": if (isTablet()) return false; break;
|
||||
case "flip":
|
||||
if (!getPackageManager().hasSystemFeature("android.hardware.sensor.hinge_angle")) return false;
|
||||
break;
|
||||
case "tv":
|
||||
if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_LEANBACK)) return false;
|
||||
break;
|
||||
default: Log.w(TAG, "Unknown form_factor: " + formFactor);
|
||||
}
|
||||
}
|
||||
if (feature != null && !getPackageManager().hasSystemFeature(feature)) return false;
|
||||
return true;
|
||||
}
|
||||
|
||||
private boolean isTablet() {
|
||||
int layout = getResources().getConfiguration().screenLayout
|
||||
& Configuration.SCREENLAYOUT_SIZE_MASK;
|
||||
return layout >= Configuration.SCREENLAYOUT_SIZE_LARGE;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -8,118 +8,106 @@ import java.io.IOException;
|
||||
|
||||
public class VendorConfig {
|
||||
private static final String TAG = "VendorConfig";
|
||||
|
||||
// Configuration path
|
||||
|
||||
public static final String VENDOR_CONFIG_PATH = "/vendor/etc/config_provisioner/vendor.cfg";
|
||||
|
||||
// Configuration keys
|
||||
private static final String KEY_ENABLE_SETUP_WIZARD = "enable_setup_wizard";
|
||||
private static final String KEY_ENABLE_PROVISIONING = "enable_provisioning";
|
||||
private static final String KEY_VENDOR_ID = "vendor_id";
|
||||
private static final String KEY_NETWORK_TIMEOUT = "network_timeout";
|
||||
private static final String KEY_CONFIG_APK_URL = "config_apk_url";
|
||||
|
||||
// Default values
|
||||
public static final boolean DEFAULT_ENABLE_SETUP_WIZARD = true;
|
||||
public static final boolean DEFAULT_ENABLE_PROVISIONING = true;
|
||||
public static final int DEFAULT_NETWORK_TIMEOUT = 30000;
|
||||
public static final String DEFAULT_CONFIG_APK_URL = "https://default.example.com/config.apk";
|
||||
|
||||
|
||||
private static final String KEY_ENABLE_SETUP_WIZARD = "enable_setup_wizard";
|
||||
private static final String KEY_ENABLE_PROVISIONING = "enable_provisioning";
|
||||
private static final String KEY_VENDOR_ID = "vendor_id";
|
||||
private static final String KEY_NETWORK_TIMEOUT = "network_timeout";
|
||||
private static final String KEY_CONFIG_APK_URL = "config_apk_url";
|
||||
private static final String KEY_CONFIG_APK_PACKAGE = "config_apk_package";
|
||||
private static final String KEY_OTA_CHECK_INTERVAL = "ota_check_interval_ms";
|
||||
|
||||
public static final boolean DEFAULT_ENABLE_SETUP_WIZARD = false;
|
||||
public static final boolean DEFAULT_ENABLE_PROVISIONING = false;
|
||||
public static final int DEFAULT_NETWORK_TIMEOUT = 30_000;
|
||||
public static final long DEFAULT_OTA_INTERVAL_MS = 24L * 60 * 60 * 1000; // 24 h
|
||||
public static final String DEFAULT_CONFIG_APK_URL = "https://default.example.com/config.apk";
|
||||
|
||||
public static boolean hasVendorConfig() {
|
||||
File configFile = new File(VENDOR_CONFIG_PATH);
|
||||
boolean exists = configFile.exists();
|
||||
boolean exists = new File(VENDOR_CONFIG_PATH).exists();
|
||||
Log.d(TAG, "Vendor config exists: " + exists + " at " + VENDOR_CONFIG_PATH);
|
||||
return exists;
|
||||
}
|
||||
|
||||
|
||||
public static boolean isSetupWizardEnabled() {
|
||||
return getConfigBoolean(KEY_ENABLE_SETUP_WIZARD, DEFAULT_ENABLE_SETUP_WIZARD);
|
||||
return getBoolean(KEY_ENABLE_SETUP_WIZARD, DEFAULT_ENABLE_SETUP_WIZARD);
|
||||
}
|
||||
|
||||
|
||||
public static boolean isProvisioningEnabled() {
|
||||
return getConfigBoolean(KEY_ENABLE_PROVISIONING, DEFAULT_ENABLE_PROVISIONING);
|
||||
return getBoolean(KEY_ENABLE_PROVISIONING, DEFAULT_ENABLE_PROVISIONING);
|
||||
}
|
||||
|
||||
|
||||
public static String getConfigApkUrl() {
|
||||
return getConfigString(KEY_CONFIG_APK_URL, DEFAULT_CONFIG_APK_URL);
|
||||
return getString(KEY_CONFIG_APK_URL, DEFAULT_CONFIG_APK_URL);
|
||||
}
|
||||
|
||||
|
||||
public static String getConfigApkPackage() {
|
||||
return getString(KEY_CONFIG_APK_PACKAGE, "app.pawlet.config");
|
||||
}
|
||||
|
||||
public static String getVendorId() {
|
||||
return getConfigString(KEY_VENDOR_ID, "default_vendor");
|
||||
return getString(KEY_VENDOR_ID, "default_vendor");
|
||||
}
|
||||
|
||||
|
||||
public static int getNetworkTimeout() {
|
||||
try {
|
||||
return Integer.parseInt(getConfigString(KEY_NETWORK_TIMEOUT,
|
||||
String.valueOf(DEFAULT_NETWORK_TIMEOUT)));
|
||||
return Integer.parseInt(getString(KEY_NETWORK_TIMEOUT,
|
||||
String.valueOf(DEFAULT_NETWORK_TIMEOUT)));
|
||||
} catch (NumberFormatException e) {
|
||||
return DEFAULT_NETWORK_TIMEOUT;
|
||||
}
|
||||
}
|
||||
|
||||
private static boolean getConfigBoolean(String key, boolean defaultValue) {
|
||||
String value = getConfigValue(key);
|
||||
if (value != null) {
|
||||
return "true".equalsIgnoreCase(value) || "1".equals(value);
|
||||
|
||||
public static long getOtaCheckIntervalMs() {
|
||||
try {
|
||||
return Long.parseLong(getString(KEY_OTA_CHECK_INTERVAL,
|
||||
String.valueOf(DEFAULT_OTA_INTERVAL_MS)));
|
||||
} catch (NumberFormatException e) {
|
||||
return DEFAULT_OTA_INTERVAL_MS;
|
||||
}
|
||||
|
||||
// Fallback to system properties if config file doesn't exist
|
||||
if (!hasVendorConfig()) {
|
||||
String propValue = android.os.SystemProperties.get("persist.configprovisioner." + key, "");
|
||||
if (!propValue.isEmpty()) {
|
||||
return "true".equalsIgnoreCase(propValue) || "1".equals(propValue);
|
||||
}
|
||||
}
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
private static String getConfigString(String key, String defaultValue) {
|
||||
String value = getConfigValue(key);
|
||||
if (value != null) return value;
|
||||
|
||||
// Fallback to system properties if config file doesn't exist
|
||||
if (!hasVendorConfig()) {
|
||||
return android.os.SystemProperties.get("persist.configprovisioner." + key, defaultValue);
|
||||
}
|
||||
|
||||
return defaultValue;
|
||||
|
||||
public static void logConfigValues() {
|
||||
if (!hasVendorConfig()) { Log.d(TAG, "No vendor config file found"); return; }
|
||||
Log.d(TAG, KEY_ENABLE_SETUP_WIZARD + "=" + isSetupWizardEnabled());
|
||||
Log.d(TAG, KEY_ENABLE_PROVISIONING + "=" + isProvisioningEnabled());
|
||||
Log.d(TAG, KEY_CONFIG_APK_URL + "=" + getConfigApkUrl());
|
||||
Log.d(TAG, KEY_VENDOR_ID + "=" + getVendorId());
|
||||
Log.d(TAG, KEY_NETWORK_TIMEOUT + "=" + getNetworkTimeout());
|
||||
Log.d(TAG, KEY_OTA_CHECK_INTERVAL + "=" + getOtaCheckIntervalMs());
|
||||
}
|
||||
|
||||
private static String getConfigValue(String key) {
|
||||
if (!hasVendorConfig()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
try (BufferedReader reader = new BufferedReader(new FileReader(VENDOR_CONFIG_PATH))) {
|
||||
|
||||
private static boolean getBoolean(String key, boolean def) {
|
||||
String v = rawValue(key);
|
||||
if (v != null) return "true".equalsIgnoreCase(v) || "1".equals(v);
|
||||
String prop = android.os.SystemProperties.get("persist.configprovisioner." + key, "");
|
||||
if (!prop.isEmpty()) return "true".equalsIgnoreCase(prop) || "1".equals(prop);
|
||||
return def;
|
||||
}
|
||||
|
||||
private static String getString(String key, String def) {
|
||||
String v = rawValue(key);
|
||||
if (v != null) return v;
|
||||
String prop = android.os.SystemProperties.get("persist.configprovisioner." + key, "");
|
||||
if (!prop.isEmpty()) return prop;
|
||||
return def;
|
||||
}
|
||||
|
||||
private static String rawValue(String key) {
|
||||
if (!new File(VENDOR_CONFIG_PATH).exists()) return null;
|
||||
try (BufferedReader r = new BufferedReader(new FileReader(VENDOR_CONFIG_PATH))) {
|
||||
String line;
|
||||
while ((line = reader.readLine()) != null) {
|
||||
while ((line = r.readLine()) != null) {
|
||||
line = line.trim();
|
||||
if (line.startsWith("#") || line.isEmpty()) continue;
|
||||
|
||||
String[] parts = line.split("=", 2);
|
||||
if (parts.length == 2 && parts[0].trim().equals(key)) {
|
||||
return parts[1].trim();
|
||||
}
|
||||
if (parts.length == 2 && parts[0].trim().equals(key)) return parts[1].trim();
|
||||
}
|
||||
} catch (IOException e) {
|
||||
Log.e(TAG, "Error reading vendor config", e);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
// Helper method to get all config values for debugging
|
||||
public static void logConfigValues() {
|
||||
if (!hasVendorConfig()) {
|
||||
Log.d(TAG, "No vendor config file found");
|
||||
return;
|
||||
}
|
||||
|
||||
Log.d(TAG, "Current configuration:");
|
||||
Log.d(TAG, KEY_ENABLE_SETUP_WIZARD + "=" + isSetupWizardEnabled());
|
||||
Log.d(TAG, KEY_ENABLE_PROVISIONING + "=" + isProvisioningEnabled());
|
||||
Log.d(TAG, KEY_CONFIG_APK_URL + "=" + getConfigApkUrl());
|
||||
Log.d(TAG, KEY_VENDOR_ID + "=" + getVendorId());
|
||||
Log.d(TAG, KEY_NETWORK_TIMEOUT + "=" + getNetworkTimeout());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user