blob: 006a9b754a1892f10799b08ebe84736466bde76d [file] [log] [blame]
/*
* Copyright (C) 2017 The Android Open Source Project
*
* 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 com.android.launcher3.model;
import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteException;
import android.util.Log;
import android.util.SparseArray;
import com.android.launcher3.R;
import com.android.launcher3.provider.LauncherDbUtils.SQLiteTransaction;
import com.android.launcher3.util.IOUtils;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Collections;
/**
* Utility class to handle DB downgrade
*/
public class DbDowngradeHelper {
private static final String TAG = "DbDowngradeHelper";
private static final String KEY_VERSION = "version";
private static final String KEY_DOWNGRADE_TO = "downgrade_to_";
private final SparseArray<String[]> mStatements = new SparseArray<>();
public final int version;
private DbDowngradeHelper(int version) {
this.version = version;
}
public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) {
ArrayList<String> allCommands = new ArrayList<>();
for (int i = oldVersion - 1; i >= newVersion; i--) {
String[] commands = mStatements.get(i);
if (commands == null) {
throw new SQLiteException("Downgrade path not supported to version " + i);
}
Collections.addAll(allCommands, commands);
}
try (SQLiteTransaction t = new SQLiteTransaction(db)) {
for (String sql : allCommands) {
db.execSQL(sql);
}
t.commit();
}
}
/**
* Creates a helper from the provided file
*/
public static DbDowngradeHelper parse(File file) throws JSONException, IOException {
return parse(IOUtils.toByteArray(file));
}
/**
* Creates a helper from the provided bytes
*/
public static DbDowngradeHelper parse(byte[] fileData) throws JSONException {
JSONObject obj = new JSONObject(new String(fileData));
DbDowngradeHelper helper = new DbDowngradeHelper(obj.getInt(KEY_VERSION));
for (int version = helper.version - 1; version > 0; version--) {
if (obj.has(KEY_DOWNGRADE_TO + version)) {
JSONArray statements = obj.getJSONArray(KEY_DOWNGRADE_TO + version);
String[] parsed = new String[statements.length()];
for (int i = 0; i < parsed.length; i++) {
parsed[i] = statements.getString(i);
}
helper.mStatements.put(version, parsed);
}
}
return helper;
}
public static void updateSchemaFile(File schemaFile, int expectedVersion, Context context) {
try {
if (DbDowngradeHelper.parse(schemaFile).version >= expectedVersion) {
return;
}
} catch (Exception e) {
// Schema error
}
// Write the updated schema
try (FileOutputStream fos = new FileOutputStream(schemaFile);
InputStream in = context.getResources().openRawResource(R.raw.downgrade_schema)) {
IOUtils.copy(in, fos);
} catch (IOException e) {
Log.e(TAG, "Error writing schema file", e);
}
}
}