2017-04-17 13:37:52 +08:00
|
|
|
|
// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
|
|
|
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
|
|
|
|
|
|
|
|
|
using System;
|
2017-04-17 16:43:48 +08:00
|
|
|
|
using System.Collections.Generic;
|
2017-07-27 14:06:10 +08:00
|
|
|
|
using System.Linq;
|
2017-04-17 16:43:48 +08:00
|
|
|
|
using System.Linq.Expressions;
|
2017-04-17 13:37:52 +08:00
|
|
|
|
using osu.Framework.Logging;
|
2017-07-26 19:22:02 +08:00
|
|
|
|
using osu.Framework.Platform;
|
2017-04-17 13:37:52 +08:00
|
|
|
|
using SQLite.Net;
|
2017-04-17 16:43:48 +08:00
|
|
|
|
using SQLiteNetExtensions.Extensions;
|
2017-04-17 13:37:52 +08:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Database
|
|
|
|
|
{
|
2017-07-27 15:56:41 +08:00
|
|
|
|
public abstract class DatabaseBackedStore
|
2017-04-17 13:37:52 +08:00
|
|
|
|
{
|
2017-07-26 19:22:02 +08:00
|
|
|
|
protected readonly Storage Storage;
|
|
|
|
|
protected readonly SQLiteConnection Connection;
|
2017-04-17 13:37:52 +08:00
|
|
|
|
|
2017-07-27 19:38:35 +08:00
|
|
|
|
protected virtual int StoreVersion => 1;
|
|
|
|
|
|
2017-07-27 15:56:41 +08:00
|
|
|
|
protected DatabaseBackedStore(SQLiteConnection connection, Storage storage = null)
|
2017-04-17 13:37:52 +08:00
|
|
|
|
{
|
2017-07-26 19:22:02 +08:00
|
|
|
|
Storage = storage;
|
2017-04-17 13:37:52 +08:00
|
|
|
|
Connection = connection;
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
Prepare();
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2017-04-17 16:43:48 +08:00
|
|
|
|
Logger.Error(e, $@"Failed to initialise the {GetType()}! Trying again with a clean database...");
|
2017-04-17 18:44:03 +08:00
|
|
|
|
Prepare(true);
|
2017-04-17 13:37:52 +08:00
|
|
|
|
}
|
2017-07-27 19:38:35 +08:00
|
|
|
|
|
|
|
|
|
checkMigrations();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void checkMigrations()
|
|
|
|
|
{
|
|
|
|
|
var storeName = GetType().Name;
|
|
|
|
|
|
|
|
|
|
var reportedVersion = Connection.Table<StoreVersion>().FirstOrDefault(s => s.StoreName == storeName) ?? new StoreVersion
|
|
|
|
|
{
|
|
|
|
|
StoreName = storeName,
|
|
|
|
|
Version = 0
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (reportedVersion.Version != StoreVersion)
|
|
|
|
|
PerformMigration(reportedVersion.Version, reportedVersion.Version = StoreVersion);
|
|
|
|
|
|
|
|
|
|
Connection.InsertOrReplace(reportedVersion);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected virtual void PerformMigration(int currentVersion, int newVersion)
|
|
|
|
|
{
|
2017-04-17 13:37:52 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Prepare this database for use.
|
|
|
|
|
/// </summary>
|
2017-04-17 18:44:03 +08:00
|
|
|
|
protected abstract void Prepare(bool reset = false);
|
2017-04-17 13:37:52 +08:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Reset this database to a default state. Undo all changes to database and storage backings.
|
|
|
|
|
/// </summary>
|
2017-04-17 18:44:03 +08:00
|
|
|
|
public void Reset() => Prepare(true);
|
2017-04-17 16:43:48 +08:00
|
|
|
|
|
2017-07-27 14:06:10 +08:00
|
|
|
|
|
|
|
|
|
public TableQuery<T> Query<T>(Expression<Func<T, bool>> filter = null) where T : class
|
2017-04-17 16:43:48 +08:00
|
|
|
|
{
|
2017-07-27 14:06:10 +08:00
|
|
|
|
checkType(typeof(T));
|
|
|
|
|
|
|
|
|
|
var query = Connection.Table<T>();
|
|
|
|
|
|
|
|
|
|
if (filter != null)
|
|
|
|
|
query = query.Where(filter);
|
|
|
|
|
|
|
|
|
|
return query;
|
2017-04-17 16:43:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-11 17:14:48 +08:00
|
|
|
|
/// <summary>
|
2017-07-27 14:06:10 +08:00
|
|
|
|
/// Query and populate results.
|
2017-05-11 17:14:48 +08:00
|
|
|
|
/// </summary>
|
2017-07-31 17:41:54 +08:00
|
|
|
|
/// <param name="filter">An filter to refine results.</param>
|
2017-07-27 14:06:10 +08:00
|
|
|
|
/// <returns></returns>
|
2017-07-31 17:41:54 +08:00
|
|
|
|
public List<T> QueryAndPopulate<T>(Expression<Func<T, bool>> filter)
|
2017-04-17 16:43:48 +08:00
|
|
|
|
where T : class
|
|
|
|
|
{
|
2017-07-27 14:06:10 +08:00
|
|
|
|
checkType(typeof(T));
|
|
|
|
|
|
|
|
|
|
return Connection.GetAllWithChildren(filter, true);
|
2017-04-17 16:43:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 14:06:10 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Populate a database-backed item.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="item"></param>
|
|
|
|
|
/// <param name="recursive">Whether population should recurse beyond a single level.</param>
|
|
|
|
|
public void Populate<T>(T item, bool recursive = true)
|
2017-04-17 16:43:48 +08:00
|
|
|
|
{
|
2017-07-27 14:06:10 +08:00
|
|
|
|
checkType(item.GetType());
|
2017-04-17 16:43:48 +08:00
|
|
|
|
|
|
|
|
|
Connection.GetChildren(item, recursive);
|
2017-07-27 14:06:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void checkType(Type type)
|
|
|
|
|
{
|
|
|
|
|
if (!ValidTypes.Contains(type))
|
2017-07-27 15:56:41 +08:00
|
|
|
|
throw new InvalidOperationException($"The requested operation specified a type of {type}, which is invalid for this {nameof(DatabaseBackedStore)}.");
|
2017-04-17 16:43:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected abstract Type[] ValidTypes { get; }
|
2017-04-17 13:37:52 +08:00
|
|
|
|
}
|
2017-07-27 14:06:10 +08:00
|
|
|
|
}
|