mirror of
https://github.com/home-assistant/iOS.git
synced 2026-06-24 10:49:41 -05:00
<!-- Thank you for submitting a Pull Request and helping to improve Home Assistant. Please complete the following sections to help the processing and review of your changes. Please do not delete anything from this template. --> ## Summary <!-- Provide a brief summary of the changes you have made and most importantly what they aim to achieve --> This PR removes the previous foundation for kiosk mode and add a simplified version fully made for SwiftUI. ## Screenshots <!-- If this is a user-facing change not in the frontend, please include screenshots in light and dark mode. --> https://github.com/user-attachments/assets/d6f1ba1e-6806-4a72-9fb4-c326f3479ad4 ## Link to pull request in Documentation repository <!-- Pull requests that add, change or remove functionality must have a corresponding pull request in the Companion App Documentation repository (https://github.com/home-assistant/companion.home-assistant). Please add the number of this pull request after the "#" --> Documentation: home-assistant/companion.home-assistant#1357 ## Any other notes <!-- If there is any other information of note, like if this Pull Request is part of a bigger change, please include it here. --> --------- Co-authored-by: Claude Opus 4.8 <noreply@anthropic.com>
139 lines
5.4 KiB
Swift
139 lines
5.4 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 = {
|
|
let database: DatabaseQueue
|
|
do {
|
|
database = try DatabaseQueue(path: databasePath())
|
|
#if targetEnvironment(simulator)
|
|
print("GRDB App database is stored at \(AppConstants.appGRDBFile.description)")
|
|
#endif
|
|
} catch {
|
|
Current.Log.error("Failed to initialize GRDB, error: \(error.localizedDescription)")
|
|
// Fallback to in-memory database so extensions don't crash
|
|
do {
|
|
database = try DatabaseQueue()
|
|
Current.Log.error("Using in-memory GRDB database as fallback")
|
|
} catch {
|
|
fatalError("Failed to create even an in-memory GRDB database: \(error.localizedDescription)")
|
|
}
|
|
}
|
|
|
|
setupSchema(database: database)
|
|
return database
|
|
}()
|
|
|
|
private static func setupSchema(database: DatabaseQueue) {
|
|
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)
|
|
}
|
|
|
|
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(),
|
|
AppIconShortcutConfigTable(),
|
|
AssistPipelinesTable(),
|
|
ServerInfoMirrorTable(),
|
|
DisplayEntityRegistryTable(),
|
|
AppDeviceRegistryTable(),
|
|
AppPanelTable(),
|
|
CustomWidgetTable(),
|
|
AppAreaTable(),
|
|
HomeViewConfigurationTable(),
|
|
AssistConfigurationTable(),
|
|
AllowedTagTable(),
|
|
KioskSettingsTable(),
|
|
]
|
|
}
|
|
|
|
internal static func deleteOldTables(database: DatabaseQueue) {
|
|
/*
|
|
Tables that existed in earlier versions and are no longer used:
|
|
- clientEvent: used to be saved in GRDB, but because of a problem of one process holding
|
|
a lock on the database and causing crash 0xdead10cc it is now saved as a json file.
|
|
More information: https://github.com/groue/GRDB.swift/issues/1626#issuecomment-2623927815
|
|
- appEntityRegistryListForDisplay and entityRegistry: both replaced by `displayEntityRegistry`,
|
|
which is sourced from config/entity_registry/list_for_display. ("entityRegistry" was the
|
|
former full-registry table; it has no enum case anymore, hence the string literal.)
|
|
*/
|
|
let obsoleteTables = [
|
|
GRDBDatabaseTable.clientEvent.rawValue,
|
|
GRDBDatabaseTable.appEntityRegistryListForDisplay.rawValue,
|
|
"entityRegistry",
|
|
]
|
|
for tableName in obsoleteTables {
|
|
do {
|
|
try database.write { db in
|
|
try db.drop(table: tableName)
|
|
}
|
|
} catch {
|
|
Current.Log.verbose(
|
|
"Failed or not needed to drop obsolete GRDB table \(tableName), error: \(error.localizedDescription)"
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|