npx claudepluginhub davidortinau/maui-skills --plugin maui-skillsThis skill uses the workspace's default tool permissions.
For full service implementation, constants, data model templates, and common patterns, see `references/sqlite-database-api.md`.
Implements secure storage in .NET MAUI apps using SecureStorage.Default APIs (SetAsync/GetAsync/Remove), platform setups for Android backups/iOS Keychain/Windows limits, pitfalls, and DI wrapper.
Guides Swift developers on SQLiteData library for type-safe SQLite persistence, @Table models, @FetchAll/@FetchOne queries, writes, migrations, and CloudKit sync.
Implements secure SQLite databases for Tauri/desktop apps using rusqlite/sea-query: parameterized queries against SQL injection, migrations, FTS5 search, WAL mode, indexing.
Share bugs, ideas, or general feedback.
For full service implementation, constants, data model templates, and common patterns, see references/sqlite-database-api.md.
<!-- ❌ WRONG — these are different libraries with incompatible APIs -->
<PackageReference Include="Microsoft.Data.Sqlite" />
<PackageReference Include="sqlite-net" />
<PackageReference Include="SQLitePCL.raw" />
<!-- ✅ CORRECT — sqlite-net-pcl by praeclarum + its bundle -->
<PackageReference Include="sqlite-net-pcl" Version="1.9.*" />
<PackageReference Include="SQLitePCLRaw.bundle_green" Version="2.1.*" />
Environment.GetFolderPath for Database Path// ❌ Not cross-platform safe — fails on some MAUI targets
var path = Path.Combine(Environment.GetFolderPath(
Environment.SpecialFolder.LocalApplicationData), "app.db3");
// ✅ Use FileSystem.AppDataDirectory for all MAUI platforms
var path = Path.Combine(FileSystem.AppDataDirectory, "app.db3");
SQLiteAsyncConnection is not thread-safe for multiple instances pointing at the same file. Use a single instance via DI singleton:
// ❌ Creating new connections per request
public async Task<List<Item>> GetItems()
{
var db = new SQLiteAsyncConnection(Constants.DatabasePath);
return await db.Table<Item>().ToListAsync();
}
// ✅ Lazy singleton — one connection, created once
private SQLiteAsyncConnection? _database;
private async Task<SQLiteAsyncConnection> GetDatabaseAsync()
{
if (_database is not null) return _database;
_database = new SQLiteAsyncConnection(Constants.DatabasePath, Constants.Flags);
await _database.ExecuteAsync("PRAGMA journal_mode=WAL;");
await _database.CreateTableAsync<TodoItem>();
return _database;
}
Without WAL, readers block writers. Always enable it at initialization:
await _database.ExecuteAsync("PRAGMA journal_mode=WAL;");
// ❌ Moving/deleting while connection is open — data corruption
File.Delete(Constants.DatabasePath);
// ✅ Always close first
await databaseService.CloseConnectionAsync();
if (File.Exists(Constants.DatabasePath))
File.Delete(Constants.DatabasePath);
| Platform | Pitfall |
|---|---|
| iOS | FileSystem.AppDataDirectory is iCloud-backed — use FileSystem.CacheDirectory to exclude DB from iCloud backup |
| All | Multiple SQLiteAsyncConnection instances to same file → data corruption |
| All | No WAL → readers block writers, poor concurrent performance |
| All | File operations on open DB → corruption |
| Question | Recommendation |
|---|---|
| DI lifetime? | Singleton — one connection, WAL handles concurrent reads |
| WAL mode? | Always enable — no reason not to on mobile |
| Database path? | FileSystem.AppDataDirectory — never Environment.GetFolderPath |
| Save pattern? | Check Id != 0 → Update, else Insert |
| Multiple tables? | Add all CreateTableAsync<T>() calls in lazy init |
| Need to export/backup? | Close connection first, then File.Copy |
RunInTransactionAsync[Indexed] to frequently queried columns — especially foreign keysToListAsync() on large tables — use Where() filtering and paginationQueryAsync<T> is faster than chained LINQ for joinssqlite-net-pcl + SQLitePCLRaw.bundle_green (not Microsoft.Data.Sqlite)FileSystem.AppDataDirectory[PrimaryKey, AutoIncrement]DatabaseService with lazy async init patternPRAGMA journal_mode=WALDatabaseService registered as singleton in DI