import 'dart:async';
import 'package:flutter/foundation.dart' show kDebugMode;
import 'package:path/path.dart' as p;
import 'package:sqflite/sqflite.dart';
var wipeOnStartFromEnv =
bool.hasEnvironment('DB_WIPE_ON_START')
? (String.fromEnvironment('DB_WIPE_ON_START').toLowerCase() == 'true')
: false;
class DatabaseService {
static final DatabaseService _instance = DatabaseService._internal();
factory DatabaseService() => _instance;
DatabaseService._internal();
static const _dbName = 'app.db';
static const _dbVersion = 3;
Database? _db;
Future<Database> get db async {
final existing = _db;
if (existing != null) return existing;
return _db = await _open();
}
Future<Database> _open() async {
final dbPath = await getDatabasesPath();
final path = p.join(dbPath, _dbName);
if (kDebugMode && wipeOnStartFromEnv) {
await deleteDatabase(path);
}
return await openDatabase(
path,
version: _dbVersion,
onConfigure: (db) async {
await db.execute('PRAGMA foreign_keys = ON;');
},
onCreate: _onCreate,
onUpgrade: _onUpgrade,
onDowngrade: _onDowngrade,
);
}
Future<void> _onCreate(Database db, int version) async {
await db.execute('''
CREATE TABLE vehicle (
id INTEGER PRIMARY KEY AUTOINCREMENT,
make TEXT NOT NULL,
model TEXT NOT NULL,
created_at INTEGER NOT NULL
);
''');
if (version > 1) {
await _runMigrations(db, 1, version);
}
}
Future<void> _onUpgrade(Database db, int oldVersion, int newVersion) async {
await _runMigrations(db, oldVersion, newVersion);
}
Future<void> _onDowngrade(Database db, int oldV, int newV) async {
}
final Map<int, Future<void> Function(DatabaseExecutor db)> _migrations = {
2: (db) async {
await db.execute('ALTER TABLE vehicle ADD COLUMN plate TEXT;');
},
3: (db) async {
await db.execute('''
CREATE TABLE payment_method (
id INTEGER PRIMARY KEY AUTOINCREMENT,
type TEXT NOT NULL, -- например: 'card', 'cash', 'paypal'
last4 TEXT, -- для карт (опционально)
created_at INTEGER NOT NULL
);
''');
},
};
Future<void> _runMigrations(Database db, int from, int to) async {
await db.transaction((txn) async {
for (var v = from + 1; v <= to; v++) {
final step = _migrations[v];
if (step != null) {
await step(txn);
}
}
});
}
Future<int> insertVehicle({
required String make,
required String model,
String? plate,
int? createdAtMillis,
}) async {
final database = await db;
return await database.insert('vehicle', {
'make': make,
'model': model,
'plate': plate,
'created_at': createdAtMillis ?? DateTime.now().millisecondsSinceEpoch,
});
}
Future<List<Map<String, Object?>>> getVehicles() async {
final database = await db;
return await database.query('vehicle', orderBy: 'id DESC');
}
Future<int> insertPaymentMethod({
required String type,
String? last4,
int? createdAtMillis,
}) async {
final database = await db;
return await database.insert('payment_method', {
'type': type,
'last4': last4,
'created_at': createdAtMillis ?? DateTime.now().millisecondsSinceEpoch,
});
}
Future<List<Map<String, Object?>>> getPaymentMethods() async {
final database = await db;
return await database.query('payment_method', orderBy: 'id DESC');
}
}