mirror of
https://github.com/home-assistant/iOS.git
synced 2026-04-12 15:26:45 -05:00
120 lines
4.5 KiB
Swift
120 lines
4.5 KiB
Swift
import Foundation
|
|
import GRDB
|
|
|
|
public extension DatabaseQueue {
|
|
// Following GRDB cocnurrency rules, we have just one database instance
|
|
// https://swiftpackageindex.com/groue/grdb.swift/v6.29.3/documentation/grdb/concurrency#Concurrency-Rules
|
|
static var appDatabase: DatabaseQueue = {
|
|
do {
|
|
let database = try DatabaseQueue(path: databasePath())
|
|
|
|
// Create tables if needed
|
|
for table in DatabaseQueue.tables() {
|
|
do {
|
|
try table.createIfNeeded(database: database)
|
|
} catch {
|
|
let className = String(describing: type(of: table))
|
|
let errorMessage = "Failed create GRDB table \(className), error: \(error.localizedDescription)"
|
|
Current.clientEventStore.addEvent(ClientEvent(text: errorMessage, type: .database))
|
|
}
|
|
}
|
|
DatabaseQueue.deleteOldTables(database: database)
|
|
|
|
#if targetEnvironment(simulator)
|
|
print("GRDB App database is stored at \(AppConstants.appGRDBFile.description)")
|
|
#endif
|
|
return database
|
|
} catch {
|
|
let errorMessage = "Failed to initialize GRDB, error: \(error.localizedDescription)"
|
|
Current.Log.error(errorMessage)
|
|
fatalError(errorMessage)
|
|
}
|
|
}()
|
|
|
|
static func databasePath() -> String {
|
|
// Path for tests
|
|
if ProcessInfo.processInfo.environment["XCTestConfigurationFilePath"] != nil {
|
|
let tempDirectory = NSTemporaryDirectory()
|
|
return (tempDirectory as NSString).appendingPathComponent("test_database.sqlite")
|
|
} else {
|
|
// Path for App use
|
|
return AppConstants.appGRDBFile.path
|
|
}
|
|
}
|
|
|
|
internal static func tables() -> [DatabaseTableProtocol] {
|
|
[
|
|
HAppEntityTable(),
|
|
WatchConfigTable(),
|
|
CarPlayConfigTable(),
|
|
AssistPipelinesTable(),
|
|
ServerInfoMirrorTable(),
|
|
AppEntityRegistryListForDisplayTable(),
|
|
AppEntityRegistryTable(),
|
|
AppDeviceRegistryTable(),
|
|
AppPanelTable(),
|
|
CustomWidgetTable(),
|
|
AppAreaTable(),
|
|
HomeViewConfigurationTable(),
|
|
CameraListConfigurationTable(),
|
|
AssistConfigurationTable(),
|
|
KioskSettingsTable(),
|
|
]
|
|
}
|
|
|
|
internal static func deleteOldTables(database: DatabaseQueue) {
|
|
do {
|
|
/*
|
|
ClientEvent used to be saved in GRDB, but because of a problem of one process holding
|
|
lock on the database and causing crash 0xdead10cc now it is saved as a json file
|
|
More information: https://github.com/groue/GRDB.swift/issues/1626#issuecomment-2623927815
|
|
*/
|
|
try database.write { db in
|
|
try db.drop(table: GRDBDatabaseTable.clientEvent.rawValue)
|
|
}
|
|
} catch {
|
|
let errorMessage =
|
|
"Failed or not needed delete client event GRDB info, error: \(error.localizedDescription)"
|
|
Current.Log.verbose(errorMessage)
|
|
}
|
|
}
|
|
}
|
|
|
|
protocol DatabaseTableProtocol {
|
|
/// The name of the database table
|
|
var tableName: String { get }
|
|
|
|
/// The list of column names defined for this table
|
|
var definedColumns: [String] { get }
|
|
|
|
/// Creates the table if it doesn't exist, or migrates it if it does
|
|
func createIfNeeded(database: DatabaseQueue) throws
|
|
}
|
|
|
|
extension DatabaseTableProtocol {
|
|
/// Migrates the table by adding new columns and removing obsolete columns
|
|
func migrateColumns(database: DatabaseQueue) throws {
|
|
try database.write { db in
|
|
let existingColumns = try db.columns(in: tableName)
|
|
let definedColumnSet = Set(definedColumns)
|
|
|
|
// Add new columns that don't exist yet
|
|
for columnName in definedColumns {
|
|
let shouldCreateColumn = !existingColumns.contains { $0.name == columnName }
|
|
if shouldCreateColumn {
|
|
try db.alter(table: tableName) { tableAlteration in
|
|
tableAlteration.add(column: columnName)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove columns that are no longer defined
|
|
for existingColumn in existingColumns where !definedColumnSet.contains(existingColumn.name) {
|
|
try db.alter(table: tableName) { tableAlteration in
|
|
tableAlteration.drop(column: existingColumn.name)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|