Files
iOS/Sources/Shared/Database/GRDB+Initialization.swift
Bruno Pantaleão Gonçalves 12c8d5ccec Reworked the foundation for Kiosk Mode (#4805)
<!-- 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>
2026-06-23 10:57:42 +02:00

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)
}
}
}
}
}