mirror of
https://github.com/Yubico/yubioath-flutter.git
synced 2024-12-23 10:11:52 +03:00
Add universal factory reset.
This commit is contained in:
parent
2db9c5e2d1
commit
9991eba44b
@ -25,6 +25,7 @@ import '../shortcuts.dart';
|
||||
import '../state.dart';
|
||||
import 'device_picker.dart';
|
||||
import 'keys.dart';
|
||||
import 'reset_dialog.dart';
|
||||
|
||||
class NavigationItem extends StatelessWidget {
|
||||
final Widget leading;
|
||||
@ -180,6 +181,7 @@ class NavigationContent extends ConsumerWidget {
|
||||
},
|
||||
)),
|
||||
// Management app
|
||||
const SizedBox(height: 32),
|
||||
if (hasManagement) ...[
|
||||
NavigationItem(
|
||||
key: managementAppDrawer,
|
||||
@ -197,6 +199,20 @@ class NavigationContent extends ConsumerWidget {
|
||||
},
|
||||
),
|
||||
],
|
||||
// TODO: This doesn't belong here longterm
|
||||
NavigationItem(
|
||||
key: const Key('app.keys.drawer.reset'),
|
||||
leading: const Icon(Icons.delete_forever),
|
||||
title: l10n.s_factory_reset,
|
||||
collapsed: !extended,
|
||||
onTap: () {
|
||||
showBlurDialog(
|
||||
context: context,
|
||||
// data must be non-null when index == 0
|
||||
builder: (context) => ResetDialog(data),
|
||||
);
|
||||
},
|
||||
),
|
||||
const SizedBox(height: 32),
|
||||
],
|
||||
],
|
||||
|
204
lib/app/views/reset_dialog.dart
Normal file
204
lib/app/views/reset_dialog.dart
Normal file
@ -0,0 +1,204 @@
|
||||
import 'dart:async';
|
||||
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
|
||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||
import 'package:logging/logging.dart';
|
||||
|
||||
import '../../app/logging.dart';
|
||||
import '../../core/models.dart';
|
||||
import '../../desktop/models.dart';
|
||||
import '../../fido/models.dart';
|
||||
import '../../fido/state.dart';
|
||||
import '../../management/models.dart';
|
||||
import '../../oath/state.dart';
|
||||
import '../../piv/state.dart';
|
||||
import '../../widgets/responsive_dialog.dart';
|
||||
import '../message.dart';
|
||||
import '../models.dart';
|
||||
import '../state.dart';
|
||||
|
||||
final _log = Logger('fido.views.reset_dialog');
|
||||
|
||||
class ResetDialog extends ConsumerStatefulWidget {
|
||||
final YubiKeyData data;
|
||||
const ResetDialog(this.data, {super.key});
|
||||
|
||||
@override
|
||||
ConsumerState<ConsumerStatefulWidget> createState() => _ResetDialogState();
|
||||
}
|
||||
|
||||
class _ResetDialogState extends ConsumerState<ResetDialog> {
|
||||
// TODO: Capabilities based on enabled/supported. Feature checks.
|
||||
Capability? _application;
|
||||
StreamSubscription<InteractionEvent>? _subscription;
|
||||
InteractionEvent? _interaction;
|
||||
int _currentStep = -1;
|
||||
final _totalSteps = 3;
|
||||
|
||||
String _getMessage() {
|
||||
final l10n = AppLocalizations.of(context)!;
|
||||
final nfc = widget.data.node.transport == Transport.nfc;
|
||||
if (_currentStep == 3) {
|
||||
return l10n.l_fido_app_reset;
|
||||
}
|
||||
return switch (_interaction) {
|
||||
InteractionEvent.remove =>
|
||||
nfc ? l10n.l_remove_yk_from_reader : l10n.l_unplug_yk,
|
||||
InteractionEvent.insert =>
|
||||
nfc ? l10n.l_replace_yk_on_reader : l10n.l_reinsert_yk,
|
||||
InteractionEvent.touch => l10n.l_touch_button_now,
|
||||
null => ''
|
||||
};
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final supported =
|
||||
widget.data.info.supportedCapabilities[widget.data.node.transport] ?? 0;
|
||||
final enabled = widget
|
||||
.data.info.config.enabledCapabilities[widget.data.node.transport] ??
|
||||
0;
|
||||
final l10n = AppLocalizations.of(context)!;
|
||||
double progress = _currentStep == -1 ? 0.0 : _currentStep / (_totalSteps);
|
||||
return ResponsiveDialog(
|
||||
title: Text(l10n.s_factory_reset),
|
||||
onCancel: switch (_application) {
|
||||
Capability.fido2 => _currentStep < 3
|
||||
? () {
|
||||
_subscription?.cancel();
|
||||
}
|
||||
: null,
|
||||
_ => null,
|
||||
},
|
||||
actions: [
|
||||
if (_currentStep < 3)
|
||||
TextButton(
|
||||
onPressed: switch (_application) {
|
||||
Capability.fido2 => _subscription == null
|
||||
? () async {
|
||||
_subscription = ref
|
||||
.read(
|
||||
fidoStateProvider(widget.data.node.path).notifier)
|
||||
.reset()
|
||||
.listen((event) {
|
||||
setState(() {
|
||||
_currentStep++;
|
||||
_interaction = event;
|
||||
});
|
||||
}, onDone: () {
|
||||
setState(() {
|
||||
_currentStep++;
|
||||
});
|
||||
_subscription = null;
|
||||
}, onError: (e) {
|
||||
_log.error('Error performing FIDO reset', e);
|
||||
Navigator.of(context).pop();
|
||||
final String errorMessage;
|
||||
// TODO: Make this cleaner than importing desktop specific RpcError.
|
||||
if (e is RpcError) {
|
||||
if (e.status == 'connection-error') {
|
||||
errorMessage = l10n.l_failed_connecting_to_fido;
|
||||
} else if (e.status == 'key-mismatch') {
|
||||
errorMessage = l10n.l_wrong_inserted_yk_error;
|
||||
} else if (e.status == 'user-action-timeout') {
|
||||
errorMessage = l10n.l_user_action_timeout_error;
|
||||
} else {
|
||||
errorMessage = e.message;
|
||||
}
|
||||
} else {
|
||||
errorMessage = e.toString();
|
||||
}
|
||||
showMessage(
|
||||
context,
|
||||
l10n.l_reset_failed(errorMessage),
|
||||
duration: const Duration(seconds: 4),
|
||||
);
|
||||
});
|
||||
}
|
||||
: null,
|
||||
Capability.oath => () async {
|
||||
await ref
|
||||
.read(oathStateProvider(widget.data.node.path).notifier)
|
||||
.reset();
|
||||
await ref.read(withContextProvider)((context) async {
|
||||
Navigator.of(context).pop();
|
||||
showMessage(context, l10n.l_oath_application_reset);
|
||||
});
|
||||
},
|
||||
Capability.piv => () async {
|
||||
await ref
|
||||
.read(pivStateProvider(widget.data.node.path).notifier)
|
||||
.reset();
|
||||
await ref.read(withContextProvider)((context) async {
|
||||
Navigator.of(context).pop();
|
||||
showMessage(context, l10n.l_piv_app_reset);
|
||||
});
|
||||
},
|
||||
null => null,
|
||||
_ => throw UnsupportedError('Application cannot be reset'),
|
||||
},
|
||||
child: Text(l10n.s_reset),
|
||||
)
|
||||
],
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 18.0),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
SegmentedButton<Capability>(
|
||||
emptySelectionAllowed: true,
|
||||
segments: [
|
||||
Capability.oath,
|
||||
Capability.fido2,
|
||||
Capability.piv,
|
||||
]
|
||||
.where((c) => supported & c.value != 0)
|
||||
.map((c) => ButtonSegment(
|
||||
value: c,
|
||||
label: Text(c.getDisplayName(l10n)),
|
||||
enabled: enabled & c.value != 0,
|
||||
))
|
||||
.toList(),
|
||||
selected: _application != null ? {_application!} : {},
|
||||
onSelectionChanged: (selected) {
|
||||
setState(() {
|
||||
_application = selected.first;
|
||||
});
|
||||
},
|
||||
),
|
||||
Text(
|
||||
switch (_application) {
|
||||
Capability.oath => l10n.p_warning_factory_reset,
|
||||
Capability.piv => l10n.p_warning_piv_reset,
|
||||
Capability.fido2 => l10n.p_warning_deletes_accounts,
|
||||
_ => 'Factory reset an application.',
|
||||
},
|
||||
style: Theme.of(context)
|
||||
.textTheme
|
||||
.bodyMedium
|
||||
?.copyWith(fontWeight: FontWeight.w700),
|
||||
),
|
||||
Text(
|
||||
switch (_application) {
|
||||
Capability.oath => l10n.p_warning_disable_credentials,
|
||||
Capability.piv => l10n.p_warning_piv_reset_desc,
|
||||
Capability.fido2 => l10n.p_warning_disable_accounts,
|
||||
_ => 'Select an application above to reset.',
|
||||
},
|
||||
),
|
||||
if (_application == Capability.fido2 && _currentStep > 0) ...[
|
||||
Text('${l10n.s_status}: ${_getMessage()}'),
|
||||
LinearProgressIndicator(value: progress)
|
||||
],
|
||||
]
|
||||
.map((e) => Padding(
|
||||
padding: const EdgeInsets.symmetric(vertical: 8.0),
|
||||
child: e,
|
||||
))
|
||||
.toList(),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user