mirror of
https://github.com/Yubico/yubioath-flutter.git
synced 2024-11-27 14:23:18 +03:00
270 lines
8.2 KiB
Dart
Executable File
270 lines
8.2 KiB
Dart
Executable File
/*
|
|
* Copyright (C) 2022-2023 Yubico.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
import 'dart:async';
|
|
import 'dart:convert';
|
|
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
|
|
import 'package:logging/logging.dart';
|
|
|
|
import '../../app/logging.dart';
|
|
import '../../app/models.dart';
|
|
import '../../app/state.dart';
|
|
import '../../app/views/user_interaction.dart';
|
|
import '../../core/models.dart';
|
|
import '../../exception/platform_exception_decoder.dart';
|
|
import '../../oath/models.dart';
|
|
import '../../oath/state.dart';
|
|
|
|
final _log = Logger('android.oath.state');
|
|
|
|
const _methods = MethodChannel('android.oath.methods');
|
|
|
|
final androidOathStateProvider = StateNotifierProvider.autoDispose
|
|
.family<OathStateNotifier, AsyncValue<OathState>, DevicePath>(
|
|
(ref, devicePath) => _AndroidOathStateNotifier());
|
|
|
|
class _AndroidOathStateNotifier extends OathStateNotifier {
|
|
final _events = const EventChannel('android.oath.sessionState');
|
|
late StreamSubscription _sub;
|
|
_AndroidOathStateNotifier() : super() {
|
|
_sub = _events.receiveBroadcastStream().listen((event) {
|
|
final json = jsonDecode(event);
|
|
if (mounted) {
|
|
if (json == null) {
|
|
state = const AsyncValue.loading();
|
|
} else {
|
|
final oathState = OathState.fromJson(json);
|
|
state = AsyncValue.data(oathState);
|
|
}
|
|
}
|
|
}, onError: (err, stackTrace) {
|
|
state = AsyncValue.error(err, stackTrace);
|
|
});
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_sub.cancel();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Future<void> reset() async {
|
|
try {
|
|
await _methods.invokeMethod('reset');
|
|
} catch (e) {
|
|
_log.debug('Calling reset failed with exception: $e');
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<Pair<bool, bool>> unlock(String password,
|
|
{bool remember = false}) async {
|
|
try {
|
|
final unlockResponse = jsonDecode(await _methods.invokeMethod(
|
|
'unlock', {'password': password, 'remember': remember}));
|
|
_log.debug('applet unlocked');
|
|
|
|
final unlocked = unlockResponse['unlocked'] == true;
|
|
final remembered = unlockResponse['remembered'] == true;
|
|
|
|
return Pair(unlocked, remembered);
|
|
} on PlatformException catch (e) {
|
|
_log.debug('Calling unlock failed with exception: $e');
|
|
return Pair(false, false);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<bool> setPassword(String? current, String password) async {
|
|
try {
|
|
await _methods.invokeMethod(
|
|
'setPassword', {'current': current, 'password': password});
|
|
return true;
|
|
} on PlatformException catch (e) {
|
|
_log.debug('Calling set password failed with exception: $e');
|
|
return false;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<bool> unsetPassword(String current) async {
|
|
try {
|
|
await _methods.invokeMethod('unsetPassword', {'current': current});
|
|
return true;
|
|
} on PlatformException catch (e) {
|
|
_log.debug('Calling unset password failed with exception: $e');
|
|
return false;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> forgetPassword() async {
|
|
try {
|
|
await _methods.invokeMethod('forgetPassword');
|
|
} on PlatformException catch (e) {
|
|
_log.debug('Calling forgetPassword failed with exception: $e');
|
|
}
|
|
}
|
|
}
|
|
|
|
final addCredentialToAnyProvider =
|
|
Provider((ref) => (Uri credentialUri, {bool requireTouch = false}) async {
|
|
try {
|
|
String resultString = await _methods.invokeMethod(
|
|
'addAccountToAny', {
|
|
'uri': credentialUri.toString(),
|
|
'requireTouch': requireTouch
|
|
});
|
|
|
|
var result = jsonDecode(resultString);
|
|
return OathCredential.fromJson(result['credential']);
|
|
} on PlatformException catch (pe) {
|
|
_log.error('Failed to add account.', pe);
|
|
throw pe.decode();
|
|
}
|
|
});
|
|
|
|
final androidCredentialListProvider = StateNotifierProvider.autoDispose
|
|
.family<OathCredentialListNotifier, List<OathPair>?, DevicePath>(
|
|
(ref, devicePath) {
|
|
var notifier = _AndroidCredentialListNotifier(
|
|
ref.watch(withContextProvider),
|
|
ref.watch(currentDeviceProvider)?.transport == Transport.usb,
|
|
);
|
|
return notifier;
|
|
},
|
|
);
|
|
|
|
class _AndroidCredentialListNotifier extends OathCredentialListNotifier {
|
|
final _events = const EventChannel('android.oath.credentials');
|
|
final WithContext _withContext;
|
|
final bool _isUsbAttached;
|
|
late StreamSubscription _sub;
|
|
|
|
_AndroidCredentialListNotifier(this._withContext, this._isUsbAttached)
|
|
: super() {
|
|
_sub = _events.receiveBroadcastStream().listen((event) {
|
|
final json = jsonDecode(event);
|
|
List<OathPair>? newState = json != null
|
|
? List.from((json as List).map((e) => OathPair.fromJson(e)).toList())
|
|
: null;
|
|
if (state != null && newState == null) {
|
|
// If we go from non-null to null this means we should stop listening to
|
|
// avoid receiving a message for a different notifier as there is only
|
|
// one channel.
|
|
_sub.cancel();
|
|
}
|
|
state = newState;
|
|
});
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_sub.cancel();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Future<OathCode> calculate(OathCredential credential) async {
|
|
// Prompt for touch if needed
|
|
UserInteractionController? controller;
|
|
Timer? touchTimer;
|
|
if (_isUsbAttached) {
|
|
void triggerTouchPrompt() async {
|
|
controller = await _withContext(
|
|
(context) async {
|
|
final l10n = AppLocalizations.of(context)!;
|
|
return promptUserInteraction(
|
|
context,
|
|
icon: const Icon(Icons.touch_app),
|
|
title: l10n.s_touch_required,
|
|
description: l10n.l_touch_button_now,
|
|
);
|
|
},
|
|
);
|
|
}
|
|
|
|
if (credential.touchRequired) {
|
|
triggerTouchPrompt();
|
|
} else if (credential.oathType == OathType.hotp) {
|
|
touchTimer =
|
|
Timer(const Duration(milliseconds: 500), triggerTouchPrompt);
|
|
}
|
|
}
|
|
|
|
try {
|
|
final resultJson = await _methods
|
|
.invokeMethod('calculate', {'credentialId': credential.id});
|
|
_log.debug('Calculate', resultJson);
|
|
return OathCode.fromJson(jsonDecode(resultJson));
|
|
} on PlatformException catch (pe) {
|
|
throw pe.decode();
|
|
} finally {
|
|
touchTimer?.cancel();
|
|
controller?.close();
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<OathCredential> addAccount(Uri credentialUri,
|
|
{bool requireTouch = false}) async {
|
|
try {
|
|
String resultString = await _methods.invokeMethod('addAccount',
|
|
{'uri': credentialUri.toString(), 'requireTouch': requireTouch});
|
|
|
|
var result = jsonDecode(resultString);
|
|
return OathCredential.fromJson(result['credential']);
|
|
} on PlatformException catch (pe) {
|
|
_log.error('Failed to add account.', pe);
|
|
throw pe.decode();
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<OathCredential> renameAccount(
|
|
OathCredential credential, String? issuer, String name) async {
|
|
try {
|
|
final response = await _methods.invokeMethod('renameAccount',
|
|
{'credentialId': credential.id, 'name': name, 'issuer': issuer});
|
|
|
|
_log.debug('Rename response: $response');
|
|
|
|
var responseJson = jsonDecode(response);
|
|
|
|
return OathCredential.fromJson(responseJson);
|
|
} on PlatformException catch (pe) {
|
|
_log.debug('Failed to execute renameOathCredential: ${pe.message}');
|
|
throw pe.decode();
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> deleteAccount(OathCredential credential) async {
|
|
try {
|
|
await _methods
|
|
.invokeMethod('deleteAccount', {'credentialId': credential.id});
|
|
} on PlatformException catch (e) {
|
|
_log.debug('Received exception: $e');
|
|
throw e.decode();
|
|
}
|
|
}
|
|
}
|