This repository has been archived on 2024-06-08. You can view files and clone it, but cannot push or open issues or pull requests.
SolarAgent/lib/auth.dart

146 lines
3.8 KiB
Dart
Raw Normal View History

2024-02-07 20:40:56 +00:00
import 'dart:convert';
2024-02-07 17:25:58 +00:00
import 'dart:io';
import 'package:flutter/material.dart';
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
import 'package:solaragent/firebase.dart';
import 'package:solaragent/preferences.dart';
import 'package:solaragent/screens/auth.dart';
2024-02-07 17:25:58 +00:00
import 'package:oauth2/oauth2.dart' as oauth2;
final authClient = AuthGuard();
2024-02-07 20:40:56 +00:00
class AuthGuard {
AuthGuard();
2024-02-07 20:40:56 +00:00
2024-02-08 07:19:37 +00:00
final deviceEndpoint =
Uri.parse('https://id.solsynth.dev/api/notifications/subscribe');
2024-02-07 20:40:56 +00:00
final authorizationEndpoint =
Uri.parse('https://id.solsynth.dev/auth/o/connect');
2024-02-07 20:40:56 +00:00
final tokenEndpoint =
Uri.parse('https://id.solsynth.dev/api/auth/token');
2024-02-07 20:40:56 +00:00
final userinfoEndpoint =
Uri.parse('https://id.solsynth.dev/api/users/me');
final redirectUrl = Uri.parse('solaragent://auth');
2024-02-07 20:40:56 +00:00
static const clientId = "solaragent";
2024-02-07 20:40:56 +00:00
static const clientSecret = "_F4%q2Eea3";
static const storage = FlutterSecureStorage();
static const storageKey = "identity";
static const profileKey = "profiles";
oauth2.Client? client;
Future<bool> pickClient() async {
if (await storage.containsKey(key: storageKey)) {
2024-02-08 07:19:37 +00:00
try {
var credentials =
oauth2.Credentials.fromJson((await storage.read(key: storageKey))!);
client = oauth2.Client(credentials,
identifier: clientId, secret: clientSecret);
await pullProfiles();
return true;
} catch (e) {
logout();
return false;
}
2024-02-07 20:40:56 +00:00
} else {
return false;
}
2024-02-07 17:25:58 +00:00
}
2024-02-07 20:40:56 +00:00
Future<oauth2.Client> createClient(BuildContext context) async {
// If logged in
if (await pickClient()) {
return client!;
}
var grant = oauth2.AuthorizationCodeGrant(
clientId,
authorizationEndpoint,
tokenEndpoint,
secret: clientSecret,
basicAuth: false,
);
2024-02-07 17:25:58 +00:00
2024-02-07 20:40:56 +00:00
var authorizationUrl = grant.getAuthorizationUrl(redirectUrl);
2024-02-07 17:25:58 +00:00
2024-02-07 20:40:56 +00:00
if (Platform.isAndroid || Platform.isIOS) {
// Use WebView to get authorization url
2024-02-10 12:08:25 +00:00
var responseUrl = await Navigator.of(context, rootNavigator: true).push(
MaterialPageRoute(
builder: (context) => AuthorizationScreen(authorizationUrl),
),
);
2024-02-07 20:40:56 +00:00
var responseUri = Uri.parse(responseUrl);
return await grant
.handleAuthorizationResponse(responseUri.queryParameters);
} else {
throw UnimplementedError("unsupported platform");
2024-02-07 17:25:58 +00:00
}
2024-02-07 20:40:56 +00:00
}
2024-02-07 17:25:58 +00:00
2024-02-08 07:19:37 +00:00
Future<void> pullProfiles() async {
if (client != null) {
var userinfo = await client!.get(userinfoEndpoint);
storage.write(key: profileKey, value: utf8.decode(userinfo.bodyBytes));
}
}
2024-02-07 20:40:56 +00:00
Future<void> login(BuildContext context) async {
2024-02-07 17:25:58 +00:00
try {
2024-02-07 20:40:56 +00:00
client = await createClient(context);
storage.write(key: storageKey, value: client!.credentials.toJson());
2024-02-08 07:19:37 +00:00
await pullProfiles();
await subscribeNotify();
2024-02-07 21:03:38 +00:00
} catch (e) {
print(e);
}
}
2024-02-07 20:40:56 +00:00
2024-02-08 07:19:37 +00:00
Future<void> subscribeNotify() async {
if (client == null) {
return;
}
var token = await initializeFirebaseMessaging();
if (token == null) {
print("failed to initialize firebase messaging...");
return;
}
var response = await client!.post(
deviceEndpoint,
headers: {"Content-Type": "application/json"},
body: jsonEncode({"device_id": token, "provider": "firebase"}),
);
if (response.statusCode != 200) {
print(response.body);
}
}
2024-02-07 21:03:38 +00:00
void logout() {
try {
storage.delete(key: profileKey);
storage.delete(key: storageKey);
2024-02-07 17:25:58 +00:00
} catch (e) {
print(e);
}
}
2024-02-07 20:40:56 +00:00
Future<bool> isAuthorized() async {
const storage = FlutterSecureStorage();
return await storage.containsKey(key: storageKey);
2024-02-07 17:25:58 +00:00
}
2024-02-07 20:40:56 +00:00
Future<dynamic> readProfiles() async {
const storage = FlutterSecureStorage();
return jsonDecode(await storage.read(key: profileKey) ?? "{}");
}
AuthGuard._internal();
2024-02-07 17:25:58 +00:00
}