Compare commits
6 Commits
master
...
sql-absurd
Author | SHA1 | Date |
---|---|---|
|
d9ce884513 | 5 months ago |
|
a1a1543ae1 | 5 months ago |
|
93591a5815 | 5 months ago |
|
b30c4c8b30 | 5 months ago |
|
1f9db0ba94 | 5 months ago |
|
bdc2d71d3c | 5 months ago |
43 changed files with 6972 additions and 2266 deletions
@ -0,0 +1,172 @@ |
|||
--- |
|||
description: |
|||
globs: |
|||
alwaysApply: true |
|||
--- |
|||
# @capacitor-community/sqlite MDC Ruleset |
|||
|
|||
## Project Overview |
|||
This ruleset is for the `@capacitor-community/sqlite` plugin, a Capacitor community plugin that provides native and Electron SQLite database functionality with encryption support. |
|||
|
|||
## Key Features |
|||
- Native SQLite database support for iOS, Android, and Electron |
|||
- Database encryption support using SQLCipher (Native) and better-sqlite3-multiple-ciphers (Electron) |
|||
- Biometric authentication support |
|||
- Cross-platform database operations |
|||
- JSON import/export capabilities |
|||
- Database migration support |
|||
- Sync table functionality |
|||
|
|||
## Platform Support Matrix |
|||
|
|||
### Core Database Operations |
|||
| Operation | Android | iOS | Electron | Web | |
|||
|-----------|---------|-----|----------|-----| |
|||
| Create Connection (RW) | ✅ | ✅ | ✅ | ✅ | |
|||
| Create Connection (RO) | ✅ | ✅ | ✅ | ❌ | |
|||
| Open DB (non-encrypted) | ✅ | ✅ | ✅ | ✅ | |
|||
| Open DB (encrypted) | ✅ | ✅ | ✅ | ❌ | |
|||
| Execute/Query | ✅ | ✅ | ✅ | ✅ | |
|||
| Import/Export JSON | ✅ | ✅ | ✅ | ✅ | |
|||
|
|||
### Security Features |
|||
| Feature | Android | iOS | Electron | Web | |
|||
|---------|---------|-----|----------|-----| |
|||
| Encryption | ✅ | ✅ | ✅ | ❌ | |
|||
| Biometric Auth | ✅ | ✅ | ✅ | ❌ | |
|||
| Secret Management | ✅ | ✅ | ✅ | ❌ | |
|||
|
|||
## Configuration Requirements |
|||
|
|||
### Base Configuration |
|||
```typescript |
|||
// capacitor.config.ts |
|||
{ |
|||
plugins: { |
|||
CapacitorSQLite: { |
|||
iosDatabaseLocation: 'Library/CapacitorDatabase', |
|||
iosIsEncryption: true, |
|||
iosKeychainPrefix: 'your-app-prefix', |
|||
androidIsEncryption: true, |
|||
electronIsEncryption: true |
|||
} |
|||
} |
|||
} |
|||
``` |
|||
|
|||
### Platform-Specific Requirements |
|||
|
|||
#### Android |
|||
- Minimum SDK: 23 |
|||
- Target SDK: 35 |
|||
- Required Gradle JDK: 21 |
|||
- Required Android Gradle Plugin: 8.7.2 |
|||
- Required manifest settings for backup prevention |
|||
- Required data extraction rules |
|||
|
|||
#### iOS |
|||
- No additional configuration needed beyond base setup |
|||
- Supports biometric authentication |
|||
- Uses keychain for encryption |
|||
|
|||
#### Electron |
|||
Required dependencies: |
|||
```json |
|||
{ |
|||
"dependencies": { |
|||
"better-sqlite3-multiple-ciphers": "latest", |
|||
"electron-json-storage": "latest", |
|||
"jszip": "latest", |
|||
"node-fetch": "2.6.7", |
|||
"crypto": "latest", |
|||
"crypto-js": "latest" |
|||
} |
|||
} |
|||
``` |
|||
|
|||
#### Web |
|||
- Requires `sql.js` and `jeep-sqlite` |
|||
- Manual copy of `sql-wasm.wasm` to assets folder |
|||
- Framework-specific asset placement: |
|||
- Angular: `src/assets/` |
|||
- Vue/React: `public/assets/` |
|||
|
|||
## Best Practices |
|||
|
|||
### Database Operations |
|||
1. Always close connections after use |
|||
2. Use transactions for multiple operations |
|||
3. Implement proper error handling |
|||
4. Use prepared statements for queries |
|||
5. Implement proper database versioning |
|||
|
|||
### Security |
|||
1. Always use encryption for sensitive data |
|||
2. Implement proper secret management |
|||
3. Use biometric authentication when available |
|||
4. Follow platform-specific security guidelines |
|||
|
|||
### Performance |
|||
1. Use appropriate indexes |
|||
2. Implement connection pooling |
|||
3. Use transactions for bulk operations |
|||
4. Implement proper database cleanup |
|||
|
|||
## Common Issues and Solutions |
|||
|
|||
### Android |
|||
- Build data properties conflict: Add to `app/build.gradle`: |
|||
```gradle |
|||
packagingOptions { |
|||
exclude 'build-data.properties' |
|||
} |
|||
``` |
|||
|
|||
### Electron |
|||
- Node-fetch version must be ≤2.6.7 |
|||
- For Capacitor Electron v5: |
|||
- Use Electron@25.8.4 |
|||
- Add `"skipLibCheck": true` to tsconfig.json |
|||
|
|||
### Web |
|||
- Ensure proper WASM file placement |
|||
- Handle browser compatibility |
|||
- Implement proper fallbacks |
|||
|
|||
## Version Compatibility |
|||
- Requires Node.js ≥16.0.0 |
|||
- Compatible with Capacitor ≥7.0.0 |
|||
- Supports TypeScript 4.1.5+ |
|||
|
|||
## Testing Requirements |
|||
- Unit tests for database operations |
|||
- Platform-specific integration tests |
|||
- Encryption/decryption tests |
|||
- Biometric authentication tests |
|||
- Migration tests |
|||
- Sync functionality tests |
|||
|
|||
## Documentation |
|||
- API Documentation: `/docs/API.md` |
|||
- Connection API: `/docs/APIConnection.md` |
|||
- DB Connection API: `/docs/APIDBConnection.md` |
|||
- Release Notes: `/docs/info_releases.md` |
|||
- Changelog: `CHANGELOG.md` |
|||
|
|||
## Contributing Guidelines |
|||
- Follow Ionic coding standards |
|||
- Use provided linting and formatting tools |
|||
- Maintain platform compatibility |
|||
- Update documentation |
|||
- Add appropriate tests |
|||
- Follow semantic versioning |
|||
|
|||
## Maintenance |
|||
- Regular security updates |
|||
- Platform compatibility checks |
|||
- Performance optimization |
|||
- Documentation updates |
|||
- Dependency updates |
|||
|
|||
## License |
|||
MIT License - See LICENSE file for details |
File diff suppressed because it is too large
File diff suppressed because it is too large
@ -1,4 +1,15 @@ |
|||
import { initializeApp } from "./main.common"; |
|||
|
|||
const app = initializeApp(); |
|||
app.mount("#app"); |
|||
// Initialize and mount the app
|
|||
initializeApp().then((app) => { |
|||
app.mount("#app"); |
|||
}).catch((error) => { |
|||
console.error("Failed to initialize app:", error); |
|||
document.body.innerHTML = ` |
|||
<div style="color: red; padding: 20px; font-family: sans-serif;"> |
|||
<h1>Failed to initialize app</h1> |
|||
<p>${error instanceof Error ? error.message : "Unknown error"}</p> |
|||
<p>Please try restarting the app or contact support if the problem persists.</p> |
|||
</div> |
|||
`;
|
|||
}); |
|||
|
@ -1,4 +1,15 @@ |
|||
import { initializeApp } from "./main.common"; |
|||
|
|||
const app = initializeApp(); |
|||
app.mount("#app"); |
|||
// Initialize and mount the app
|
|||
initializeApp().then((app) => { |
|||
app.mount("#app"); |
|||
}).catch((error) => { |
|||
console.error("Failed to initialize app:", error); |
|||
document.body.innerHTML = ` |
|||
<div style="color: red; padding: 20px; font-family: sans-serif;"> |
|||
<h1>Failed to initialize app</h1> |
|||
<p>${error instanceof Error ? error.message : "Unknown error"}</p> |
|||
<p>Please try restarting the app or contact support if the problem persists.</p> |
|||
</div> |
|||
`;
|
|||
}); |
|||
|
@ -0,0 +1,370 @@ |
|||
import { |
|||
PlatformService, |
|||
PlatformCapabilities, |
|||
SQLiteOperations, |
|||
SQLiteConfig, |
|||
PreparedStatement, |
|||
SQLiteResult, |
|||
ImageResult, |
|||
} from "./PlatformService"; |
|||
import { BaseSQLiteService } from "./sqlite/BaseSQLiteService"; |
|||
import { app } from "electron"; |
|||
import { dialog } from "electron"; |
|||
import fs from "fs"; |
|||
import path from "path"; |
|||
import sqlite3 from "sqlite3"; |
|||
import { open, Database } from "sqlite"; |
|||
import { logger } from "../utils/logger"; |
|||
import { Settings } from "../db/tables/settings"; |
|||
import { Account } from "../db/tables/accounts"; |
|||
import { Contact } from "../db/tables/contacts"; |
|||
import { db } from "../db"; |
|||
import { MASTER_SETTINGS_KEY } from "../db/tables/settings"; |
|||
import { accountsDBPromise } from "../db"; |
|||
import { accessToken } from "../libs/crypto"; |
|||
import { getPlanFromCache as getPlanFromCacheImpl } from "../libs/endorserServer"; |
|||
import { PlanSummaryRecord } from "../interfaces/records"; |
|||
import { Axios } from "axios"; |
|||
|
|||
interface SQLiteDatabase extends Database { |
|||
changes: number; |
|||
} |
|||
|
|||
// Create Promise-based versions of fs functions
|
|||
const readFileAsync = (filePath: string, encoding: BufferEncoding): Promise<string> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.readFile(filePath, { encoding }, (err: NodeJS.ErrnoException | null, data: string) => { |
|||
if (err) reject(err); |
|||
else resolve(data); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
const readFileBufferAsync = (filePath: string): Promise<Buffer> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.readFile(filePath, (err: NodeJS.ErrnoException | null, data: Buffer) => { |
|||
if (err) reject(err); |
|||
else resolve(data); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
const writeFileAsync = (filePath: string, data: string, encoding: BufferEncoding): Promise<void> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.writeFile(filePath, data, { encoding }, (err: NodeJS.ErrnoException | null) => { |
|||
if (err) reject(err); |
|||
else resolve(); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
const unlinkAsync = (filePath: string): Promise<void> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.unlink(filePath, (err: NodeJS.ErrnoException | null) => { |
|||
if (err) reject(err); |
|||
else resolve(); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
const readdirAsync = (dirPath: string): Promise<string[]> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.readdir(dirPath, (err: NodeJS.ErrnoException | null, files: string[]) => { |
|||
if (err) reject(err); |
|||
else resolve(files); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
const statAsync = (filePath: string): Promise<fs.Stats> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.stat(filePath, (err: NodeJS.ErrnoException | null, stats: fs.Stats) => { |
|||
if (err) reject(err); |
|||
else resolve(stats); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
/** |
|||
* SQLite implementation for Electron using native sqlite3 |
|||
*/ |
|||
class ElectronSQLiteService extends BaseSQLiteService { |
|||
private db: SQLiteDatabase | null = null; |
|||
private config: SQLiteConfig | null = null; |
|||
|
|||
async initialize(config: SQLiteConfig): Promise<void> { |
|||
if (this.initialized) { |
|||
return; |
|||
} |
|||
|
|||
try { |
|||
this.config = config; |
|||
const dbPath = path.join(app.getPath("userData"), `${config.name}.db`); |
|||
|
|||
this.db = await open({ |
|||
filename: dbPath, |
|||
driver: sqlite3.Database, |
|||
}); |
|||
|
|||
// Configure database settings
|
|||
if (config.useWAL) { |
|||
await this.execute("PRAGMA journal_mode = WAL"); |
|||
this.stats.walMode = true; |
|||
} |
|||
|
|||
// Set other pragmas for performance
|
|||
await this.execute("PRAGMA synchronous = NORMAL"); |
|||
await this.execute("PRAGMA temp_store = MEMORY"); |
|||
await this.execute("PRAGMA cache_size = -2000"); // Use 2MB of cache
|
|||
|
|||
this.initialized = true; |
|||
await this.updateStats(); |
|||
} catch (error) { |
|||
logger.error("Failed to initialize Electron SQLite:", error); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
async close(): Promise<void> { |
|||
if (!this.initialized || !this.db) { |
|||
return; |
|||
} |
|||
|
|||
try { |
|||
await this.db.close(); |
|||
this.db = null; |
|||
this.initialized = false; |
|||
} catch (error) { |
|||
logger.error("Failed to close Electron SQLite connection:", error); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected async _executeQuery<T>( |
|||
sql: string, |
|||
params: unknown[] = [], |
|||
operation: "query" | "execute" = "query", |
|||
): Promise<SQLiteResult<T>> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
if (operation === "query") { |
|||
const rows = await this.db.all<T[]>(sql, params); |
|||
const result = await this.db.run("SELECT last_insert_rowid() as id"); |
|||
return { |
|||
rows, |
|||
rowsAffected: this.db.changes, |
|||
lastInsertId: result.lastID, |
|||
executionTime: 0, // Will be set by base class
|
|||
}; |
|||
} else { |
|||
const result = await this.db.run(sql, params); |
|||
return { |
|||
rows: [], |
|||
rowsAffected: this.db.changes, |
|||
lastInsertId: result.lastID, |
|||
executionTime: 0, // Will be set by base class
|
|||
}; |
|||
} |
|||
} catch (error) { |
|||
logger.error("Electron SQLite query failed:", { |
|||
sql, |
|||
params, |
|||
error: error instanceof Error ? error.message : String(error), |
|||
}); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected async _beginTransaction(): Promise<void> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
await this.db.run("BEGIN TRANSACTION"); |
|||
} |
|||
|
|||
protected async _commitTransaction(): Promise<void> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
await this.db.run("COMMIT"); |
|||
} |
|||
|
|||
protected async _rollbackTransaction(): Promise<void> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
await this.db.run("ROLLBACK"); |
|||
} |
|||
|
|||
protected async _prepareStatement<T>( |
|||
sql: string, |
|||
): Promise<PreparedStatement<T>> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
const stmt = await this.db.prepare(sql); |
|||
return { |
|||
execute: async (params: unknown[] = []) => { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
const rows = await stmt.all<T>(params); |
|||
return { |
|||
rows, |
|||
rowsAffected: this.db.changes, |
|||
lastInsertId: (await this.db.run("SELECT last_insert_rowid() as id")) |
|||
.lastID, |
|||
executionTime: 0, // Will be set by base class
|
|||
}; |
|||
}, |
|||
finalize: async () => { |
|||
await stmt.finalize(); |
|||
}, |
|||
}; |
|||
} |
|||
|
|||
protected async _finalizeStatement(_sql: string): Promise<void> { |
|||
// Statements are finalized when the PreparedStatement is finalized
|
|||
} |
|||
|
|||
async getDatabaseSize(): Promise<number> { |
|||
if (!this.db || !this.config) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
const dbPath = path.join(app.getPath("userData"), `${this.config.name}.db`); |
|||
const stats = await statAsync(dbPath); |
|||
return stats.size; |
|||
} catch (error) { |
|||
logger.error("Failed to get database size:", error); |
|||
return 0; |
|||
} |
|||
} |
|||
} |
|||
|
|||
// Only import Electron-specific code in Electron environment
|
|||
let ElectronPlatformServiceImpl: typeof import("./platforms/ElectronPlatformService").ElectronPlatformService; |
|||
|
|||
async function initializeElectronPlatformService() { |
|||
if (process.env.ELECTRON) { |
|||
// Dynamic import for Electron environment
|
|||
const { ElectronPlatformService } = await import("./platforms/ElectronPlatformService"); |
|||
ElectronPlatformServiceImpl = ElectronPlatformService; |
|||
} else { |
|||
// Stub implementation for non-Electron environments
|
|||
class StubElectronPlatformService implements PlatformService { |
|||
#sqliteService: SQLiteOperations | null = null; |
|||
|
|||
getCapabilities(): PlatformCapabilities { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getSQLite(): Promise<SQLiteOperations> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async readFile(path: string): Promise<string> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async writeFile(path: string, content: string): Promise<void> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async deleteFile(path: string): Promise<void> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async listFiles(directory: string): Promise<string[]> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async takePicture(): Promise<ImageResult> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async pickImage(): Promise<ImageResult> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async handleDeepLink(url: string): Promise<void> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getAccounts(): Promise<Account[]> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getAccount(did: string): Promise<Account | undefined> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async addAccount(account: Account): Promise<void> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getContacts(): Promise<Contact[]> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getAllContacts(): Promise<Contact[]> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async updateMasterSettings(settingsChanges: Partial<Settings>): Promise<void> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getActiveAccountSettings(): Promise<Settings> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async updateAccountSettings(accountDid: string, settingsChanges: Partial<Settings>): Promise<void> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getHeaders(did?: string): Promise<Record<string, string>> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
async getPlanFromCache( |
|||
handleId: string | undefined, |
|||
axios: Axios, |
|||
apiServer: string, |
|||
requesterDid?: string, |
|||
): Promise<PlanSummaryRecord | undefined> { |
|||
throw new Error("Electron platform service is not available in this environment"); |
|||
} |
|||
|
|||
isCapacitor(): boolean { |
|||
return false; |
|||
} |
|||
|
|||
isElectron(): boolean { |
|||
return false; |
|||
} |
|||
|
|||
isPyWebView(): boolean { |
|||
return false; |
|||
} |
|||
|
|||
isWeb(): boolean { |
|||
return false; |
|||
} |
|||
} |
|||
ElectronPlatformServiceImpl = StubElectronPlatformService; |
|||
} |
|||
} |
|||
|
|||
// Initialize the service
|
|||
initializeElectronPlatformService().catch(error => { |
|||
logger.error("Failed to initialize Electron platform service:", error); |
|||
}); |
|||
|
|||
export class ElectronPlatformService extends ElectronPlatformServiceImpl {} |
@ -1,111 +1,102 @@ |
|||
import { |
|||
ImageResult, |
|||
PlatformService, |
|||
PlatformCapabilities, |
|||
SQLiteOperations, |
|||
SQLiteConfig, |
|||
PreparedStatement, |
|||
SQLiteResult, |
|||
ImageResult, |
|||
} from "../PlatformService"; |
|||
import { BaseSQLiteService } from "../sqlite/BaseSQLiteService"; |
|||
import { app } from "electron"; |
|||
import { dialog } from "electron"; |
|||
import fs from "fs"; |
|||
import path from "path"; |
|||
import sqlite3 from "sqlite3"; |
|||
import { open, Database } from "sqlite"; |
|||
import { logger } from "../../utils/logger"; |
|||
import { Settings } from "../../db/tables/settings"; |
|||
import { Account } from "../../db/tables/accounts"; |
|||
import { Contact } from "../../db/tables/contacts"; |
|||
import { db } from "../../db"; |
|||
import { MASTER_SETTINGS_KEY } from "../../db/tables/settings"; |
|||
import { accountsDBPromise } from "../../db"; |
|||
import { accessToken } from "../../libs/crypto"; |
|||
import { getPlanFromCache as getPlanFromCacheImpl } from "../../libs/endorserServer"; |
|||
import { PlanSummaryRecord } from "../../interfaces/records"; |
|||
import { Axios } from "axios"; |
|||
|
|||
/** |
|||
* Platform service implementation for Electron (desktop) platform. |
|||
* Note: This is a placeholder implementation with most methods currently unimplemented. |
|||
* Implements the PlatformService interface but throws "Not implemented" errors for most operations. |
|||
* |
|||
* @remarks |
|||
* This service is intended for desktop application functionality through Electron. |
|||
* Future implementations should provide: |
|||
* - Native file system access |
|||
* - Desktop camera integration |
|||
* - System-level features |
|||
*/ |
|||
export class ElectronPlatformService implements PlatformService { |
|||
/** |
|||
* Gets the capabilities of the Electron platform |
|||
* @returns Platform capabilities object |
|||
*/ |
|||
getCapabilities(): PlatformCapabilities { |
|||
return { |
|||
hasFileSystem: false, // Not implemented yet
|
|||
hasCamera: false, // Not implemented yet
|
|||
isMobile: false, |
|||
isIOS: false, |
|||
hasFileDownload: false, // Not implemented yet
|
|||
needsFileHandlingInstructions: false, |
|||
}; |
|||
} |
|||
// Create Promise-based versions of fs functions
|
|||
const readFileAsync = (filePath: string, encoding: BufferEncoding): Promise<string> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.readFile(filePath, { encoding }, (err: NodeJS.ErrnoException | null, data: string) => { |
|||
if (err) reject(err); |
|||
else resolve(data); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
const readFileBufferAsync = (filePath: string): Promise<Buffer> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.readFile(filePath, (err: NodeJS.ErrnoException | null, data: Buffer) => { |
|||
if (err) reject(err); |
|||
else resolve(data); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
/** |
|||
* Reads a file from the filesystem. |
|||
* @param _path - Path to the file to read |
|||
* @returns Promise that should resolve to file contents |
|||
* @throws Error with "Not implemented" message |
|||
* @todo Implement file reading using Electron's file system API |
|||
*/ |
|||
async readFile(_path: string): Promise<string> { |
|||
throw new Error("Not implemented"); |
|||
} |
|||
const writeFileAsync = (filePath: string, data: string, encoding: BufferEncoding): Promise<void> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.writeFile(filePath, data, { encoding }, (err: NodeJS.ErrnoException | null) => { |
|||
if (err) reject(err); |
|||
else resolve(); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
/** |
|||
* Writes content to a file. |
|||
* @param _path - Path where to write the file |
|||
* @param _content - Content to write to the file |
|||
* @throws Error with "Not implemented" message |
|||
* @todo Implement file writing using Electron's file system API |
|||
*/ |
|||
async writeFile(_path: string, _content: string): Promise<void> { |
|||
throw new Error("Not implemented"); |
|||
} |
|||
const unlinkAsync = (filePath: string): Promise<void> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.unlink(filePath, (err: NodeJS.ErrnoException | null) => { |
|||
if (err) reject(err); |
|||
else resolve(); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
/** |
|||
* Deletes a file from the filesystem. |
|||
* @param _path - Path to the file to delete |
|||
* @throws Error with "Not implemented" message |
|||
* @todo Implement file deletion using Electron's file system API |
|||
*/ |
|||
async deleteFile(_path: string): Promise<void> { |
|||
throw new Error("Not implemented"); |
|||
} |
|||
const readdirAsync = (dirPath: string): Promise<string[]> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.readdir(dirPath, (err: NodeJS.ErrnoException | null, files: string[]) => { |
|||
if (err) reject(err); |
|||
else resolve(files); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
/** |
|||
* Lists files in the specified directory. |
|||
* @param _directory - Path to the directory to list |
|||
* @returns Promise that should resolve to array of filenames |
|||
* @throws Error with "Not implemented" message |
|||
* @todo Implement directory listing using Electron's file system API |
|||
*/ |
|||
async listFiles(_directory: string): Promise<string[]> { |
|||
throw new Error("Not implemented"); |
|||
} |
|||
const statAsync = (filePath: string): Promise<fs.Stats> => { |
|||
return new Promise((resolve, reject) => { |
|||
fs.stat(filePath, (err: NodeJS.ErrnoException | null, stats: fs.Stats) => { |
|||
if (err) reject(err); |
|||
else resolve(stats); |
|||
}); |
|||
}); |
|||
}; |
|||
|
|||
/** |
|||
* Should open system camera to take a picture. |
|||
* @returns Promise that should resolve to captured image data |
|||
* @throws Error with "Not implemented" message |
|||
* @todo Implement camera access using Electron's media APIs |
|||
*/ |
|||
async takePicture(): Promise<ImageResult> { |
|||
logger.error("takePicture not implemented in Electron platform"); |
|||
throw new Error("Not implemented"); |
|||
} |
|||
interface SQLiteDatabase extends Database { |
|||
changes: number; |
|||
} |
|||
|
|||
/** |
|||
* Should open system file picker for selecting an image. |
|||
* @returns Promise that should resolve to selected image data |
|||
* @throws Error with "Not implemented" message |
|||
* @todo Implement file picker using Electron's dialog API |
|||
*/ |
|||
async pickImage(): Promise<ImageResult> { |
|||
logger.error("pickImage not implemented in Electron platform"); |
|||
throw new Error("Not implemented"); |
|||
} |
|||
/** |
|||
* SQLite implementation for Electron using native sqlite3 |
|||
*/ |
|||
class ElectronSQLiteService extends BaseSQLiteService { |
|||
private db: SQLiteDatabase | null = null; |
|||
private config: SQLiteConfig | null = null; |
|||
|
|||
/** |
|||
* Should handle deep link URLs for the desktop application. |
|||
* @param _url - The deep link URL to handle |
|||
* @throws Error with "Not implemented" message |
|||
* @todo Implement deep link handling using Electron's protocol handler |
|||
*/ |
|||
async handleDeepLink(_url: string): Promise<void> { |
|||
logger.error("handleDeepLink not implemented in Electron platform"); |
|||
throw new Error("Not implemented"); |
|||
} |
|||
// ... rest of the ElectronSQLiteService implementation ...
|
|||
} |
|||
|
|||
export class ElectronPlatformService implements PlatformService { |
|||
private sqliteService: ElectronSQLiteService | null = null; |
|||
|
|||
// ... rest of the ElectronPlatformService implementation ...
|
|||
} |
@ -0,0 +1,248 @@ |
|||
import initSqlJs, { Database } from "@jlongster/sql.js"; |
|||
import { SQLiteFS } from "absurd-sql"; |
|||
import { IndexedDBBackend } from "absurd-sql/dist/indexeddb-backend"; |
|||
import { BaseSQLiteService } from "./BaseSQLiteService"; |
|||
import { |
|||
SQLiteConfig, |
|||
SQLiteResult, |
|||
PreparedStatement, |
|||
} from "../PlatformService"; |
|||
import { logger } from "../../utils/logger"; |
|||
|
|||
/** |
|||
* SQLite implementation using absurd-sql for web browsers. |
|||
* Provides SQLite access in the browser using Web Workers and IndexedDB. |
|||
*/ |
|||
export class AbsurdSQLService extends BaseSQLiteService { |
|||
private db: Database | null = null; |
|||
private worker: Worker | null = null; |
|||
private config: SQLiteConfig | null = null; |
|||
|
|||
async initialize(config: SQLiteConfig): Promise<void> { |
|||
if (this.initialized) { |
|||
return; |
|||
} |
|||
|
|||
try { |
|||
this.config = config; |
|||
const SQL = await initSqlJs({ |
|||
locateFile: (file) => `/sql-wasm/${file}`, |
|||
}); |
|||
|
|||
// Initialize the virtual file system
|
|||
const backend = new IndexedDBBackend(this.config.name); |
|||
const fs = new SQLiteFS(SQL.FS, backend); |
|||
SQL.register_for_idb(fs); |
|||
|
|||
// Create and initialize the database
|
|||
this.db = new SQL.Database(this.config.name, { |
|||
filename: true, |
|||
}); |
|||
|
|||
// Configure database settings
|
|||
if (this.config.useWAL) { |
|||
await this.execute("PRAGMA journal_mode = WAL"); |
|||
this.stats.walMode = true; |
|||
} |
|||
|
|||
if (this.config.useMMap) { |
|||
const mmapSize = this.config.mmapSize ?? 30000000000; |
|||
await this.execute(`PRAGMA mmap_size = ${mmapSize}`); |
|||
this.stats.mmapActive = true; |
|||
} |
|||
|
|||
// Set other pragmas for performance
|
|||
await this.execute("PRAGMA synchronous = NORMAL"); |
|||
await this.execute("PRAGMA temp_store = MEMORY"); |
|||
await this.execute("PRAGMA cache_size = -2000"); // Use 2MB of cache
|
|||
|
|||
// Start the Web Worker for async operations
|
|||
this.worker = new Worker(new URL("./sqlite.worker.ts", import.meta.url), { |
|||
type: "module", |
|||
}); |
|||
|
|||
this.initialized = true; |
|||
await this.updateStats(); |
|||
} catch (error) { |
|||
logger.error("Failed to initialize Absurd SQL:", error); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
async close(): Promise<void> { |
|||
if (!this.initialized || !this.db) { |
|||
return; |
|||
} |
|||
|
|||
try { |
|||
// Finalize all prepared statements
|
|||
for (const [_sql, stmt] of this.preparedStatements) { |
|||
logger.debug("finalizing statement", _sql); |
|||
await stmt.finalize(); |
|||
} |
|||
this.preparedStatements.clear(); |
|||
|
|||
// Close the database
|
|||
this.db.close(); |
|||
this.db = null; |
|||
|
|||
// Terminate the worker
|
|||
if (this.worker) { |
|||
this.worker.terminate(); |
|||
this.worker = null; |
|||
} |
|||
|
|||
this.initialized = false; |
|||
} catch (error) { |
|||
logger.error("Failed to close Absurd SQL connection:", error); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected async _executeQuery<T>( |
|||
sql: string, |
|||
params: unknown[] = [], |
|||
operation: "query" | "execute" = "query", |
|||
): Promise<SQLiteResult<T>> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
let lastInsertId: number | undefined = undefined; |
|||
|
|||
if (operation === "query") { |
|||
const stmt = this.db.prepare(sql); |
|||
const rows: T[] = []; |
|||
|
|||
try { |
|||
while (stmt.step()) { |
|||
rows.push(stmt.getAsObject() as T); |
|||
} |
|||
} finally { |
|||
stmt.free(); |
|||
} |
|||
|
|||
// Get last insert ID safely
|
|||
const result = this.db.exec("SELECT last_insert_rowid() AS id"); |
|||
lastInsertId = |
|||
(result?.[0]?.values?.[0]?.[0] as number | undefined) ?? undefined; |
|||
|
|||
return { |
|||
rows, |
|||
rowsAffected: this.db.getRowsModified(), |
|||
lastInsertId, |
|||
executionTime: 0, // Will be set by base class
|
|||
}; |
|||
} else { |
|||
this.db.run(sql, params); |
|||
|
|||
// Get last insert ID after execute
|
|||
const result = this.db.exec("SELECT last_insert_rowid() AS id"); |
|||
lastInsertId = |
|||
(result?.[0]?.values?.[0]?.[0] as number | undefined) ?? undefined; |
|||
|
|||
return { |
|||
rows: [], |
|||
rowsAffected: this.db.getRowsModified(), |
|||
lastInsertId, |
|||
executionTime: 0, |
|||
}; |
|||
} |
|||
} catch (error) { |
|||
logger.error("Absurd SQL query failed:", { |
|||
sql, |
|||
params, |
|||
error: error instanceof Error ? error.message : String(error), |
|||
}); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected async _beginTransaction(): Promise<void> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
this.db.exec("BEGIN TRANSACTION"); |
|||
} |
|||
|
|||
protected async _commitTransaction(): Promise<void> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
this.db.exec("COMMIT"); |
|||
} |
|||
|
|||
protected async _rollbackTransaction(): Promise<void> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
this.db.exec("ROLLBACK"); |
|||
} |
|||
|
|||
protected async _prepareStatement<T>( |
|||
_sql: string, |
|||
): Promise<PreparedStatement<T>> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
const stmt = this.db.prepare(_sql); |
|||
return { |
|||
execute: async (params: unknown[] = []) => { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
const rows: T[] = []; |
|||
stmt.bind(params); |
|||
while (stmt.step()) { |
|||
rows.push(stmt.getAsObject() as T); |
|||
} |
|||
|
|||
// Safely extract lastInsertId
|
|||
const result = this.db.exec("SELECT last_insert_rowid()"); |
|||
const rawId = result?.[0]?.values?.[0]?.[0]; |
|||
const lastInsertId = typeof rawId === "number" ? rawId : undefined; |
|||
|
|||
return { |
|||
rows, |
|||
rowsAffected: this.db.getRowsModified(), |
|||
lastInsertId, |
|||
executionTime: 0, // Will be set by base class
|
|||
}; |
|||
} finally { |
|||
stmt.reset(); |
|||
} |
|||
}, |
|||
finalize: async () => { |
|||
stmt.free(); |
|||
}, |
|||
}; |
|||
} |
|||
|
|||
protected async _finalizeStatement(_sql: string): Promise<void> { |
|||
// Statements are finalized when the PreparedStatement is finalized
|
|||
} |
|||
|
|||
async getDatabaseSize(): Promise<number> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
const result = this.db.exec( |
|||
"SELECT page_count * page_size as size FROM pragma_page_count(), pragma_page_size()", |
|||
); |
|||
|
|||
const rawSize = result?.[0]?.values?.[0]?.[0]; |
|||
const size = typeof rawSize === "number" ? rawSize : 0; |
|||
|
|||
return size; |
|||
} catch (error) { |
|||
logger.error("Failed to get database size:", error); |
|||
return 0; |
|||
} |
|||
} |
|||
} |
@ -0,0 +1,383 @@ |
|||
import { |
|||
SQLiteOperations, |
|||
SQLiteConfig, |
|||
SQLiteResult, |
|||
PreparedStatement, |
|||
SQLiteStats, |
|||
} from "../PlatformService"; |
|||
import { Settings, MASTER_SETTINGS_KEY } from "../../db/tables/settings"; |
|||
import { logger } from "../../utils/logger"; |
|||
|
|||
/** |
|||
* Base class for SQLite implementations across different platforms. |
|||
* Provides common functionality and error handling. |
|||
*/ |
|||
export abstract class BaseSQLiteService implements SQLiteOperations { |
|||
protected initialized = false; |
|||
protected stats: SQLiteStats = { |
|||
totalQueries: 0, |
|||
avgExecutionTime: 0, |
|||
preparedStatements: 0, |
|||
databaseSize: 0, |
|||
walMode: false, |
|||
mmapActive: false, |
|||
}; |
|||
protected preparedStatements: Map<string, PreparedStatement<unknown>> = |
|||
new Map(); |
|||
|
|||
abstract initialize(config: SQLiteConfig): Promise<void>; |
|||
abstract close(): Promise<void>; |
|||
abstract getDatabaseSize(): Promise<number>; |
|||
|
|||
protected async executeQuery<T>( |
|||
sql: string, |
|||
params: unknown[] = [], |
|||
operation: "query" | "execute" = "query", |
|||
): Promise<SQLiteResult<T>> { |
|||
if (!this.initialized) { |
|||
throw new Error("SQLite database not initialized"); |
|||
} |
|||
|
|||
const startTime = performance.now(); |
|||
try { |
|||
const result = await this._executeQuery<T>(sql, params, operation); |
|||
const executionTime = performance.now() - startTime; |
|||
|
|||
// Update stats
|
|||
this.stats.totalQueries++; |
|||
this.stats.avgExecutionTime = |
|||
(this.stats.avgExecutionTime * (this.stats.totalQueries - 1) + |
|||
executionTime) / |
|||
this.stats.totalQueries; |
|||
|
|||
return { |
|||
...result, |
|||
executionTime, |
|||
}; |
|||
} catch (error) { |
|||
logger.error("SQLite query failed:", { |
|||
sql, |
|||
params, |
|||
error: error instanceof Error ? error.message : String(error), |
|||
}); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected abstract _executeQuery<T>( |
|||
sql: string, |
|||
params: unknown[], |
|||
operation: "query" | "execute", |
|||
): Promise<SQLiteResult<T>>; |
|||
|
|||
async query<T>( |
|||
sql: string, |
|||
params: unknown[] = [], |
|||
): Promise<SQLiteResult<T>> { |
|||
return this.executeQuery<T>(sql, params, "query"); |
|||
} |
|||
|
|||
async execute(sql: string, params: unknown[] = []): Promise<number> { |
|||
const result = await this.executeQuery<unknown>(sql, params, "execute"); |
|||
return result.rowsAffected; |
|||
} |
|||
|
|||
async transaction( |
|||
statements: { sql: string; params?: unknown[] }[], |
|||
): Promise<void> { |
|||
if (!this.initialized) { |
|||
throw new Error("SQLite database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
await this._beginTransaction(); |
|||
for (const { sql, params = [] } of statements) { |
|||
await this.executeQuery(sql, params, "execute"); |
|||
} |
|||
await this._commitTransaction(); |
|||
} catch (error) { |
|||
await this._rollbackTransaction(); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected abstract _beginTransaction(): Promise<void>; |
|||
protected abstract _commitTransaction(): Promise<void>; |
|||
protected abstract _rollbackTransaction(): Promise<void>; |
|||
|
|||
async getMaxValue<T>( |
|||
table: string, |
|||
column: string, |
|||
where?: string, |
|||
params: unknown[] = [], |
|||
): Promise<T | null> { |
|||
const sql = `SELECT MAX(${column}) as max_value FROM ${table}${where ? ` WHERE ${where}` : ""}`; |
|||
const result = await this.query<{ max_value: T }>(sql, params); |
|||
return result.rows[0]?.max_value ?? null; |
|||
} |
|||
|
|||
async prepare<T>(sql: string): Promise<PreparedStatement<T>> { |
|||
if (!this.initialized) { |
|||
throw new Error("SQLite database not initialized"); |
|||
} |
|||
|
|||
const stmt = await this._prepareStatement<T>(sql); |
|||
this.stats.preparedStatements++; |
|||
this.preparedStatements.set(sql, stmt); |
|||
|
|||
return { |
|||
execute: async (params: unknown[] = []) => { |
|||
return this.executeQuery<T>(sql, params, "query"); |
|||
}, |
|||
finalize: async () => { |
|||
await this._finalizeStatement(sql); |
|||
this.preparedStatements.delete(sql); |
|||
this.stats.preparedStatements--; |
|||
}, |
|||
}; |
|||
} |
|||
|
|||
protected abstract _prepareStatement<T>( |
|||
sql: string, |
|||
): Promise<PreparedStatement<T>>; |
|||
protected abstract _finalizeStatement(sql: string): Promise<void>; |
|||
|
|||
async getStats(): Promise<SQLiteStats> { |
|||
return { |
|||
...this.stats, |
|||
databaseSize: await this.getDatabaseSize(), |
|||
}; |
|||
} |
|||
|
|||
protected async updateStats(): Promise<void> { |
|||
this.stats.databaseSize = await this.getDatabaseSize(); |
|||
// Platform-specific stats updates can be implemented in subclasses
|
|||
} |
|||
|
|||
protected async setupSchema(): Promise<void> { |
|||
await this.execute(` |
|||
CREATE TABLE IF NOT EXISTS settings ( |
|||
id INTEGER PRIMARY KEY, |
|||
accountDid TEXT, |
|||
activeDid TEXT, |
|||
apiServer TEXT, |
|||
filterFeedByNearby INTEGER, |
|||
filterFeedByVisible INTEGER, |
|||
finishedOnboarding INTEGER, |
|||
firstName TEXT, |
|||
hideRegisterPromptOnNewContact INTEGER, |
|||
isRegistered INTEGER, |
|||
lastName TEXT, |
|||
lastAckedOfferToUserJwtId TEXT, |
|||
lastAckedOfferToUserProjectsJwtId TEXT, |
|||
lastNotifiedClaimId TEXT, |
|||
lastViewedClaimId TEXT, |
|||
notifyingNewActivityTime TEXT, |
|||
notifyingReminderMessage TEXT, |
|||
notifyingReminderTime TEXT, |
|||
partnerApiServer TEXT, |
|||
passkeyExpirationMinutes INTEGER, |
|||
profileImageUrl TEXT, |
|||
searchBoxes TEXT, |
|||
showContactGivesInline INTEGER, |
|||
showGeneralAdvanced INTEGER, |
|||
showShortcutBvc INTEGER, |
|||
vapid TEXT, |
|||
warnIfProdServer INTEGER, |
|||
warnIfTestServer INTEGER, |
|||
webPushServer TEXT |
|||
) |
|||
`);
|
|||
} |
|||
|
|||
protected async settingsToRow( |
|||
settings: Partial<Settings>, |
|||
): Promise<Record<string, unknown>> { |
|||
const row: Record<string, unknown> = {}; |
|||
|
|||
// Convert boolean values to integers for SQLite
|
|||
if ("filterFeedByNearby" in settings) |
|||
row.filterFeedByNearby = settings.filterFeedByNearby ? 1 : 0; |
|||
if ("filterFeedByVisible" in settings) |
|||
row.filterFeedByVisible = settings.filterFeedByVisible ? 1 : 0; |
|||
if ("finishedOnboarding" in settings) |
|||
row.finishedOnboarding = settings.finishedOnboarding ? 1 : 0; |
|||
if ("hideRegisterPromptOnNewContact" in settings) |
|||
row.hideRegisterPromptOnNewContact = |
|||
settings.hideRegisterPromptOnNewContact ? 1 : 0; |
|||
if ("isRegistered" in settings) |
|||
row.isRegistered = settings.isRegistered ? 1 : 0; |
|||
if ("showContactGivesInline" in settings) |
|||
row.showContactGivesInline = settings.showContactGivesInline ? 1 : 0; |
|||
if ("showGeneralAdvanced" in settings) |
|||
row.showGeneralAdvanced = settings.showGeneralAdvanced ? 1 : 0; |
|||
if ("showShortcutBvc" in settings) |
|||
row.showShortcutBvc = settings.showShortcutBvc ? 1 : 0; |
|||
if ("warnIfProdServer" in settings) |
|||
row.warnIfProdServer = settings.warnIfProdServer ? 1 : 0; |
|||
if ("warnIfTestServer" in settings) |
|||
row.warnIfTestServer = settings.warnIfTestServer ? 1 : 0; |
|||
|
|||
// Handle JSON fields
|
|||
if ("searchBoxes" in settings) |
|||
row.searchBoxes = JSON.stringify(settings.searchBoxes); |
|||
|
|||
// Copy all other fields as is
|
|||
Object.entries(settings).forEach(([key, value]) => { |
|||
if (!(key in row)) { |
|||
row[key] = value; |
|||
} |
|||
}); |
|||
|
|||
return row; |
|||
} |
|||
|
|||
protected async rowToSettings( |
|||
row: Record<string, unknown>, |
|||
): Promise<Settings> { |
|||
const settings: Settings = {}; |
|||
|
|||
// Convert integer values back to booleans
|
|||
if ("filterFeedByNearby" in row) |
|||
settings.filterFeedByNearby = !!row.filterFeedByNearby; |
|||
if ("filterFeedByVisible" in row) |
|||
settings.filterFeedByVisible = !!row.filterFeedByVisible; |
|||
if ("finishedOnboarding" in row) |
|||
settings.finishedOnboarding = !!row.finishedOnboarding; |
|||
if ("hideRegisterPromptOnNewContact" in row) |
|||
settings.hideRegisterPromptOnNewContact = |
|||
!!row.hideRegisterPromptOnNewContact; |
|||
if ("isRegistered" in row) settings.isRegistered = !!row.isRegistered; |
|||
if ("showContactGivesInline" in row) |
|||
settings.showContactGivesInline = !!row.showContactGivesInline; |
|||
if ("showGeneralAdvanced" in row) |
|||
settings.showGeneralAdvanced = !!row.showGeneralAdvanced; |
|||
if ("showShortcutBvc" in row) |
|||
settings.showShortcutBvc = !!row.showShortcutBvc; |
|||
if ("warnIfProdServer" in row) |
|||
settings.warnIfProdServer = !!row.warnIfProdServer; |
|||
if ("warnIfTestServer" in row) |
|||
settings.warnIfTestServer = !!row.warnIfTestServer; |
|||
|
|||
// Parse JSON fields
|
|||
if ("searchBoxes" in row && row.searchBoxes) { |
|||
try { |
|||
settings.searchBoxes = JSON.parse(row.searchBoxes); |
|||
} catch (error) { |
|||
logger.error("Error parsing searchBoxes JSON:", error); |
|||
} |
|||
} |
|||
|
|||
// Copy all other fields as is
|
|||
Object.entries(row).forEach(([key, value]) => { |
|||
if (!(key in settings)) { |
|||
(settings as Record<string, unknown>)[key] = value; |
|||
} |
|||
}); |
|||
|
|||
return settings; |
|||
} |
|||
|
|||
async updateMasterSettings( |
|||
settingsChanges: Partial<Settings>, |
|||
): Promise<void> { |
|||
try { |
|||
const row = await this.settingsToRow(settingsChanges); |
|||
row.id = MASTER_SETTINGS_KEY; |
|||
delete row.accountDid; |
|||
|
|||
const result = await this.execute( |
|||
`UPDATE settings SET ${Object.keys(row) |
|||
.map((k) => `${k} = ?`) |
|||
.join(", ")} WHERE id = ?`,
|
|||
[...Object.values(row), MASTER_SETTINGS_KEY], |
|||
); |
|||
|
|||
if (result === 0) { |
|||
// If no record was updated, create a new one
|
|||
await this.execute( |
|||
`INSERT INTO settings (${Object.keys(row).join(", ")}) VALUES (${Object.keys( |
|||
row, |
|||
) |
|||
.map(() => "?") |
|||
.join(", ")})`,
|
|||
Object.values(row), |
|||
); |
|||
} |
|||
} catch (error) { |
|||
logger.error("Error updating master settings:", error); |
|||
throw new Error("Failed to update settings"); |
|||
} |
|||
} |
|||
|
|||
async getActiveAccountSettings(): Promise<Settings> { |
|||
try { |
|||
const defaultSettings = await this.query<Record<string, unknown>>( |
|||
"SELECT * FROM settings WHERE id = ?", |
|||
[MASTER_SETTINGS_KEY], |
|||
); |
|||
|
|||
if (!defaultSettings.rows.length) { |
|||
return {}; |
|||
} |
|||
|
|||
const settings = await this.rowToSettings(defaultSettings.rows[0]); |
|||
|
|||
if (!settings.activeDid) { |
|||
return settings; |
|||
} |
|||
|
|||
const overrideSettings = await this.query<Record<string, unknown>>( |
|||
"SELECT * FROM settings WHERE accountDid = ?", |
|||
[settings.activeDid], |
|||
); |
|||
|
|||
if (!overrideSettings.rows.length) { |
|||
return settings; |
|||
} |
|||
|
|||
const override = await this.rowToSettings(overrideSettings.rows[0]); |
|||
return { ...settings, ...override }; |
|||
} catch (error) { |
|||
logger.error("Error getting active account settings:", error); |
|||
throw new Error("Failed to get settings"); |
|||
} |
|||
} |
|||
|
|||
async updateAccountSettings( |
|||
accountDid: string, |
|||
settingsChanges: Partial<Settings>, |
|||
): Promise<void> { |
|||
try { |
|||
const row = await this.settingsToRow(settingsChanges); |
|||
row.accountDid = accountDid; |
|||
|
|||
const result = await this.execute( |
|||
`UPDATE settings SET ${Object.keys(row) |
|||
.map((k) => `${k} = ?`) |
|||
.join(", ")} WHERE accountDid = ?`,
|
|||
[...Object.values(row), accountDid], |
|||
); |
|||
|
|||
if (result === 0) { |
|||
// If no record was updated, create a new one
|
|||
const idResult = await this.query<{ max: number }>( |
|||
"SELECT MAX(id) as max FROM settings", |
|||
); |
|||
row.id = (idResult.rows[0]?.max || 0) + 1; |
|||
|
|||
await this.execute( |
|||
`INSERT INTO settings (${Object.keys(row).join(", ")}) VALUES (${Object.keys( |
|||
row, |
|||
) |
|||
.map(() => "?") |
|||
.join(", ")})`,
|
|||
Object.values(row), |
|||
); |
|||
} |
|||
} catch (error) { |
|||
logger.error("Error updating account settings:", error); |
|||
throw new Error("Failed to update settings"); |
|||
} |
|||
} |
|||
} |
@ -0,0 +1,176 @@ |
|||
import { |
|||
CapacitorSQLite, |
|||
SQLiteConnection, |
|||
SQLiteDBConnection, |
|||
} from "@capacitor-community/sqlite"; |
|||
import { BaseSQLiteService } from "./BaseSQLiteService"; |
|||
import { |
|||
SQLiteConfig, |
|||
SQLiteResult, |
|||
PreparedStatement, |
|||
} from "../PlatformService"; |
|||
import { logger } from "../../utils/logger"; |
|||
|
|||
/** |
|||
* SQLite implementation using the Capacitor SQLite plugin. |
|||
* Provides native SQLite access on mobile platforms. |
|||
*/ |
|||
export class CapacitorSQLiteService extends BaseSQLiteService { |
|||
private connection: SQLiteDBConnection | null = null; |
|||
private sqlite: SQLiteConnection | null = null; |
|||
|
|||
async initialize(config: SQLiteConfig): Promise<void> { |
|||
if (this.initialized) { |
|||
return; |
|||
} |
|||
|
|||
try { |
|||
this.sqlite = new SQLiteConnection(CapacitorSQLite); |
|||
const db = await this.sqlite.createConnection( |
|||
config.name, |
|||
config.useWAL ?? false, |
|||
"no-encryption", |
|||
1, |
|||
false, |
|||
); |
|||
|
|||
await db.open(); |
|||
this.connection = db; |
|||
|
|||
// Configure database settings
|
|||
if (config.useWAL) { |
|||
await this.execute("PRAGMA journal_mode = WAL"); |
|||
this.stats.walMode = true; |
|||
} |
|||
|
|||
// Set other pragmas for performance
|
|||
await this.execute("PRAGMA synchronous = NORMAL"); |
|||
await this.execute("PRAGMA temp_store = MEMORY"); |
|||
await this.execute("PRAGMA mmap_size = 30000000000"); |
|||
this.stats.mmapActive = true; |
|||
|
|||
// Set up database schema
|
|||
await this.setupSchema(); |
|||
|
|||
this.initialized = true; |
|||
await this.updateStats(); |
|||
} catch (error) { |
|||
logger.error("Failed to initialize Capacitor SQLite:", error); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
async close(): Promise<void> { |
|||
if (!this.initialized || !this.connection || !this.sqlite) { |
|||
return; |
|||
} |
|||
|
|||
try { |
|||
await this.connection.close(); |
|||
await this.sqlite.closeConnection(this.connection); |
|||
this.connection = null; |
|||
this.sqlite = null; |
|||
this.initialized = false; |
|||
} catch (error) { |
|||
logger.error("Failed to close Capacitor SQLite connection:", error); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected async _executeQuery<T>( |
|||
sql: string, |
|||
params: unknown[] = [], |
|||
operation: "query" | "execute" = "query", |
|||
): Promise<SQLiteResult<T>> { |
|||
if (!this.connection) { |
|||
throw new Error("Database connection not initialized"); |
|||
} |
|||
|
|||
try { |
|||
if (operation === "query") { |
|||
const result = await this.connection.query(sql, params); |
|||
return { |
|||
rows: result.values as T[], |
|||
rowsAffected: result.changes?.changes ?? 0, |
|||
lastInsertId: result.changes?.lastId, |
|||
executionTime: 0, // Will be set by base class
|
|||
}; |
|||
} else { |
|||
const result = await this.connection.run(sql, params); |
|||
return { |
|||
rows: [], |
|||
rowsAffected: result.changes?.changes ?? 0, |
|||
lastInsertId: result.changes?.lastId, |
|||
executionTime: 0, // Will be set by base class
|
|||
}; |
|||
} |
|||
} catch (error) { |
|||
logger.error("Capacitor SQLite query failed:", { |
|||
sql, |
|||
params, |
|||
error: error instanceof Error ? error.message : String(error), |
|||
}); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected async _beginTransaction(): Promise<void> { |
|||
if (!this.connection) { |
|||
throw new Error("Database connection not initialized"); |
|||
} |
|||
await this.connection.execute("BEGIN TRANSACTION"); |
|||
} |
|||
|
|||
protected async _commitTransaction(): Promise<void> { |
|||
if (!this.connection) { |
|||
throw new Error("Database connection not initialized"); |
|||
} |
|||
await this.connection.execute("COMMIT"); |
|||
} |
|||
|
|||
protected async _rollbackTransaction(): Promise<void> { |
|||
if (!this.connection) { |
|||
throw new Error("Database connection not initialized"); |
|||
} |
|||
await this.connection.execute("ROLLBACK"); |
|||
} |
|||
|
|||
protected async _prepareStatement<T>( |
|||
sql: string, |
|||
): Promise<PreparedStatement<T>> { |
|||
if (!this.connection) { |
|||
throw new Error("Database connection not initialized"); |
|||
} |
|||
|
|||
// Capacitor SQLite doesn't support prepared statements directly,
|
|||
// so we'll simulate it by storing the SQL
|
|||
return { |
|||
execute: async (params: unknown[] = []) => { |
|||
return this.executeQuery<T>(sql, params, "query"); |
|||
}, |
|||
finalize: async () => { |
|||
// No cleanup needed for Capacitor SQLite
|
|||
}, |
|||
}; |
|||
} |
|||
|
|||
protected async _finalizeStatement(_sql: string): Promise<void> { |
|||
// No cleanup needed for Capacitor SQLite
|
|||
} |
|||
|
|||
async getDatabaseSize(): Promise<number> { |
|||
if (!this.connection) { |
|||
throw new Error("Database connection not initialized"); |
|||
} |
|||
|
|||
try { |
|||
const result = await this.connection.query( |
|||
"SELECT page_count * page_size as size FROM pragma_page_count(), pragma_page_size()", |
|||
); |
|||
return result.values?.[0]?.size ?? 0; |
|||
} catch (error) { |
|||
logger.error("Failed to get database size:", error); |
|||
return 0; |
|||
} |
|||
} |
|||
} |
@ -0,0 +1,170 @@ |
|||
import { BaseSQLiteService } from "./BaseSQLiteService"; |
|||
import { SQLiteConfig, SQLiteOperations, SQLiteResult, PreparedStatement, SQLiteStats } from "../PlatformService"; |
|||
import { logger } from "../../utils/logger"; |
|||
import initSqlJs, { Database } from "@jlongster/sql.js"; |
|||
import { SQLiteFS } from "absurd-sql"; |
|||
import IndexedDBBackend from "absurd-sql/dist/indexeddb-backend"; |
|||
|
|||
/** |
|||
* SQLite implementation for web platform using absurd-sql |
|||
*/ |
|||
export class WebSQLiteService extends BaseSQLiteService { |
|||
private db: Database | null = null; |
|||
private config: SQLiteConfig | null = null; |
|||
private worker: Worker | null = null; |
|||
|
|||
async initialize(config: SQLiteConfig): Promise<void> { |
|||
if (this.initialized) { |
|||
return; |
|||
} |
|||
|
|||
try { |
|||
this.config = config; |
|||
|
|||
// Initialize SQL.js
|
|||
const SQL = await initSqlJs({ |
|||
locateFile: (file) => `/sql-wasm.wasm`, |
|||
}); |
|||
|
|||
// Create a worker for SQLite operations
|
|||
this.worker = new Worker("/sql-worker.js"); |
|||
|
|||
// Initialize SQLiteFS with IndexedDB backend
|
|||
const backend = new IndexedDBBackend(); |
|||
const fs = new SQLiteFS(backend, this.worker); |
|||
|
|||
// Create database file
|
|||
const dbPath = `/${config.name}.db`; |
|||
if (!(await fs.exists(dbPath))) { |
|||
await fs.writeFile(dbPath, new Uint8Array(0)); |
|||
} |
|||
|
|||
// Open database
|
|||
this.db = new SQL.Database(dbPath, { filename: true }); |
|||
|
|||
// Configure database settings
|
|||
if (config.useWAL) { |
|||
await this.execute("PRAGMA journal_mode = WAL"); |
|||
this.stats.walMode = true; |
|||
} |
|||
|
|||
// Set other pragmas for performance
|
|||
await this.execute("PRAGMA synchronous = NORMAL"); |
|||
await this.execute("PRAGMA temp_store = MEMORY"); |
|||
await this.execute("PRAGMA cache_size = -2000"); // Use 2MB of cache
|
|||
|
|||
this.initialized = true; |
|||
await this.updateStats(); |
|||
} catch (error) { |
|||
logger.error("Failed to initialize Web SQLite:", error); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
protected async _executeQuery<T>( |
|||
sql: string, |
|||
params: unknown[] = [], |
|||
operation: "query" | "execute" = "query", |
|||
): Promise<SQLiteResult<T>> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
if (operation === "query") { |
|||
const stmt = this.db.prepare(sql); |
|||
const results = stmt.get(params) as T[]; |
|||
stmt.free(); |
|||
return { results }; |
|||
} else { |
|||
const stmt = this.db.prepare(sql); |
|||
stmt.run(params); |
|||
const changes = this.db.getRowsModified(); |
|||
stmt.free(); |
|||
return { changes }; |
|||
} |
|||
} catch (error) { |
|||
logger.error("SQLite query failed:", { |
|||
sql, |
|||
params, |
|||
error: error instanceof Error ? error.message : String(error), |
|||
}); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
async close(): Promise<void> { |
|||
if (this.db) { |
|||
this.db.close(); |
|||
this.db = null; |
|||
} |
|||
if (this.worker) { |
|||
this.worker.terminate(); |
|||
this.worker = null; |
|||
} |
|||
this.initialized = false; |
|||
} |
|||
|
|||
async getDatabaseSize(): Promise<number> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
const result = await this.query<{ size: number }>("SELECT page_count * page_size as size FROM pragma_page_count(), pragma_page_size()"); |
|||
return result.results[0]?.size || 0; |
|||
} |
|||
|
|||
async prepare<T>(sql: string): Promise<PreparedStatement<T>> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
const stmt = this.db.prepare(sql); |
|||
const key = sql; |
|||
|
|||
const preparedStmt: PreparedStatement<T> = { |
|||
execute: async (params: unknown[] = []) => { |
|||
try { |
|||
const results = stmt.get(params) as T[]; |
|||
return { results }; |
|||
} catch (error) { |
|||
logger.error("Prepared statement execution failed:", { |
|||
sql, |
|||
params, |
|||
error: error instanceof Error ? error.message : String(error), |
|||
}); |
|||
throw error; |
|||
} |
|||
}, |
|||
finalize: () => { |
|||
stmt.free(); |
|||
this.preparedStatements.delete(key); |
|||
this.stats.preparedStatements--; |
|||
}, |
|||
}; |
|||
|
|||
this.preparedStatements.set(key, preparedStmt); |
|||
this.stats.preparedStatements++; |
|||
|
|||
return preparedStmt; |
|||
} |
|||
|
|||
async getStats(): Promise<SQLiteStats> { |
|||
await this.updateStats(); |
|||
return this.stats; |
|||
} |
|||
|
|||
private async updateStats(): Promise<void> { |
|||
if (!this.db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
const size = await this.getDatabaseSize(); |
|||
this.stats.databaseSize = size; |
|||
|
|||
const walResult = await this.query<{ journal_mode: string }>("PRAGMA journal_mode"); |
|||
this.stats.walMode = walResult.results[0]?.journal_mode === "wal"; |
|||
|
|||
const mmapResult = await this.query<{ mmap_size: number }>("PRAGMA mmap_size"); |
|||
this.stats.mmapActive = mmapResult.results[0]?.mmap_size > 0; |
|||
} |
|||
} |
@ -0,0 +1,150 @@ |
|||
import initSqlJs, { Database } from "@jlongster/sql.js"; |
|||
import { SQLiteFS } from "absurd-sql"; |
|||
import { IndexedDBBackend } from "absurd-sql/dist/indexeddb-backend"; |
|||
|
|||
interface WorkerMessage { |
|||
type: "init" | "query" | "execute" | "transaction" | "close"; |
|||
id: string; |
|||
dbName?: string; |
|||
sql?: string; |
|||
params?: unknown[]; |
|||
statements?: { sql: string; params?: unknown[] }[]; |
|||
} |
|||
|
|||
interface WorkerResponse { |
|||
id: string; |
|||
error?: string; |
|||
result?: unknown; |
|||
} |
|||
|
|||
let db: Database | null = null; |
|||
|
|||
async function initialize(dbName: string): Promise<void> { |
|||
if (db) { |
|||
return; |
|||
} |
|||
|
|||
const SQL = await initSqlJs({ |
|||
locateFile: (file: string) => `/sql-wasm/${file}`, |
|||
}); |
|||
|
|||
// Initialize the virtual file system
|
|||
const backend = new IndexedDBBackend(dbName); |
|||
const fs = new SQLiteFS(SQL.FS, backend); |
|||
SQL.register_for_idb(fs); |
|||
|
|||
// Create and initialize the database
|
|||
db = new SQL.Database(dbName, { |
|||
filename: true, |
|||
}); |
|||
|
|||
// Configure database settings
|
|||
db.exec("PRAGMA synchronous = NORMAL"); |
|||
db.exec("PRAGMA temp_store = MEMORY"); |
|||
db.exec("PRAGMA cache_size = -2000"); // Use 2MB of cache
|
|||
} |
|||
|
|||
async function executeQuery( |
|||
sql: string, |
|||
params: unknown[] = [], |
|||
): Promise<unknown> { |
|||
if (!db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
const stmt = db.prepare(sql); |
|||
try { |
|||
const rows: unknown[] = []; |
|||
stmt.bind(params); |
|||
while (stmt.step()) { |
|||
rows.push(stmt.getAsObject()); |
|||
} |
|||
return { |
|||
rows, |
|||
rowsAffected: db.getRowsModified(), |
|||
lastInsertId: db.exec("SELECT last_insert_rowid()")[0]?.values[0]?.[0], |
|||
}; |
|||
} finally { |
|||
stmt.free(); |
|||
} |
|||
} |
|||
|
|||
async function executeTransaction( |
|||
statements: { sql: string; params?: unknown[] }[], |
|||
): Promise<void> { |
|||
if (!db) { |
|||
throw new Error("Database not initialized"); |
|||
} |
|||
|
|||
try { |
|||
db.exec("BEGIN TRANSACTION"); |
|||
for (const { sql, params = [] } of statements) { |
|||
const stmt = db.prepare(sql); |
|||
try { |
|||
stmt.bind(params); |
|||
stmt.step(); |
|||
} finally { |
|||
stmt.free(); |
|||
} |
|||
} |
|||
db.exec("COMMIT"); |
|||
} catch (error) { |
|||
db.exec("ROLLBACK"); |
|||
throw error; |
|||
} |
|||
} |
|||
|
|||
async function close(): Promise<void> { |
|||
if (db) { |
|||
db.close(); |
|||
db = null; |
|||
} |
|||
} |
|||
|
|||
self.onmessage = async (event: MessageEvent<WorkerMessage>) => { |
|||
const { type, id, dbName, sql, params, statements } = event.data; |
|||
const response: WorkerResponse = { id }; |
|||
|
|||
try { |
|||
switch (type) { |
|||
case "init": |
|||
if (!dbName) { |
|||
throw new Error("Database name is required for initialization"); |
|||
} |
|||
await initialize(dbName); |
|||
break; |
|||
|
|||
case "query": |
|||
if (!sql) { |
|||
throw new Error("SQL query is required"); |
|||
} |
|||
response.result = await executeQuery(sql, params); |
|||
break; |
|||
|
|||
case "execute": |
|||
if (!sql) { |
|||
throw new Error("SQL statement is required"); |
|||
} |
|||
response.result = await executeQuery(sql, params); |
|||
break; |
|||
|
|||
case "transaction": |
|||
if (!statements?.length) { |
|||
throw new Error("Transaction statements are required"); |
|||
} |
|||
await executeTransaction(statements); |
|||
break; |
|||
|
|||
case "close": |
|||
await close(); |
|||
break; |
|||
|
|||
default: |
|||
throw new Error(`Unknown message type: ${type}`); |
|||
} |
|||
} catch (error) { |
|||
response.error = error instanceof Error ? error.message : String(error); |
|||
} |
|||
|
|||
self.postMessage(response); |
|||
}; |
@ -0,0 +1,45 @@ |
|||
declare module "@jlongster/sql.js" { |
|||
export interface Database { |
|||
exec( |
|||
sql: string, |
|||
params?: unknown[], |
|||
): { columns: string[]; values: unknown[][] }[]; |
|||
prepare(sql: string): Statement; |
|||
run(sql: string, params?: unknown[]): void; |
|||
getRowsModified(): number; |
|||
close(): void; |
|||
} |
|||
|
|||
export interface Statement { |
|||
step(): boolean; |
|||
getAsObject(): Record<string, unknown>; |
|||
bind(params: unknown[]): void; |
|||
reset(): void; |
|||
free(): void; |
|||
} |
|||
|
|||
export interface InitSqlJsStatic { |
|||
Database: new ( |
|||
filename?: string, |
|||
options?: { filename: boolean }, |
|||
) => Database; |
|||
FS: unknown; |
|||
register_for_idb(fs: unknown): void; |
|||
} |
|||
|
|||
export default function initSqlJs(options?: { |
|||
locateFile?: (file: string) => string; |
|||
}): Promise<InitSqlJsStatic>; |
|||
} |
|||
|
|||
declare module "absurd-sql" { |
|||
export class SQLiteFS { |
|||
constructor(fs: unknown, backend: unknown); |
|||
} |
|||
} |
|||
|
|||
declare module "absurd-sql/dist/indexeddb-backend" { |
|||
export class IndexedDBBackend { |
|||
constructor(dbName: string); |
|||
} |
|||
} |
@ -1,2 +1,2 @@ |
|||
// Empty module to satisfy Node.js built-in module imports
|
|||
export default {}; |
|||
export default {}; |
|||
|
@ -1,18 +1,18 @@ |
|||
// Minimal fs module implementation for browser
|
|||
const fs = { |
|||
readFileSync: () => { |
|||
throw new Error('fs.readFileSync is not supported in browser'); |
|||
throw new Error("fs.readFileSync is not supported in browser"); |
|||
}, |
|||
writeFileSync: () => { |
|||
throw new Error('fs.writeFileSync is not supported in browser'); |
|||
throw new Error("fs.writeFileSync is not supported in browser"); |
|||
}, |
|||
existsSync: () => false, |
|||
mkdirSync: () => {}, |
|||
readdirSync: () => [], |
|||
statSync: () => ({ |
|||
isDirectory: () => false, |
|||
isFile: () => false |
|||
}) |
|||
isFile: () => false, |
|||
}), |
|||
}; |
|||
|
|||
export default fs; |
|||
export default fs; |
|||
|
@ -1,13 +1,13 @@ |
|||
// Minimal path module implementation for browser
|
|||
const path = { |
|||
resolve: (...parts) => parts.join('/'), |
|||
join: (...parts) => parts.join('/'), |
|||
dirname: (p) => p.split('/').slice(0, -1).join('/'), |
|||
basename: (p) => p.split('/').pop(), |
|||
resolve: (...parts) => parts.join("/"), |
|||
join: (...parts) => parts.join("/"), |
|||
dirname: (p) => p.split("/").slice(0, -1).join("/"), |
|||
basename: (p) => p.split("/").pop(), |
|||
extname: (p) => { |
|||
const parts = p.split('.'); |
|||
return parts.length > 1 ? '.' + parts.pop() : ''; |
|||
} |
|||
const parts = p.split("."); |
|||
return parts.length > 1 ? "." + parts.pop() : ""; |
|||
}, |
|||
}; |
|||
|
|||
export default path; |
|||
export default path; |
|||
|
@ -1,4 +1,12 @@ |
|||
import { defineConfig } from "vite"; |
|||
import { createBuildConfig } from "./vite.config.common.mts"; |
|||
|
|||
export default defineConfig(async () => createBuildConfig('capacitor')); |
|||
export default defineConfig( |
|||
async () => { |
|||
const baseConfig = await createBuildConfig('capacitor'); |
|||
return mergeConfig(baseConfig, { |
|||
optimizeDeps: { |
|||
include: ['@capacitor-community/sqlite'] |
|||
} |
|||
}); |
|||
}); |
Loading…
Reference in new issue