mirror of https://github.com/tasks/tasks
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1150 lines
45 KiB
Java
1150 lines
45 KiB
Java
/**
|
|
* Copyright (c) 2012 Todoroo Inc
|
|
*
|
|
* See the file "LICENSE" for the full license governing this code.
|
|
*/
|
|
package com.localytics.android;
|
|
|
|
import java.io.File;
|
|
import java.util.Arrays;
|
|
import java.util.Collections;
|
|
import java.util.HashMap;
|
|
import java.util.HashSet;
|
|
import java.util.Map;
|
|
import java.util.Set;
|
|
|
|
import android.content.ContentValues;
|
|
import android.content.Context;
|
|
import android.database.Cursor;
|
|
import android.database.DatabaseUtils;
|
|
import android.database.sqlite.SQLiteDatabase;
|
|
import android.database.sqlite.SQLiteOpenHelper;
|
|
import android.database.sqlite.SQLiteQueryBuilder;
|
|
import android.provider.BaseColumns;
|
|
import android.util.Log;
|
|
|
|
/**
|
|
* Implements the storage mechanism for the Localytics library. The interface and implementation are similar to a ContentProvider
|
|
* but modified to be better suited to a library. The interface is table-oriented, rather than Uri-oriented.
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
/* package */final class LocalyticsProvider
|
|
{
|
|
/**
|
|
* Name of the Localytics database, stored in the host application's {@link Context#getDatabasePath(String)}.
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
/*
|
|
* This field is made package-accessible for unit testing. While the exact file name is arbitrary, this name was chosen to
|
|
* avoid collisions with app developers because it is sufficiently long and uses the Localytics package namespace.
|
|
*/
|
|
/* package */static final String DATABASE_FILE = "com.localytics.android.%s.sqlite"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* Version of the database.
|
|
* <p>
|
|
* Version history:
|
|
* <ol>
|
|
* <li>1: Initial version</li>
|
|
* <li>2: No format changes--just deleting bad data stranded in the database</li>
|
|
* </ol>
|
|
*/
|
|
private static final int DATABASE_VERSION = 2;
|
|
|
|
/**
|
|
* Singleton instance of the {@link LocalyticsProvider}. Lazily initialized via {@link #getInstance(Context, String)}.
|
|
*/
|
|
private static final Map<String, LocalyticsProvider> sLocalyticsProviderMap = new HashMap<String, LocalyticsProvider>();
|
|
|
|
/**
|
|
* Intrinsic lock for synchronizing the initialization of {@link #sLocalyticsProviderMap}.
|
|
*/
|
|
/*
|
|
* Fun fact: Object[0] is more efficient that Object for an intrinsic lock
|
|
*/
|
|
private static final Object[] sLocalyticsProviderIntrinsicLock = new Object[0];
|
|
|
|
/**
|
|
* Unmodifiable set of valid table names.
|
|
*/
|
|
private static final Set<String> sValidTables = Collections.unmodifiableSet(getValidTables());
|
|
|
|
/**
|
|
* SQLite database owned by the provider.
|
|
*/
|
|
private final SQLiteDatabase mDb;
|
|
|
|
/**
|
|
* Obtains an instance of the Localytics Provider. Since the provider is a singleton object, only a single instance will be
|
|
* returned.
|
|
* <p>
|
|
* Note: if {@code context} is an instance of {@link android.test.RenamingDelegatingContext}, then a new object will be
|
|
* returned every time. This is not a "public" API, but is documented here as it aids unit testing.
|
|
*
|
|
* @param context Application context. Cannot be null.
|
|
* @param apiKey TODO
|
|
* @return An instance of {@link LocalyticsProvider}.
|
|
* @throws IllegalArgumentException if {@code context} is null
|
|
*/
|
|
public static LocalyticsProvider getInstance(final Context context, final String apiKey)
|
|
{
|
|
/*
|
|
* Note: Don't call getApplicationContext() on the context, as that would return a different context and defeat useful
|
|
* contexts such as RenamingDelegatingContext.
|
|
*/
|
|
|
|
if (Constants.ENABLE_PARAMETER_CHECKING)
|
|
{
|
|
if (null == context)
|
|
{
|
|
throw new IllegalArgumentException("context cannot be null"); //$NON-NLS-1$
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Although RenamingDelegatingContext is part of the Android SDK, the class isn't present in the ClassLoader unless the
|
|
* process is being run as a unit test. For that reason, comparing class names is necessary instead of doing instanceof.
|
|
*/
|
|
if (context.getClass().getName().equals("android.test.RenamingDelegatingContext")) //$NON-NLS-1$
|
|
{
|
|
return new LocalyticsProvider(context, apiKey);
|
|
}
|
|
|
|
synchronized (sLocalyticsProviderIntrinsicLock)
|
|
{
|
|
LocalyticsProvider provider = sLocalyticsProviderMap.get(apiKey);
|
|
|
|
if (null == provider)
|
|
{
|
|
provider = new LocalyticsProvider(context, apiKey);
|
|
sLocalyticsProviderMap.put(apiKey, provider);
|
|
}
|
|
|
|
return provider;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Constructs a new Localytics Provider.
|
|
* <p>
|
|
* Note: this method may perform disk operations.
|
|
*
|
|
* @param context application context. Cannot be null.
|
|
*/
|
|
private LocalyticsProvider(final Context context, final String apiKey)
|
|
{
|
|
/*
|
|
* Rather than use the API key directly in the file name, it is put through SHA-256. The main reason for doing that is to
|
|
* decouple the requirements of the Android file system from the possible values of the API key string. There is a very,
|
|
* very small risk of a collision with the SHA-256 algorithm, but most clients will only have a single API key. Those with
|
|
* multiple keys may have 2 or 3, so the risk of a collision there is also very low.
|
|
*/
|
|
|
|
mDb = new DatabaseHelper(context, String.format(DATABASE_FILE, DatapointHelper.getSha256(apiKey)), DATABASE_VERSION).getWritableDatabase();
|
|
}
|
|
|
|
/**
|
|
* Inserts a new record.
|
|
* <p>
|
|
* Note: this method may perform disk operations.
|
|
*
|
|
* @param tableName name of the table operate on. Must be one of the recognized tables. Cannot be null.
|
|
* @param values ContentValues to insert. Cannot be null.
|
|
* @return the {@link BaseColumns#_ID} of the inserted row or -1 if an error occurred.
|
|
* @throws IllegalArgumentException if tableName is null or not a valid table name.
|
|
* @throws IllegalArgumentException if values are null.
|
|
*/
|
|
public long insert(final String tableName, final ContentValues values)
|
|
{
|
|
if (Constants.ENABLE_PARAMETER_CHECKING)
|
|
{
|
|
if (!isValidTable(tableName))
|
|
{
|
|
throw new IllegalArgumentException(String.format("tableName %s is invalid", tableName)); //$NON-NLS-1$
|
|
}
|
|
|
|
if (null == values)
|
|
{
|
|
throw new IllegalArgumentException("values cannot be null"); //$NON-NLS-1$
|
|
}
|
|
}
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, String.format("Insert table: %s, values: %s", tableName, values.toString())); //$NON-NLS-1$
|
|
}
|
|
|
|
final long result = mDb.insertOrThrow(tableName, null, values);
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, String.format("Inserted row with new id %d", Long.valueOf(result))); //$NON-NLS-1$
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Performs a query.
|
|
* <p>
|
|
* Note: this method may perform disk operations.
|
|
*
|
|
* @param tableName name of the table operate on. Must be one of the recognized tables. Cannot be null.
|
|
* @param projection The list of columns to include. If null, then all columns are included by default.
|
|
* @param selection A filter to apply to all rows, like the SQLite WHERE clause. Passing null will query all rows. This param
|
|
* may contain ? symbols, which will be replaced by values from the {@code selectionArgs} param.
|
|
* @param selectionArgs An optional string array of replacements for ? symbols in {@code selection}. May be null.
|
|
* @param sortOrder How the rows in the cursor should be sorted. If null, then the sort order is undefined.
|
|
* @return Cursor for the query. To the receiver: Don't forget to call .close() on the cursor when finished with it.
|
|
* @throws IllegalArgumentException if tableName is null or not a valid table name.
|
|
*/
|
|
public Cursor query(final String tableName, final String[] projection, final String selection, final String[] selectionArgs, final String sortOrder)
|
|
{
|
|
if (Constants.ENABLE_PARAMETER_CHECKING)
|
|
{
|
|
if (!isValidTable(tableName))
|
|
{
|
|
throw new IllegalArgumentException(String.format("tableName %s is invalid", tableName)); //$NON-NLS-1$
|
|
}
|
|
}
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, String.format("Query table: %s, projection: %s, selection: %s, selectionArgs: %s", tableName, Arrays.toString(projection), selection, Arrays.toString(selectionArgs))); //$NON-NLS-1$
|
|
}
|
|
|
|
final SQLiteQueryBuilder qb = new SQLiteQueryBuilder();
|
|
qb.setTables(tableName);
|
|
|
|
final Cursor result = qb.query(mDb, projection, selection, selectionArgs, null, null, sortOrder);
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, "Query result is: " + DatabaseUtils.dumpCursorToString(result)); //$NON-NLS-1$
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Updates row(s).
|
|
* <p>
|
|
* Note: this method may perform disk operations.
|
|
*
|
|
* @param tableName name of the table operate on. Must be one of the recognized tables. Cannot be null.
|
|
* @param values A ContentValues mapping from column names (see the associated BaseColumns class for the table) to new column
|
|
* values.
|
|
* @param selection A filter to limit which rows are updated, like the SQLite WHERE clause. Passing null implies all rows.
|
|
* This param may contain ? symbols, which will be replaced by values from the {@code selectionArgs} param.
|
|
* @param selectionArgs An optional string array of replacements for ? symbols in {@code selection}. May be null.
|
|
* @return int representing the number of rows modified, which is in the range from 0 to the number of items in the table.
|
|
* @throws IllegalArgumentException if tableName is null or not a valid table name.
|
|
*/
|
|
public int update(final String tableName, final ContentValues values, final String selection, final String[] selectionArgs)
|
|
{
|
|
if (Constants.ENABLE_PARAMETER_CHECKING)
|
|
{
|
|
if (!isValidTable(tableName))
|
|
{
|
|
throw new IllegalArgumentException(String.format("tableName %s is invalid", tableName)); //$NON-NLS-1$
|
|
}
|
|
}
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, String.format("Update table: %s, values: %s, selection: %s, selectionArgs: %s", tableName, values.toString(), selection, Arrays.toString(selectionArgs))); //$NON-NLS-1$
|
|
}
|
|
|
|
return mDb.update(tableName, values, selection, selectionArgs);
|
|
}
|
|
|
|
/**
|
|
* Deletes row(s).
|
|
* <p>
|
|
* Note: this method may perform disk operations.
|
|
*
|
|
* @param tableName name of the table operate on. Must be one of the recognized tables. Cannot be null.
|
|
* @param selection A filter to limit which rows are deleted, like the SQLite WHERE clause. Passing null implies all rows.
|
|
* This param may contain ? symbols, which will be replaced by values from the {@code selectionArgs} param.
|
|
* @param selectionArgs An optional string array of replacements for ? symbols in {@code selection}. May be null.
|
|
* @return The number of rows affected, which is in the range from 0 to the number of items in the table.
|
|
* @throws IllegalArgumentException if tableName is null or not a valid table name.
|
|
*/
|
|
public int delete(final String tableName, final String selection, final String[] selectionArgs)
|
|
{
|
|
if (Constants.ENABLE_PARAMETER_CHECKING)
|
|
{
|
|
if (!isValidTable(tableName))
|
|
{
|
|
throw new IllegalArgumentException(String.format("tableName %s is invalid", tableName)); //$NON-NLS-1$
|
|
}
|
|
}
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, String.format("Delete table: %s, selection: %s, selectionArgs: %s", tableName, selection, Arrays.toString(selectionArgs))); //$NON-NLS-1$
|
|
}
|
|
|
|
final int count;
|
|
if (null == selection)
|
|
{
|
|
count = mDb.delete(tableName, "1", null); //$NON-NLS-1$
|
|
}
|
|
else
|
|
{
|
|
count = mDb.delete(tableName, selection, selectionArgs);
|
|
}
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, String.format("Deleted %d rows", Integer.valueOf(count))); //$NON-NLS-1$
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
/**
|
|
* Executes an arbitrary runnable with exclusive access to the database, essentially allowing an atomic transaction.
|
|
*
|
|
* @param runnable Runnable to execute. Cannot be null.
|
|
* @throws IllegalArgumentException if {@code runnable} is null
|
|
*/
|
|
/*
|
|
* This implementation is sort of a hack. In the future, it would be better model this after applyBatch() with a list of
|
|
* ContentProviderOperation objects. But that API isn't available until Android 2.0.
|
|
*
|
|
* An alternative implementation would have been to expose the begin/end transaction methods on the Provider object. While
|
|
* that would work, it makes it harder to transition to a ContentProviderOperation model in the future.
|
|
*/
|
|
public void runBatchTransaction(final Runnable runnable)
|
|
{
|
|
if (Constants.ENABLE_PARAMETER_CHECKING)
|
|
{
|
|
if (null == runnable)
|
|
{
|
|
throw new IllegalArgumentException("runnable cannot be null"); //$NON-NLS-1$
|
|
}
|
|
}
|
|
|
|
mDb.beginTransaction();
|
|
try
|
|
{
|
|
runnable.run();
|
|
|
|
mDb.setTransactionSuccessful();
|
|
}
|
|
finally
|
|
{
|
|
mDb.endTransaction();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Private helper to test whether a given table name is valid
|
|
*
|
|
* @param table name of a table to check. This param may be null.
|
|
* @return true if the table is valid, false if the table is invalid. If {@code table} is null, returns false.
|
|
*/
|
|
private static boolean isValidTable(final String table)
|
|
{
|
|
if (null == table)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (!sValidTables.contains(table))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Private helper that knows all the tables that {@link LocalyticsProvider} can operate on.
|
|
*
|
|
* @return returns a set of the valid tables.
|
|
*/
|
|
private static Set<String> getValidTables()
|
|
{
|
|
final HashSet<String> tables = new HashSet<String>();
|
|
|
|
tables.add(ApiKeysDbColumns.TABLE_NAME);
|
|
tables.add(AttributesDbColumns.TABLE_NAME);
|
|
tables.add(EventsDbColumns.TABLE_NAME);
|
|
tables.add(EventHistoryDbColumns.TABLE_NAME);
|
|
tables.add(SessionsDbColumns.TABLE_NAME);
|
|
tables.add(UploadBlobsDbColumns.TABLE_NAME);
|
|
tables.add(UploadBlobEventsDbColumns.TABLE_NAME);
|
|
|
|
return tables;
|
|
}
|
|
|
|
/**
|
|
* Private helper that deletes files from older versions of the Localytics library.
|
|
* <p>
|
|
* Note: This is a private method that is only made package-accessible for unit testing.
|
|
*
|
|
* @param context application context
|
|
* @throws IllegalArgumentException if {@code context} is null
|
|
*/
|
|
/* package */static void deleteOldFiles(final Context context)
|
|
{
|
|
if (Constants.ENABLE_PARAMETER_CHECKING)
|
|
{
|
|
if (null == context)
|
|
{
|
|
throw new IllegalArgumentException("context cannot be null"); //$NON-NLS-1$
|
|
}
|
|
}
|
|
|
|
deleteDirectory(new File(context.getFilesDir(), "localytics")); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* Private helper to delete a directory, regardless of whether the directory is empty.
|
|
*
|
|
* @param directory Directory or file to delete. Cannot be null.
|
|
* @return true if deletion was successful. False if deletion failed.
|
|
*/
|
|
private static boolean deleteDirectory(final File directory)
|
|
{
|
|
if (directory.exists() && directory.isDirectory())
|
|
{
|
|
for (final String child : directory.list())
|
|
{
|
|
final boolean success = deleteDirectory(new File(directory, child));
|
|
if (!success)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// The directory is now empty so delete it
|
|
return directory.delete();
|
|
}
|
|
|
|
/**
|
|
* A private helper class to open and create the Localytics SQLite database.
|
|
*/
|
|
private static final class DatabaseHelper extends SQLiteOpenHelper
|
|
{
|
|
/**
|
|
* Constant representing the SQLite value for true
|
|
*/
|
|
private static final String SQLITE_BOOLEAN_TRUE = "1"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* Constant representing the SQLite value for false
|
|
*/
|
|
private static final String SQLITE_BOOLEAN_FALSE = "0"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* @param context Application context. Cannot be null.
|
|
* @param name File name of the database. Cannot be null or empty. A database with this name will be opened in
|
|
* {@link Context#getDatabasePath(String)}.
|
|
* @param version version of the database.
|
|
*/
|
|
public DatabaseHelper(final Context context, final String name, final int version)
|
|
{
|
|
super(context, name, null, version);
|
|
}
|
|
|
|
/**
|
|
* Initializes the tables of the database.
|
|
* <p>
|
|
* If an error occurs during initialization and an exception is thrown, {@link SQLiteDatabase#close()} will not be called
|
|
* by this method. That responsibility is left to the caller.
|
|
*
|
|
* @param db The database to perform post-creation processing on. db cannot not be null
|
|
* @throws IllegalArgumentException if db is null
|
|
*/
|
|
@Override
|
|
public void onCreate(final SQLiteDatabase db)
|
|
{
|
|
if (null == db)
|
|
{
|
|
throw new IllegalArgumentException("db cannot be null"); //$NON-NLS-1$
|
|
}
|
|
|
|
// api_keys table
|
|
db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s TEXT UNIQUE NOT NULL, %s TEXT UNIQUE NOT NULL, %s INTEGER NOT NULL CHECK (%s >= 0), %s INTEGER NOT NULL CHECK(%s IN (%s, %s)));", ApiKeysDbColumns.TABLE_NAME, ApiKeysDbColumns._ID, ApiKeysDbColumns.API_KEY, ApiKeysDbColumns.UUID, ApiKeysDbColumns.CREATED_TIME, ApiKeysDbColumns.CREATED_TIME, ApiKeysDbColumns.OPT_OUT, ApiKeysDbColumns.OPT_OUT, SQLITE_BOOLEAN_FALSE, SQLITE_BOOLEAN_TRUE)); //$NON-NLS-1$
|
|
|
|
// sessions table
|
|
db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s INTEGER REFERENCES %s(%s) NOT NULL, %s TEXT UNIQUE NOT NULL, %s INTEGER NOT NULL CHECK (%s >= 0), %s TEXT NOT NULL, %s TEXT NOT NULL, %s INTEGER NOT NULL, %s TEXT NOT NULL, %s TEXT NOT NULL, %s TEXT NOT NULL, %s TEXT NOT NULL, %s TEXT, %s TEXT, %s TEXT, %s TEXT NOT NULL, %s TEXT NOT NULL, %s TEXT, %s TEXT, %s TEXT, %s TEXT, %s TEXT, %s TEXT);", SessionsDbColumns.TABLE_NAME, SessionsDbColumns._ID, SessionsDbColumns.API_KEY_REF, ApiKeysDbColumns.TABLE_NAME, ApiKeysDbColumns._ID, SessionsDbColumns.UUID, SessionsDbColumns.SESSION_START_WALL_TIME, SessionsDbColumns.SESSION_START_WALL_TIME, SessionsDbColumns.LOCALYTICS_LIBRARY_VERSION, SessionsDbColumns.APP_VERSION, SessionsDbColumns.ANDROID_VERSION, SessionsDbColumns.ANDROID_SDK, SessionsDbColumns.DEVICE_MODEL, SessionsDbColumns.DEVICE_MANUFACTURER, SessionsDbColumns.DEVICE_ANDROID_ID_HASH, SessionsDbColumns.DEVICE_TELEPHONY_ID, SessionsDbColumns.DEVICE_TELEPHONY_ID_HASH, SessionsDbColumns.DEVICE_SERIAL_NUMBER_HASH, SessionsDbColumns.LOCALE_LANGUAGE, SessionsDbColumns.LOCALE_COUNTRY, SessionsDbColumns.NETWORK_CARRIER, SessionsDbColumns.NETWORK_COUNTRY, SessionsDbColumns.NETWORK_TYPE, SessionsDbColumns.DEVICE_COUNTRY, SessionsDbColumns.LATITUDE, SessionsDbColumns.LONGITUDE)); //$NON-NLS-1$
|
|
|
|
// events table
|
|
db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s INTEGER REFERENCES %s(%s) NOT NULL, %s TEXT UNIQUE NOT NULL, %s TEXT NOT NULL, %s INTEGER NOT NULL CHECK (%s >= 0), %s INTEGER NOT NULL CHECK (%s >= 0));", EventsDbColumns.TABLE_NAME, EventsDbColumns._ID, EventsDbColumns.SESSION_KEY_REF, SessionsDbColumns.TABLE_NAME, SessionsDbColumns._ID, EventsDbColumns.UUID, EventsDbColumns.EVENT_NAME, EventsDbColumns.REAL_TIME, EventsDbColumns.REAL_TIME, EventsDbColumns.WALL_TIME, EventsDbColumns.WALL_TIME)); //$NON-NLS-1$
|
|
|
|
// event_history table
|
|
/*
|
|
* Note: the events history should be using foreign key constrains on the upload blobs table, but that is currently
|
|
* disabled to simplify the implementation of the upload processing.
|
|
*/
|
|
db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s INTEGER REFERENCES %s(%s) NOT NULL, %s TEXT NOT NULL CHECK(%s IN (%s, %s)), %s TEXT NOT NULL, %s INTEGER);", EventHistoryDbColumns.TABLE_NAME, EventHistoryDbColumns._ID, EventHistoryDbColumns.SESSION_KEY_REF, SessionsDbColumns.TABLE_NAME, SessionsDbColumns._ID, EventHistoryDbColumns.TYPE, EventHistoryDbColumns.TYPE, Integer.valueOf(EventHistoryDbColumns.TYPE_EVENT), Integer.valueOf(EventHistoryDbColumns.TYPE_SCREEN), EventHistoryDbColumns.NAME, EventHistoryDbColumns.PROCESSED_IN_BLOB)); //$NON-NLS-1$
|
|
//db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s INTEGER REFERENCES %s(%s) NOT NULL, %s TEXT NOT NULL CHECK(%s IN (%s, %s)), %s TEXT NOT NULL, %s INTEGER REFERENCES %s(%s));", EventHistoryDbColumns.TABLE_NAME, EventHistoryDbColumns._ID, EventHistoryDbColumns.SESSION_KEY_REF, SessionsDbColumns.TABLE_NAME, SessionsDbColumns._ID, EventHistoryDbColumns.TYPE, EventHistoryDbColumns.TYPE, Integer.valueOf(EventHistoryDbColumns.TYPE_EVENT), Integer.valueOf(EventHistoryDbColumns.TYPE_SCREEN), EventHistoryDbColumns.NAME, EventHistoryDbColumns.PROCESSED_IN_BLOB, UploadBlobsDbColumns.TABLE_NAME, UploadBlobsDbColumns._ID)); //$NON-NLS-1$
|
|
|
|
// attributes table
|
|
db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s INTEGER REFERENCES %s(%s) NOT NULL, %s TEXT NOT NULL, %s TEXT NOT NULL);", AttributesDbColumns.TABLE_NAME, AttributesDbColumns._ID, AttributesDbColumns.EVENTS_KEY_REF, EventsDbColumns.TABLE_NAME, EventsDbColumns._ID, AttributesDbColumns.ATTRIBUTE_KEY, AttributesDbColumns.ATTRIBUTE_VALUE)); //$NON-NLS-1$
|
|
|
|
// upload blobs
|
|
db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s TEXT UNIQUE NOT NULL);", UploadBlobsDbColumns.TABLE_NAME, UploadBlobsDbColumns._ID, UploadBlobsDbColumns.UUID)); //$NON-NLS-1$
|
|
|
|
// upload events
|
|
db.execSQL(String.format("CREATE TABLE %s (%s INTEGER PRIMARY KEY AUTOINCREMENT, %s INTEGER REFERENCES %s(%s) NOT NULL, %s INTEGER REFERENCES %s(%s) NOT NULL);", UploadBlobEventsDbColumns.TABLE_NAME, UploadBlobEventsDbColumns._ID, UploadBlobEventsDbColumns.UPLOAD_BLOBS_KEY_REF, UploadBlobsDbColumns.TABLE_NAME, UploadBlobsDbColumns._ID, UploadBlobEventsDbColumns.EVENTS_KEY_REF, EventsDbColumns.TABLE_NAME, EventsDbColumns._ID)); //$NON-NLS-1$
|
|
}
|
|
|
|
@Override
|
|
public void onOpen(final SQLiteDatabase db)
|
|
{
|
|
super.onOpen(db);
|
|
|
|
if (Constants.IS_LOGGABLE)
|
|
{
|
|
Log.v(Constants.LOG_TAG, String.format("SQLite library version is: %s", DatabaseUtils.stringForQuery(db, "select sqlite_version()", null))); //$NON-NLS-1$//$NON-NLS-2$
|
|
}
|
|
|
|
if (!db.isReadOnly())
|
|
{
|
|
/*
|
|
* Enable foreign key support
|
|
*/
|
|
db.execSQL("PRAGMA foreign_keys = ON;"); //$NON-NLS-1$
|
|
|
|
// if (Constants.IS_LOGGABLE)
|
|
// {
|
|
// try
|
|
// {
|
|
// final String result1 = DatabaseUtils.stringForQuery(db, "PRAGMA foreign_keys;", null); //$NON-NLS-1$
|
|
// Log.v(Constants.LOG_TAG, String.format("Foreign keys support result was: %s", result1)); //$NON-NLS-1$
|
|
// }
|
|
// catch (final SQLiteDoneException e)
|
|
// {
|
|
// Log.w(Constants.LOG_TAG, e);
|
|
// }
|
|
// }
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onUpgrade(final SQLiteDatabase db, final int oldVersion, final int newVersion)
|
|
{
|
|
if (1 == oldVersion)
|
|
{
|
|
// delete stranded sessions that don't have any events
|
|
Cursor sessionsCursor = null;
|
|
try
|
|
{
|
|
sessionsCursor = db.query(SessionsDbColumns.TABLE_NAME, new String[]
|
|
{ SessionsDbColumns._ID }, null, null, null, null, null);
|
|
|
|
while (sessionsCursor.moveToNext())
|
|
{
|
|
Cursor eventsCursor = null;
|
|
try
|
|
{
|
|
String sessionId = Long.toString(sessionsCursor.getLong(sessionsCursor.getColumnIndexOrThrow(SessionsDbColumns._ID)));
|
|
eventsCursor = db.query(EventsDbColumns.TABLE_NAME, new String[]
|
|
{ EventsDbColumns._ID }, String.format("%s = ?", EventsDbColumns.SESSION_KEY_REF), new String[] //$NON-NLS-1$
|
|
{ sessionId }, null, null, null);
|
|
|
|
if (eventsCursor.getCount() == 0)
|
|
{
|
|
db.delete(SessionsDbColumns.TABLE_NAME, String.format("%s = ?", SessionsDbColumns._ID), new String[] { sessionId }); //$NON-NLS-1$
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
if (null != eventsCursor)
|
|
{
|
|
eventsCursor.close();
|
|
eventsCursor = null;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
if (null != sessionsCursor)
|
|
{
|
|
sessionsCursor.close();
|
|
sessionsCursor = null;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// @Override
|
|
// public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion)
|
|
// {
|
|
// }
|
|
}
|
|
|
|
/**
|
|
* Table for the API keys used and the opt-out preferences for each API key.
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
public static final class ApiKeysDbColumns implements BaseColumns
|
|
{
|
|
/**
|
|
* Private constructor prevents instantiation
|
|
*
|
|
* @throws UnsupportedOperationException because this class cannot be instantiated.
|
|
*/
|
|
private ApiKeysDbColumns()
|
|
{
|
|
throw new UnsupportedOperationException("This class is non-instantiable"); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* SQLite table name
|
|
*/
|
|
public static final String TABLE_NAME = "api_keys"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* The Localytics API key.
|
|
* <p>
|
|
* Constraints: This column is unique and cannot be null.
|
|
*/
|
|
public static final String API_KEY = "api_key"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* A UUID for the installation.
|
|
* <p>
|
|
* Constraints: This column is unique and cannot be null.
|
|
*/
|
|
public static final String UUID = "uuid"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code boolean}
|
|
* <p>
|
|
* A flag indicating whether the user has opted out of data collection.
|
|
* <p>
|
|
* Constraints: This column must be in the set {0, 1} and cannot be null.
|
|
*/
|
|
public static final String OPT_OUT = "opt_out"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A long representing the {@link System#currentTimeMillis()} when the row was created. Once created, this row will not be
|
|
* modified.
|
|
* <p>
|
|
* Constraints: This column must be >=0. This column cannot be null.
|
|
*/
|
|
public static final String CREATED_TIME = "created_time"; //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* Database table for the session attributes. There is a one-to-many relationship between one event in the
|
|
* {@link EventsDbColumns} table and the many attributes associated with that event.
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
public static final class AttributesDbColumns implements BaseColumns
|
|
{
|
|
/**
|
|
* Private constructor prevents instantiation
|
|
*
|
|
* @throws UnsupportedOperationException because this class cannot be instantiated.
|
|
*/
|
|
private AttributesDbColumns()
|
|
{
|
|
throw new UnsupportedOperationException("This class is non-instantiable"); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* SQLite table name
|
|
*/
|
|
public static final String TABLE_NAME = "attributes"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A one-to-many relationship with {@link EventsDbColumns#_ID}.
|
|
* <p>
|
|
* Constraints: This is a foreign key with the {@link EventsDbColumns#_ID} column. This cannot be null.
|
|
*/
|
|
public static final String EVENTS_KEY_REF = "events_key_ref"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the key name of the attribute.
|
|
* <p>
|
|
* Constraints: This cannot be null.
|
|
*/
|
|
public static final String ATTRIBUTE_KEY = "attribute_key"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the value of the attribute.
|
|
* <p>
|
|
* Constraints: This cannot be null.
|
|
*/
|
|
public static final String ATTRIBUTE_VALUE = "attribute_value"; //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
/**
|
|
* Database table for the session events. There is a one-to-many relationship between one session data entry in the
|
|
* {@link SessionsDbColumns} table and the many events associated with that session.
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
public static final class EventsDbColumns implements BaseColumns
|
|
{
|
|
/**
|
|
* Private constructor prevents instantiation
|
|
*
|
|
* @throws UnsupportedOperationException because this class cannot be instantiated.
|
|
*/
|
|
private EventsDbColumns()
|
|
{
|
|
throw new UnsupportedOperationException("This class is non-instantiable"); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* SQLite table name
|
|
*/
|
|
public static final String TABLE_NAME = "events"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A one-to-many relationship with {@link SessionsDbColumns#_ID}.
|
|
* <p>
|
|
* Constraints: This is a foreign key with the {@link SessionsDbColumns#_ID} column. This cannot be null.
|
|
*/
|
|
public static final String SESSION_KEY_REF = "session_key_ref"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Unique ID of the event, as generated from {@link java.util.UUID}.
|
|
* <p>
|
|
* Constraints: This is unique and cannot be null.
|
|
*/
|
|
public static final String UUID = "uuid"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the name of the event.
|
|
* <p>
|
|
* Constraints: This cannot be null.
|
|
*/
|
|
public static final String EVENT_NAME = "event_name"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A long representing the {@link android.os.SystemClock#elapsedRealtime()} when the event occurred.
|
|
* <p>
|
|
* Constraints: This column must be >=0. This column cannot be null.
|
|
*/
|
|
public static final String REAL_TIME = "real_time"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A long representing the {@link System#currentTimeMillis()} when the event occurred.
|
|
* <p>
|
|
* Constraints: This column must be >=0. This column cannot be null.
|
|
*/
|
|
public static final String WALL_TIME = "wall_time"; //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
/**
|
|
* Database table for tracking the history of events and screens. There is a one-to-many relationship between one session data
|
|
* entry in the {@link SessionsDbColumns} table and the many historical events associated with that session.
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
public static final class EventHistoryDbColumns implements BaseColumns
|
|
{
|
|
/**
|
|
* Private constructor prevents instantiation
|
|
*
|
|
* @throws UnsupportedOperationException because this class cannot be instantiated.
|
|
*/
|
|
private EventHistoryDbColumns()
|
|
{
|
|
throw new UnsupportedOperationException("This class is non-instantiable"); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* SQLite table name
|
|
*/
|
|
public static final String TABLE_NAME = "event_history"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A one-to-many relationship with {@link SessionsDbColumns#_ID}.
|
|
* <p>
|
|
* Constraints: This is a foreign key with the {@link SessionsDbColumns#_ID} column. This cannot be null.
|
|
*/
|
|
public static final String SESSION_KEY_REF = "session_key_ref"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Unique ID of the event, as generated from {@link java.util.UUID}.
|
|
* <p>
|
|
* Constraints: This is unique and cannot be null.
|
|
*/
|
|
public static final String TYPE = "type"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the name of the screen or event.
|
|
* <p>
|
|
* Constraints: This cannot be null.
|
|
*/
|
|
public static final String NAME = "name"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code boolean}
|
|
* <p>
|
|
* Foreign key to the upload blob that this event was processed in. May be null indicating that this event wasn't
|
|
* processed yet.
|
|
*/
|
|
public static final String PROCESSED_IN_BLOB = "processed_in_blob"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* Type value for {@link #TYPE} indicates an event event.
|
|
*/
|
|
public static final int TYPE_EVENT = 0;
|
|
|
|
/**
|
|
* Type value for {@link #TYPE} that indicates a screen event.
|
|
*/
|
|
public static final int TYPE_SCREEN = 1;
|
|
}
|
|
|
|
/**
|
|
* Database table for the session data. There is a one-to-many relationship between one API key entry in the
|
|
* {@link ApiKeysDbColumns} table and many sessions for that API key.
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
public static final class SessionsDbColumns implements BaseColumns
|
|
{
|
|
/**
|
|
* Private constructor prevents instantiation
|
|
*
|
|
* @throws UnsupportedOperationException because this class cannot be instantiated.
|
|
*/
|
|
private SessionsDbColumns()
|
|
{
|
|
throw new UnsupportedOperationException("This class is non-instantiable"); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* SQLite table name
|
|
*/
|
|
public static final String TABLE_NAME = "sessions"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A one-to-one relationship with {@link ApiKeysDbColumns#_ID}.
|
|
* <p>
|
|
* Constraints: This is a foreign key with the {@link ApiKeysDbColumns#_ID} column. This cannot be null.
|
|
*/
|
|
public static final String API_KEY_REF = "api_key_ref"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Unique ID of the event, as generated from {@link java.util.UUID}.
|
|
* <p>
|
|
* Constraints: This is unique and cannot be null.
|
|
*/
|
|
public static final String UUID = "uuid"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* The wall time when the session started.
|
|
* <p>
|
|
* Constraints: This column must be >=0. This column cannot be null.
|
|
*/
|
|
/*
|
|
* Note: While this same information is encoded in {@link EventsDbColumns#WALL_TIME} for the session open event, that row
|
|
* may not be available when an upload occurs and the upload needs to compute the duration of the session.
|
|
*/
|
|
public static final String SESSION_START_WALL_TIME = "session_start_wall_time"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Version of the Localytics client library.
|
|
*
|
|
* @see Constants#LOCALYTICS_CLIENT_LIBRARY_VERSION
|
|
*/
|
|
public static final String LOCALYTICS_LIBRARY_VERSION = "localytics_library_version"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the app's versionName
|
|
* <p>
|
|
* Constraints: This cannot be null.
|
|
*/
|
|
public static final String APP_VERSION = "app_version"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the version of Android
|
|
* <p>
|
|
* Constraints: This cannot be null.
|
|
*/
|
|
public static final String ANDROID_VERSION = "android_version"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code int}
|
|
* <p>
|
|
* Integer the Android SDK
|
|
* <p>
|
|
* Constraints: Must be an integer and cannot be null.
|
|
*
|
|
* @see android.os.Build.VERSION#SDK
|
|
*/
|
|
public static final String ANDROID_SDK = "android_sdk"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the device model
|
|
* <p>
|
|
* Constraints: None
|
|
*
|
|
* @see android.os.Build#MODEL
|
|
*/
|
|
public static final String DEVICE_MODEL = "device_model"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the device manufacturer
|
|
* <p>
|
|
* Constraints: None
|
|
*
|
|
* @see android.os.Build#MANUFACTURER
|
|
*/
|
|
public static final String DEVICE_MANUFACTURER = "device_manufacturer"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing a hash of the device Android ID
|
|
* <p>
|
|
* Constraints: None
|
|
*
|
|
* @see android.provider.Settings.Secure#ANDROID_ID
|
|
*/
|
|
public static final String DEVICE_ANDROID_ID_HASH = "device_android_id_hash"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing the telephony ID of the device. May be null for non-telephony devices. May also be null if the
|
|
* parent application doesn't have {@link android.Manifest.permission#READ_PHONE_STATE}.
|
|
* <p>
|
|
* Constraints: None
|
|
*
|
|
* @see android.telephony.TelephonyManager#getDeviceId()
|
|
*/
|
|
public static final String DEVICE_TELEPHONY_ID = "device_telephony_id"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing a hash of the telephony ID of the device. May be null for non-telephony devices. May also be null
|
|
* if the parent application doesn't have {@link android.Manifest.permission#READ_PHONE_STATE}.
|
|
* <p>
|
|
* Constraints: None
|
|
*
|
|
* @see android.telephony.TelephonyManager#getDeviceId()
|
|
*/
|
|
public static final String DEVICE_TELEPHONY_ID_HASH = "device_telephony_id_hash"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* String representing a hash of the the serial number of the device. May be null for some telephony devices.
|
|
* <p>
|
|
* Constraints: None
|
|
*/
|
|
public static final String DEVICE_SERIAL_NUMBER_HASH = "device_serial_number_hash"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Represents the locale language of the device.
|
|
* <p>
|
|
* Constraints: Cannot be null.
|
|
*/
|
|
public static final String LOCALE_LANGUAGE = "locale_language"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Represents the locale country of the device.
|
|
* <p>
|
|
* Constraints: Cannot be null.
|
|
*/
|
|
public static final String LOCALE_COUNTRY = "locale_country"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Represents the locale country of the device, according to the SIM card.
|
|
* <p>
|
|
* Constraints: Cannot be null.
|
|
*/
|
|
public static final String DEVICE_COUNTRY = "device_country"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Represents the network carrier of the device. May be null for non-telephony devices.
|
|
* <p>
|
|
* Constraints: None
|
|
*/
|
|
public static final String NETWORK_CARRIER = "network_carrier"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Represents the network country of the device. May be null for non-telephony devices.
|
|
* <p>
|
|
* Constraints: None
|
|
*/
|
|
public static final String NETWORK_COUNTRY = "network_country"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Represents the primary network connection type for the device. This could be any type, including Wi-Fi, various cell
|
|
* networks, Ethernet, etc.
|
|
* <p>
|
|
* Constraints: None
|
|
*
|
|
* @see android.telephony.TelephonyManager
|
|
*/
|
|
public static final String NETWORK_TYPE = "network_type"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code double}
|
|
* <p>
|
|
* Represents the latitude of the device. May be null if no longitude is known.
|
|
* <p>
|
|
* Constraints: None
|
|
*/
|
|
public static final String LATITUDE = "latitude"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code double}
|
|
* <p>
|
|
* Represents the longitude of the device. May be null if no longitude is known.
|
|
* <p>
|
|
* Constraints: None
|
|
*/
|
|
public static final String LONGITUDE = "longitude"; //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
/**
|
|
* Database table for the events associated with a given upload blob. There is a one-to-many relationship between one upload
|
|
* blob in the {@link UploadBlobsDbColumns} table and the blob events. There is a one-to-one relationship between each blob
|
|
* event entry and the actual events in the {@link EventsDbColumns} table. *
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
public static final class UploadBlobEventsDbColumns implements BaseColumns
|
|
{
|
|
/**
|
|
* Private constructor prevents instantiation
|
|
*
|
|
* @throws UnsupportedOperationException because this class cannot be instantiated.
|
|
*/
|
|
private UploadBlobEventsDbColumns()
|
|
{
|
|
throw new UnsupportedOperationException("This class is non-instantiable"); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* SQLite table name
|
|
*/
|
|
public static final String TABLE_NAME = "upload_blob_events"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* A one-to-many relationship with {@link UploadBlobsDbColumns#_ID}.
|
|
* <p>
|
|
* Constraints: This is a foreign key with the {@link UploadBlobsDbColumns#_ID} column. This cannot be null.
|
|
*/
|
|
public static final String UPLOAD_BLOBS_KEY_REF = "upload_blobs_key_ref"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code long}
|
|
* <p>
|
|
* A one-to-one relationship with {@link EventsDbColumns#_ID}.
|
|
* <p>
|
|
* Constraints: This is a foreign key with the {@link EventsDbColumns#_ID} column. This cannot be null.
|
|
*/
|
|
public static final String EVENTS_KEY_REF = "events_key_ref"; //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* Database table for the upload blobs. Logically, a blob owns many events. In terms of the implementation, some indirection
|
|
* is introduced by a blob having a one-to-many relationship with {@link UploadBlobsDbColumns} and
|
|
* {@link UploadBlobsDbColumns} having a one-to-one relationship with {@link EventsDbColumns}
|
|
* <p>
|
|
* This is not a public API.
|
|
*/
|
|
public static final class UploadBlobsDbColumns implements BaseColumns
|
|
{
|
|
/**
|
|
* Private constructor prevents instantiation
|
|
*
|
|
* @throws UnsupportedOperationException because this class cannot be instantiated.
|
|
*/
|
|
private UploadBlobsDbColumns()
|
|
{
|
|
throw new UnsupportedOperationException("This class is non-instantiable"); //$NON-NLS-1$
|
|
}
|
|
|
|
/**
|
|
* SQLite table name
|
|
*/
|
|
public static final String TABLE_NAME = "upload_blobs"; //$NON-NLS-1$
|
|
|
|
/**
|
|
* TYPE: {@code String}
|
|
* <p>
|
|
* Unique ID of the upload blob, as generated from {@link java.util.UUID}.
|
|
* <p>
|
|
* Constraints: This is unique and cannot be null.
|
|
*/
|
|
public static final String UUID = "uuid"; //$NON-NLS-1$
|
|
|
|
}
|
|
}
|