OpenDelta: Add support for stream flashing on AB devices

Directly download and flash at the same time.
Requires extra info in json to work.
Will just show the user the flash button and the download one to preserve old functionality

TBD: Adapt the json generating shell script in build to include missing info
This commit is contained in:
Ido Ben-Hur 2023-09-24 13:56:26 +03:00
parent f23e9ad47a
commit 9f67d8ad95
No known key found for this signature in database
GPG Key ID: 0B827201D8C20BFE
4 changed files with 205 additions and 119 deletions

View File

@ -44,6 +44,7 @@ class ABUpdate {
private static final String PAYLOAD_PROPERTIES_PATH = "payload_properties.txt"; private static final String PAYLOAD_PROPERTIES_PATH = "payload_properties.txt";
private static final String PREFS_IS_INSTALLING_UPDATE = "prefs_is_installing_update"; private static final String PREFS_IS_INSTALLING_UPDATE = "prefs_is_installing_update";
private static final String PREFS_IS_SUSPENDED = "prefs_is_suspended"; private static final String PREFS_IS_SUSPENDED = "prefs_is_suspended";
private static final String FILE_PREFIX = "file://";
private static final long WAKELOCK_TIMEOUT = 60 * 60 * 1000; /* 1 hour */ private static final long WAKELOCK_TIMEOUT = 60 * 60 * 1000; /* 1 hour */
// non UpdateEngine errors // non UpdateEngine errors
@ -61,14 +62,16 @@ class ABUpdate {
private String mZipPath; private String mZipPath;
private ProgressListener mProgressListener; private ProgressListener mProgressListener;
private boolean mBound; private boolean mBound;
private boolean mIsStream;
private final UpdateEngineCallback mUpdateEngineCallback = new UpdateEngineCallback() { private final UpdateEngineCallback mUpdateEngineCallback = new UpdateEngineCallback() {
@Override @Override
public void onStatusUpdate(int status, float percent) { public void onStatusUpdate(int status, float percent) {
Logger.d("onStatusUpdate = " + status + " " + percent + "%%"); Logger.d("onStatusUpdate = " + status + " " + percent + "%%");
// downloading stage: 0% - 30% // downloading stage: 0% - 30%
// when streaming: 0% - 50%
int offset = 0; int offset = 0;
int weight = 30; int weight = mIsStream ? 50 : 30;
switch (status) { switch (status) {
case UpdateEngine.UpdateStatusConstants.UPDATED_NEED_REBOOT: case UpdateEngine.UpdateStatusConstants.UPDATED_NEED_REBOOT:
@ -81,13 +84,15 @@ class ABUpdate {
return; return;
case UpdateEngine.UpdateStatusConstants.VERIFYING: case UpdateEngine.UpdateStatusConstants.VERIFYING:
// verifying stage: 30% - 35% // verifying stage: 30% - 35%
offset = 30; // when streaming: 50% - 55%
offset = mIsStream ? 50 : 30;
weight = 5; weight = 5;
break; break;
case UpdateEngine.UpdateStatusConstants.FINALIZING: case UpdateEngine.UpdateStatusConstants.FINALIZING:
// finalizing stage: 35% - 100% // finalizing stage: 35% - 100%
offset = 35; // when streaming: 55% - 100%
weight = 65; offset = mIsStream ? 55 : 35;
weight = mIsStream ? 45 : 65;
break; break;
} }
@ -120,12 +125,18 @@ class ABUpdate {
Logger.ex(ex); Logger.ex(ex);
return ERROR_INVALID; return ERROR_INVALID;
} }
mZipPath = zipPath; return start(zipPath, null, 0, 0, listener);
}
public int start(String url, String[] headerKeyValuePairs,
long offset, long size, ProgressListener listener) {
mIsStream = headerKeyValuePairs != null;
mZipPath = url;
mProgressListener = listener; mProgressListener = listener;
if (isInstallingUpdate(mUpdateService)) { if (isInstallingUpdate(mUpdateService)) {
return -1; return -1;
} }
final int installing = startUpdate(); final int installing = startUpdate(headerKeyValuePairs, offset, size);
setInstallingUpdate(installing < 0, mUpdateService); setInstallingUpdate(installing < 0, mUpdateService);
return installing; return installing;
} }
@ -218,31 +229,32 @@ class ABUpdate {
return true; return true;
} }
private int startUpdate() { private int startUpdate(String[] headerKeyValuePairs, long offset, long size) {
Logger.d("startUpdate. mIsStream=" + mIsStream);
File file = new File(mZipPath); File file = new File(mZipPath);
if (!file.exists()) { if (!mIsStream) {
Log.e(TAG, "The given update doesn't exist"); if (!file.exists()) {
return ERROR_NOT_FOUND; Log.e(TAG, "The given update doesn't exist");
} return ERROR_NOT_FOUND;
}
long offset; try (ZipFile zipFile = new ZipFile(file)) {
String[] headerKeyValuePairs; offset = getZipEntryOffset(zipFile, PAYLOAD_BIN_PATH);
try (ZipFile zipFile = new ZipFile(file)) { ZipEntry payloadPropEntry = zipFile.getEntry(PAYLOAD_PROPERTIES_PATH);
offset = getZipEntryOffset(zipFile, PAYLOAD_BIN_PATH); try (InputStream is = zipFile.getInputStream(payloadPropEntry);
ZipEntry payloadPropEntry = zipFile.getEntry(PAYLOAD_PROPERTIES_PATH); InputStreamReader isr = new InputStreamReader(is);
try (InputStream is = zipFile.getInputStream(payloadPropEntry); BufferedReader br = new BufferedReader(isr)) {
InputStreamReader isr = new InputStreamReader(is); List<String> lines = new ArrayList<>();
BufferedReader br = new BufferedReader(isr)) { for (String line; (line = br.readLine()) != null;) {
List<String> lines = new ArrayList<>(); lines.add(line);
for (String line; (line = br.readLine()) != null;) { }
lines.add(line); headerKeyValuePairs = new String[lines.size()];
} headerKeyValuePairs = lines.toArray(headerKeyValuePairs);
headerKeyValuePairs = new String[lines.size()]; }
headerKeyValuePairs = lines.toArray(headerKeyValuePairs); Logger.d("payload offset=" + offset);
} catch (IOException | IllegalArgumentException e) {
Log.e(TAG, "Could not prepare " + file, e);
return ERROR_CORRUPTED;
} }
} catch (IOException | IllegalArgumentException e) {
Log.e(TAG, "Could not prepare " + file, e);
return ERROR_CORRUPTED;
} }
try { try {
@ -260,8 +272,17 @@ class ABUpdate {
} }
} }
if (!bindCallbacks()) return ERROR_NOT_READY; if (!bindCallbacks()) return ERROR_NOT_READY;
String zipFileUri = "file://" + file.getAbsolutePath(); String zipFileUri = mIsStream ? mZipPath : FILE_PREFIX + file.getAbsolutePath();
mUpdateEngine.applyPayload(zipFileUri, offset, 0, headerKeyValuePairs);
Logger.d("Applying payload with params:");
Logger.d("URI: " + zipFileUri);
Logger.d("offset: " + offset);
Logger.d("size: " + size);
Logger.d("headerKeyValuePairs:");
for (int i = 0; i < headerKeyValuePairs.length; i++)
Logger.d(headerKeyValuePairs[i]);
mUpdateEngine.applyPayload(zipFileUri, offset, size, headerKeyValuePairs);
return -1; return -1;
} }

View File

@ -447,12 +447,14 @@ public class MainActivity extends Activity {
mPrefs.edit().putString(UpdateService.PREF_LATEST_FULL_NAME, null).commit(); mPrefs.edit().putString(UpdateService.PREF_LATEST_FULL_NAME, null).commit();
break; break;
case State.ACTION_AVAILABLE: case State.ACTION_AVAILABLE:
case State.ACTION_AVAILABLE_STREAM:
final String latest = mPrefs.getString( final String latest = mPrefs.getString(
UpdateService.PREF_LATEST_FULL_NAME, null); UpdateService.PREF_LATEST_FULL_NAME, null);
if (latest != null) { if (latest != null) {
String latestBase = latest.substring(0, String latestBase = latest.substring(0,
latest.lastIndexOf('.')); latest.lastIndexOf('.'));
enableBuild = true; enableBuild = true;
enableFlash = mState.equals(State.ACTION_AVAILABLE_STREAM);
enableChangelog = true; enableChangelog = true;
updateVersion = latestBase; updateVersion = latestBase;
title = getString(R.string.state_action_build_full); title = getString(R.string.state_action_build_full);
@ -562,7 +564,7 @@ public class MainActivity extends Activity {
mFileFlashButton.setEnabled(mPermOk && !disableCheck); mFileFlashButton.setEnabled(mPermOk && !disableCheck);
mCheckBtn.setVisibility(hideCheck ? View.GONE : View.VISIBLE); mCheckBtn.setVisibility(hideCheck ? View.GONE : View.VISIBLE);
mFlashBtn.setVisibility(enableFlash ? View.VISIBLE : View.GONE); mFlashBtn.setVisibility(enableFlash ? View.VISIBLE : View.GONE);
mBuildBtn.setVisibility(!enableBuild || enableFlash ? View.GONE : View.VISIBLE); mBuildBtn.setVisibility(enableBuild ? View.VISIBLE : View.GONE);
mRebootBtn.setVisibility(enableReboot ? View.VISIBLE : View.GONE); mRebootBtn.setVisibility(enableReboot ? View.VISIBLE : View.GONE);
mFileFlashButton.setVisibility(hideCheck ? View.GONE : View.VISIBLE); mFileFlashButton.setVisibility(hideCheck ? View.GONE : View.VISIBLE);
@ -615,10 +617,14 @@ public class MainActivity extends Activity {
public void onButtonFlashNowClick(View v) { public void onButtonFlashNowClick(View v) {
if (Config.isABDevice()) { if (Config.isABDevice()) {
if (mState.equals(State.ACTION_AVAILABLE_STREAM)) {
streamStart.run();
return;
}
flashStart.run(); flashStart.run();
} else { return;
flashRecoveryWarning.run();
} }
flashRecoveryWarning.run();
} }
public void onButtonStopClick(View v) { public void onButtonStopClick(View v) {
@ -722,6 +728,13 @@ public class MainActivity extends Activity {
startUpdateService(UpdateService.ACTION_FLASH); startUpdateService(UpdateService.ACTION_FLASH);
}; };
private final Runnable streamStart = () -> {
mCheckBtn.setEnabled(false);
mFlashBtn.setEnabled(false);
mBuildBtn.setEnabled(false);
startUpdateService(UpdateService.ACTION_STREAM);
};
private void requestPermissions() { private void requestPermissions() {
mPermOk = true; mPermOk = true;
if (!Environment.isExternalStorageManager()) { if (!Environment.isExternalStorageManager()) {

View File

@ -39,6 +39,7 @@ public class State {
public static final String ACTION_AB_PAUSED = "action_ab_paused"; public static final String ACTION_AB_PAUSED = "action_ab_paused";
public static final String ACTION_AB_FINISHED = "action_ab_finished"; public static final String ACTION_AB_FINISHED = "action_ab_finished";
public static final String ACTION_AVAILABLE = "action_available"; public static final String ACTION_AVAILABLE = "action_available";
public static final String ACTION_AVAILABLE_STREAM = "action_available_stream";
public static final String ACTION_FLASH_FILE_NO_SUM = "action_flash_file_no_sum"; public static final String ACTION_FLASH_FILE_NO_SUM = "action_flash_file_no_sum";
public static final String ACTION_FLASH_FILE_INVALID_SUM = "action_flash_file_invalid_sum"; public static final String ACTION_FLASH_FILE_INVALID_SUM = "action_flash_file_invalid_sum";
public static final String ACTION_FLASH_FILE_READY = "action_flash_file_ready"; public static final String ACTION_FLASH_FILE_READY = "action_flash_file_ready";

View File

@ -63,8 +63,11 @@ import java.nio.charset.StandardCharsets;
import java.security.MessageDigest; import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException; import java.security.NoSuchAlgorithmException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Locale; import java.util.Locale;
import java.util.Set;
import java.util.stream.Collectors;
import org.json.JSONArray; import org.json.JSONArray;
import org.json.JSONException; import org.json.JSONException;
@ -104,6 +107,7 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
public static final String ACTION_DOWNLOAD_STOP = "eu.chainfire.opendelta.action.DOWNLOAD_STOP"; public static final String ACTION_DOWNLOAD_STOP = "eu.chainfire.opendelta.action.DOWNLOAD_STOP";
public static final String ACTION_DOWNLOAD_PAUSE = "eu.chainfire.opendelta.action.DOWNLOAD_PAUSE"; public static final String ACTION_DOWNLOAD_PAUSE = "eu.chainfire.opendelta.action.DOWNLOAD_PAUSE";
public static final String ACTION_FLASH = "eu.chainfire.opendelta.action.FLASH"; public static final String ACTION_FLASH = "eu.chainfire.opendelta.action.FLASH";
public static final String ACTION_STREAM = "eu.chainfire.opendelta.action.STREAM";
public static final String ACTION_ALARM = "eu.chainfire.opendelta.action.ALARM"; public static final String ACTION_ALARM = "eu.chainfire.opendelta.action.ALARM";
public static final String ACTION_SCHEDULER = "eu.chainfire.opendelta.action.SCHEDULER"; public static final String ACTION_SCHEDULER = "eu.chainfire.opendelta.action.SCHEDULER";
public static final String EXTRA_ALARM_ID = "eu.chainfire.opendelta.extra.ALARM_ID"; public static final String EXTRA_ALARM_ID = "eu.chainfire.opendelta.extra.ALARM_ID";
@ -118,6 +122,9 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
public static final int NOTIFICATION_UPDATE = 2; public static final int NOTIFICATION_UPDATE = 2;
public static final int NOTIFICATION_ERROR = 3; public static final int NOTIFICATION_ERROR = 3;
private static final String PAYLOAD_PROP_OFFSET = "offset=";
private static final String PAYLOAD_PROP_SIZE = "FILE_SIZE=";
public static final String PREF_READY_FILENAME_NAME = "ready_filename"; public static final String PREF_READY_FILENAME_NAME = "ready_filename";
public static final String PREF_LATEST_CHANGELOG = "latest_changelog"; public static final String PREF_LATEST_CHANGELOG = "latest_changelog";
@ -141,6 +148,7 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
public static final String PREF_AUTO_UPDATE_METERED_NETWORKS = "auto_update_metered_networks"; public static final String PREF_AUTO_UPDATE_METERED_NETWORKS = "auto_update_metered_networks";
public static final String PREF_LATEST_FULL_NAME = "latest_full_name"; public static final String PREF_LATEST_FULL_NAME = "latest_full_name";
public static final String PREF_LATEST_PAYLOAD_PROPS = "latest_payload_props";
public static final String PREF_DOWNLOAD_SIZE = "download_size_long"; public static final String PREF_DOWNLOAD_SIZE = "download_size_long";
public static final int PREF_AUTO_DOWNLOAD_DISABLED = 0; public static final int PREF_AUTO_DOWNLOAD_DISABLED = 0;
@ -372,6 +380,9 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
else flashUpdate(); else flashUpdate();
} }
break; break;
case ACTION_STREAM:
if (checkPermissions()) flashABUpdate(true);
break;
case ACTION_FLASH_FILE: case ACTION_FLASH_FILE:
if (intent.hasExtra(EXTRA_FILENAME)) { if (intent.hasExtra(EXTRA_FILENAME)) {
String flashFilename = intent.getStringExtra(EXTRA_FILENAME); String flashFilename = intent.getStringExtra(EXTRA_FILENAME);
@ -532,8 +543,10 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
} }
Logger.d("Assuming update available"); Logger.d("Assuming update available");
mState.update(State.ACTION_AVAILABLE, mPrefs.getLong(PREF_LAST_CHECK_TIME_NAME, final Set<String> propSet = mPrefs.getStringSet(PREF_LATEST_PAYLOAD_PROPS, null);
PREF_LAST_CHECK_TIME_DEFAULT)); final String state = (propSet != null && propSet.size() > 0)
? State.ACTION_AVAILABLE_STREAM : State.ACTION_AVAILABLE;
mState.update(state, mPrefs.getLong(PREF_LAST_CHECK_TIME_NAME, PREF_LAST_CHECK_TIME_DEFAULT));
maybeNotify(notify, latestBuild, null); maybeNotify(notify, latestBuild, null);
return; return;
} }
@ -706,75 +719,6 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
return false; return false;
} }
private List<String> getNewestBuild() {
Logger.d("Checking for latest build");
String url = mConfig.getUrlBaseJson();
String buildData = Download.asString(url);
if (buildData == null || buildData.length() == 0) {
mState.update(State.ERROR_DOWNLOAD, url, Download.ERROR_CODE_NEWEST_BUILD);
mNotificationManager.cancel(NOTIFICATION_BUSY);
return null;
}
JSONObject object;
try {
object = new JSONObject(buildData);
JSONArray updatesList = object.getJSONArray("response");
String latestBuild = null;
String urlOverride = null;
String sumOverride = null;
for (int i = 0; i < updatesList.length(); i++) {
if (updatesList.isNull(i)) {
continue;
}
try {
JSONObject build = updatesList.getJSONObject(i);
String fileName = new File(build.getString("filename")).getName();
String urlOvr = null;
String sumOvr = null;
if (build.has("url"))
urlOvr = build.getString("url");
if (build.has("sha256url"))
sumOvr = build.getString("sha256url");
Logger.d("parsed from json:");
Logger.d("fileName= " + fileName);
if (isMatchingImage(fileName))
latestBuild = fileName;
if (urlOvr != null && !urlOvr.equals("")) {
urlOverride = urlOvr;
Logger.d("url= " + urlOverride);
}
if (sumOvr != null && !sumOvr.equals("")) {
sumOverride = sumOvr;
Logger.d("sha256 url= " + sumOverride);
}
} catch (JSONException e) {
Logger.ex(e);
}
}
List<String> ret = new ArrayList<>();
if (latestBuild != null) {
ret.add(latestBuild);
if (urlOverride != null) {
ret.add(urlOverride);
if (sumOverride != null) {
ret.add(sumOverride);
mIsUrlOverride = true;
mSumUrlOvr = sumOverride;
}
}
}
return ret;
} catch (Exception e) {
Logger.ex(e);
}
mState.update(State.ERROR_UNOFFICIAL, mConfig.getVersion());
return null;
}
public ProgressListener getSUMProgress(String state, String filename) { public ProgressListener getSUMProgress(String state, String filename) {
final long[] last = new long[] { 0, SystemClock.elapsedRealtime() }; final long[] last = new long[] { 0, SystemClock.elapsedRealtime() };
final String _state = state; final String _state = state;
@ -1067,10 +1011,16 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
} }
private void flashABUpdate() { private void flashABUpdate() {
Logger.d("flashABUpdate"); flashABUpdate(false);
}
private void flashABUpdate(final boolean isStream) {
Logger.d("flashABUpdate. isStream=" + isStream);
String flashFilename; String flashFilename;
try { try {
flashFilename = handleUpdateCleanup(); flashFilename = isStream
? mPrefs.getString(PREF_READY_FILENAME_NAME, null)
: handleUpdateCleanup();
} catch (Exception ex) { } catch (Exception ex) {
mIsUpdateRunning = false; mIsUpdateRunning = false;
mState.update(State.ERROR_AB_FLASH, ABUpdate.ERROR_NOT_FOUND); mState.update(State.ERROR_AB_FLASH, ABUpdate.ERROR_NOT_FOUND);
@ -1084,12 +1034,41 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
// Clear the Download size to hide while flashing // Clear the Download size to hide while flashing
mPrefs.edit().putLong(PREF_DOWNLOAD_SIZE, -1).commit(); mPrefs.edit().putLong(PREF_DOWNLOAD_SIZE, -1).commit();
final String _filename = new File(flashFilename).getName(); String _filename = null;
if (isStream) {
final int eIndex = flashFilename.lastIndexOf('.');
final int sIndex = flashFilename.lastIndexOf('/', eIndex);
_filename = flashFilename.substring(sIndex + 1, eIndex);
} else {
_filename = new File(flashFilename).getName();
}
mState.update(State.ACTION_AB_FLASH, 0f, 0L, 100L, _filename, null); mState.update(State.ACTION_AB_FLASH, 0f, 0L, 100L, _filename, null);
newFlashNotification(_filename); newFlashNotification(_filename);
final int code = ABUpdate.getInstance(this).start(flashFilename, mProgressListener); int code = -1;
if (isStream) {
Set<String> payloadSet = mPrefs.getStringSet(PREF_LATEST_PAYLOAD_PROPS, null);
List<String> payloadProps = new ArrayList<>();
long offset = 0;
long size = 0;
for (String str : payloadSet) {
if (offset == 0 && str.startsWith(PAYLOAD_PROP_OFFSET)) {
offset = Long.parseLong(str.substring(PAYLOAD_PROP_OFFSET.length(), str.length()));
continue;
}
if (size == 0 && str.startsWith(PAYLOAD_PROP_SIZE))
size = Long.parseLong(str.substring(PAYLOAD_PROP_SIZE.length(), str.length()));
payloadProps.add(str);
}
String[] headerKeyValuePairs = new String[payloadProps.size()];
for (int i = 0; i < payloadProps.size(); i++)
headerKeyValuePairs[i] = payloadProps.get(i);
code = ABUpdate.getInstance(this).start(flashFilename, headerKeyValuePairs,
offset, size, mProgressListener);
} else {
code = ABUpdate.getInstance(this).start(flashFilename, mProgressListener);
}
if (code < 0) { if (code < 0) {
mLastProgressTime = new long[] { 0, SystemClock.elapsedRealtime() }; mLastProgressTime = new long[] { 0, SystemClock.elapsedRealtime() };
mProgressListener.setStatus(_filename); mProgressListener.setStatus(_filename);
@ -1284,6 +1263,7 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
private void clearState() { private void clearState() {
SharedPreferences.Editor editor = mPrefs.edit(); SharedPreferences.Editor editor = mPrefs.edit();
editor.putString(PREF_LATEST_FULL_NAME, null); editor.putString(PREF_LATEST_FULL_NAME, null);
editor.putString(PREF_LATEST_PAYLOAD_PROPS, null);
editor.putString(PREF_READY_FILENAME_NAME, null); editor.putString(PREF_READY_FILENAME_NAME, null);
editor.putString(PREF_LATEST_CHANGELOG, null); editor.putString(PREF_LATEST_CHANGELOG, null);
editor.putLong(PREF_DOWNLOAD_SIZE, -1); editor.putLong(PREF_DOWNLOAD_SIZE, -1);
@ -1329,26 +1309,88 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
(new File(mConfig.getPathBase())).mkdir(); (new File(mConfig.getPathBase())).mkdir();
(new File(mConfig.getPathFlashAfterUpdate())).mkdir(); (new File(mConfig.getPathFlashAfterUpdate())).mkdir();
List<String> latestBuildWithUrl = getNewestBuild(); Logger.d("Checking for latest build");
String latestBuild;
String url = mConfig.getUrlBaseJson();
String latestBuild = null;
String urlOverride = null;
String sumOverride = null;
List<String> payloadProps = null;
String buildData = Download.asString(url);
if (buildData == null || buildData.length() == 0) {
mState.update(State.ERROR_DOWNLOAD, url, Download.ERROR_CODE_NEWEST_BUILD);
mNotificationManager.cancel(NOTIFICATION_BUSY);
}
JSONObject object;
try {
object = new JSONObject(buildData);
JSONArray updatesList = object.getJSONArray("response");
for (int i = 0; i < updatesList.length(); i++) {
if (updatesList.isNull(i)) {
continue;
}
try {
JSONObject build = updatesList.getJSONObject(i);
String fileName = new File(build.getString("filename")).getName();
if (build.has("url"))
urlOverride = build.getString("url");
if (build.has("sha256url"))
sumOverride = build.getString("sha256url");
if (build.has("payload")) {
payloadProps = new ArrayList<>();
JSONArray payloadList = build.getJSONArray("payload");
for (int j = 0; j < payloadList.length(); j++) {
if (payloadList.isNull(j)) continue;
JSONObject prop = payloadList.getJSONObject(j);
Iterator<String> keys = prop.keys();
while (keys.hasNext()) {
final String key = keys.next();
payloadProps.add(key + "=" + prop.get(key));
}
}
}
Logger.d("parsed from json:");
Logger.d("fileName= " + fileName);
if (isMatchingImage(fileName))
latestBuild = fileName;
if (urlOverride != null && !urlOverride.equals(""))
Logger.d("url= " + urlOverride);
if (sumOverride != null && !sumOverride.equals("")) {
Logger.d("sha256 url= " + sumOverride);
}
if (payloadProps != null) {
for (String str : payloadProps) {
Logger.d(str);
}
}
} catch (JSONException e) {
Logger.ex(e);
}
}
} catch (Exception e) {
Logger.ex(e);
mState.update(State.ERROR_UNOFFICIAL, mConfig.getVersion());
return;
}
// if we don't even find a build on dl no sense to continue // if we don't even find a build on dl no sense to continue
if (latestBuildWithUrl == null || latestBuildWithUrl.size() == 0) { if (latestBuild == null || latestBuild.length() == 0) {
Logger.d("no latest build found at " + mConfig.getUrlBaseJson() + Logger.d("no latest build found at " + mConfig.getUrlBaseJson() +
" for " + mConfig.getDevice()); " for " + mConfig.getDevice());
return; return;
} }
latestBuild = latestBuildWithUrl.get(0);
String latestFetch; String latestFetch;
String latestFetchSUM; String latestFetchSUM;
if (latestBuildWithUrl.size() < 3) { if (urlOverride == null || sumOverride == null) {
latestFetch = mConfig.getUrlBase() + latestFetch = mConfig.getUrlBase() +
latestBuild + mConfig.getUrlSuffix(); latestBuild + mConfig.getUrlSuffix();
latestFetchSUM = mConfig.getUrlBaseSum() + latestFetchSUM = mConfig.getUrlBaseSum() +
latestBuild + ".sha256sum" + mConfig.getUrlSuffix(); latestBuild + ".sha256sum" + mConfig.getUrlSuffix();
} else { } else {
latestFetch = latestBuildWithUrl.get(1); latestFetch = urlOverride;
latestFetchSUM = latestBuildWithUrl.get(2); latestFetchSUM = sumOverride;
} }
Logger.d("latest build for device " + mConfig.getDevice() + " is " + latestFetch); Logger.d("latest build for device " + mConfig.getDevice() + " is " + latestFetch);
@ -1374,10 +1416,19 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
updateAvailable ? latestBuild : null).commit(); updateAvailable ? latestBuild : null).commit();
if (!updateAvailable) return; if (!updateAvailable) return;
if (payloadProps != null) {
mPrefs.edit().putStringSet(PREF_LATEST_PAYLOAD_PROPS,
payloadProps.stream().collect(Collectors.toSet())).commit();
mPrefs.edit().putString(PREF_READY_FILENAME_NAME, latestFetch).commit();
Logger.d("update supports streaming");
} else {
mPrefs.edit().remove(PREF_LATEST_PAYLOAD_PROPS).commit();
}
final String changelog = getChangelogString(); final String changelog = getChangelogString();
mPrefs.edit().putString(PREF_LATEST_CHANGELOG, changelog).commit(); mPrefs.edit().putString(PREF_LATEST_CHANGELOG, changelog).commit();
if (checkExistingBuild(latestBuildWithUrl, latestFetchSUM)) return; if (checkExistingBuild(latestBuild, latestFetchSUM)) return;
final long size = Download.getSize(latestFetch); final long size = Download.getSize(latestFetch);
mPrefs.edit().putLong(PREF_DOWNLOAD_SIZE, size).commit(); mPrefs.edit().putLong(PREF_DOWNLOAD_SIZE, size).commit();
@ -1437,8 +1488,8 @@ public class UpdateService extends Service implements OnSharedPreferenceChangeLi
}); });
} }
private boolean checkExistingBuild(List<String> latestBuildWithUrl, String latestFetchSUM) { private boolean checkExistingBuild(String latestBuild, String latestFetchSUM) {
String fn = mConfig.getPathBase() + latestBuildWithUrl.get(0); String fn = mConfig.getPathBase() + latestBuild;
File file = new File(fn); File file = new File(fn);
if (file.exists()) { if (file.exists()) {
if (checkBuildSHA256Sum(latestFetchSUM, fn)) { if (checkBuildSHA256Sum(latestFetchSUM, fn)) {