feat(useragent): vibe-coded access list
This commit is contained in:
19
useragent/lib/providers/sdk_clients/details.dart
Normal file
19
useragent/lib/providers/sdk_clients/details.dart
Normal file
@@ -0,0 +1,19 @@
|
||||
import 'package:arbiter/proto/user_agent.pb.dart';
|
||||
import 'package:arbiter/providers/sdk_clients/list.dart';
|
||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||
|
||||
part 'details.g.dart';
|
||||
|
||||
@riverpod
|
||||
Future<SdkClientEntry?> clientDetails(Ref ref, int clientId) async {
|
||||
final clients = await ref.watch(sdkClientsProvider.future);
|
||||
if (clients == null) {
|
||||
return null;
|
||||
}
|
||||
for (final client in clients) {
|
||||
if (client.id == clientId) {
|
||||
return client;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
85
useragent/lib/providers/sdk_clients/details.g.dart
Normal file
85
useragent/lib/providers/sdk_clients/details.g.dart
Normal file
@@ -0,0 +1,85 @@
|
||||
// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||
|
||||
part of 'details.dart';
|
||||
|
||||
// **************************************************************************
|
||||
// RiverpodGenerator
|
||||
// **************************************************************************
|
||||
|
||||
// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||
// ignore_for_file: type=lint, type=warning
|
||||
|
||||
@ProviderFor(clientDetails)
|
||||
final clientDetailsProvider = ClientDetailsFamily._();
|
||||
|
||||
final class ClientDetailsProvider
|
||||
extends
|
||||
$FunctionalProvider<
|
||||
AsyncValue<SdkClientEntry?>,
|
||||
SdkClientEntry?,
|
||||
FutureOr<SdkClientEntry?>
|
||||
>
|
||||
with $FutureModifier<SdkClientEntry?>, $FutureProvider<SdkClientEntry?> {
|
||||
ClientDetailsProvider._({
|
||||
required ClientDetailsFamily super.from,
|
||||
required int super.argument,
|
||||
}) : super(
|
||||
retry: null,
|
||||
name: r'clientDetailsProvider',
|
||||
isAutoDispose: true,
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
);
|
||||
|
||||
@override
|
||||
String debugGetCreateSourceHash() => _$clientDetailsHash();
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return r'clientDetailsProvider'
|
||||
''
|
||||
'($argument)';
|
||||
}
|
||||
|
||||
@$internal
|
||||
@override
|
||||
$FutureProviderElement<SdkClientEntry?> $createElement(
|
||||
$ProviderPointer pointer,
|
||||
) => $FutureProviderElement(pointer);
|
||||
|
||||
@override
|
||||
FutureOr<SdkClientEntry?> create(Ref ref) {
|
||||
final argument = this.argument as int;
|
||||
return clientDetails(ref, argument);
|
||||
}
|
||||
|
||||
@override
|
||||
bool operator ==(Object other) {
|
||||
return other is ClientDetailsProvider && other.argument == argument;
|
||||
}
|
||||
|
||||
@override
|
||||
int get hashCode {
|
||||
return argument.hashCode;
|
||||
}
|
||||
}
|
||||
|
||||
String _$clientDetailsHash() => r'21449a1a2cc4fa4e65ce761e6342e97c1d957a7a';
|
||||
|
||||
final class ClientDetailsFamily extends $Family
|
||||
with $FunctionalFamilyOverride<FutureOr<SdkClientEntry?>, int> {
|
||||
ClientDetailsFamily._()
|
||||
: super(
|
||||
retry: null,
|
||||
name: r'clientDetailsProvider',
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
isAutoDispose: true,
|
||||
);
|
||||
|
||||
ClientDetailsProvider call(int clientId) =>
|
||||
ClientDetailsProvider._(argument: clientId, from: this);
|
||||
|
||||
@override
|
||||
String toString() => r'clientDetailsProvider';
|
||||
}
|
||||
@@ -1,25 +1,174 @@
|
||||
|
||||
import 'package:arbiter/proto/user_agent.pb.dart';
|
||||
import 'package:arbiter/features/connection/evm/wallet_access.dart';
|
||||
import 'package:arbiter/proto/evm.pb.dart';
|
||||
import 'package:arbiter/providers/connection/connection_manager.dart';
|
||||
import 'package:mtcore/markettakers.dart';
|
||||
import 'package:protobuf/well_known_types/google/protobuf/empty.pb.dart';
|
||||
import 'package:arbiter/providers/evm/evm.dart';
|
||||
import 'package:flutter/foundation.dart';
|
||||
import 'package:hooks_riverpod/experimental/mutation.dart';
|
||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||
|
||||
part 'wallet_access.g.dart';
|
||||
|
||||
@riverpod
|
||||
Future<List<SdkClientWalletAccess>?> walletAccess(Ref ref) async {
|
||||
final connection = await ref.watch(connectionManagerProvider.future);
|
||||
if (connection == null) {
|
||||
return null;
|
||||
}
|
||||
class ClientWalletOption {
|
||||
const ClientWalletOption({required this.walletId, required this.address});
|
||||
|
||||
final accesses = await connection.ask(UserAgentRequest(listWalletAccess: Empty()));
|
||||
final int walletId;
|
||||
final String address;
|
||||
}
|
||||
|
||||
if (accesses.hasListWalletAccessResponse()) {
|
||||
return accesses.listWalletAccessResponse.accesses.toList();
|
||||
} else {
|
||||
talker.warning('Received unexpected response for listWalletAccess: $accesses');
|
||||
return null;
|
||||
class ClientWalletAccessState {
|
||||
const ClientWalletAccessState({
|
||||
this.searchQuery = '',
|
||||
this.originalWalletIds = const {},
|
||||
this.selectedWalletIds = const {},
|
||||
});
|
||||
|
||||
final String searchQuery;
|
||||
final Set<int> originalWalletIds;
|
||||
final Set<int> selectedWalletIds;
|
||||
|
||||
bool get hasChanges => !setEquals(originalWalletIds, selectedWalletIds);
|
||||
|
||||
ClientWalletAccessState copyWith({
|
||||
String? searchQuery,
|
||||
Set<int>? originalWalletIds,
|
||||
Set<int>? selectedWalletIds,
|
||||
}) {
|
||||
return ClientWalletAccessState(
|
||||
searchQuery: searchQuery ?? this.searchQuery,
|
||||
originalWalletIds: originalWalletIds ?? this.originalWalletIds,
|
||||
selectedWalletIds: selectedWalletIds ?? this.selectedWalletIds,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
final saveClientWalletAccessMutation = Mutation<void>();
|
||||
|
||||
abstract class ClientWalletAccessRepository {
|
||||
Future<Set<int>> fetchSelectedWalletIds(int clientId);
|
||||
Future<void> saveSelectedWalletIds(int clientId, Set<int> walletIds);
|
||||
}
|
||||
|
||||
class ServerClientWalletAccessRepository
|
||||
implements ClientWalletAccessRepository {
|
||||
ServerClientWalletAccessRepository(this.ref);
|
||||
|
||||
final Ref ref;
|
||||
|
||||
@override
|
||||
Future<Set<int>> fetchSelectedWalletIds(int clientId) async {
|
||||
final connection = await ref.read(connectionManagerProvider.future);
|
||||
if (connection == null) {
|
||||
throw Exception('Not connected to the server.');
|
||||
}
|
||||
return readClientWalletAccess(connection, clientId: clientId);
|
||||
}
|
||||
|
||||
@override
|
||||
Future<void> saveSelectedWalletIds(int clientId, Set<int> walletIds) async {
|
||||
final connection = await ref.read(connectionManagerProvider.future);
|
||||
if (connection == null) {
|
||||
throw Exception('Not connected to the server.');
|
||||
}
|
||||
await writeClientWalletAccess(
|
||||
connection,
|
||||
clientId: clientId,
|
||||
walletIds: walletIds,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@riverpod
|
||||
ClientWalletAccessRepository clientWalletAccessRepository(Ref ref) {
|
||||
return ServerClientWalletAccessRepository(ref);
|
||||
}
|
||||
|
||||
@riverpod
|
||||
Future<List<ClientWalletOption>> clientWalletOptions(Ref ref) async {
|
||||
final wallets = await ref.watch(evmProvider.future) ?? const <WalletEntry>[];
|
||||
return [
|
||||
for (var index = 0; index < wallets.length; index++)
|
||||
ClientWalletOption(
|
||||
walletId: index + 1,
|
||||
address: formatWalletAddress(wallets[index].address),
|
||||
),
|
||||
];
|
||||
}
|
||||
|
||||
@riverpod
|
||||
Future<Set<int>> clientWalletAccessSelection(Ref ref, int clientId) async {
|
||||
final repository = ref.watch(clientWalletAccessRepositoryProvider);
|
||||
return repository.fetchSelectedWalletIds(clientId);
|
||||
}
|
||||
|
||||
@riverpod
|
||||
class ClientWalletAccessController extends _$ClientWalletAccessController {
|
||||
@override
|
||||
ClientWalletAccessState build(int clientId) {
|
||||
final selection = ref.read(clientWalletAccessSelectionProvider(clientId));
|
||||
|
||||
void sync(AsyncValue<Set<int>> value) {
|
||||
value.when(data: hydrate, error: (_, _) {}, loading: () {});
|
||||
}
|
||||
|
||||
ref.listen<AsyncValue<Set<int>>>(
|
||||
clientWalletAccessSelectionProvider(clientId),
|
||||
(_, next) => sync(next),
|
||||
);
|
||||
return selection.when(
|
||||
data: (walletIds) => ClientWalletAccessState(
|
||||
originalWalletIds: Set.of(walletIds),
|
||||
selectedWalletIds: Set.of(walletIds),
|
||||
),
|
||||
error: (error, _) => const ClientWalletAccessState(),
|
||||
loading: () => const ClientWalletAccessState(),
|
||||
);
|
||||
}
|
||||
|
||||
void hydrate(Set<int> selectedWalletIds) {
|
||||
state = state.copyWith(
|
||||
originalWalletIds: Set.of(selectedWalletIds),
|
||||
selectedWalletIds: Set.of(selectedWalletIds),
|
||||
);
|
||||
}
|
||||
|
||||
void setSearchQuery(String value) {
|
||||
state = state.copyWith(searchQuery: value);
|
||||
}
|
||||
|
||||
void toggleWallet(int walletId) {
|
||||
final next = Set<int>.of(state.selectedWalletIds);
|
||||
if (!next.add(walletId)) {
|
||||
next.remove(walletId);
|
||||
}
|
||||
state = state.copyWith(selectedWalletIds: next);
|
||||
}
|
||||
|
||||
void discardChanges() {
|
||||
state = state.copyWith(selectedWalletIds: Set.of(state.originalWalletIds));
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> executeSaveClientWalletAccess(
|
||||
MutationTarget ref, {
|
||||
required int clientId,
|
||||
}) {
|
||||
final mutation = saveClientWalletAccessMutation(clientId);
|
||||
return mutation.run(ref, (tsx) async {
|
||||
final repository = tsx.get(clientWalletAccessRepositoryProvider);
|
||||
final controller = tsx.get(
|
||||
clientWalletAccessControllerProvider(clientId).notifier,
|
||||
);
|
||||
final selectedWalletIds = tsx
|
||||
.get(clientWalletAccessControllerProvider(clientId))
|
||||
.selectedWalletIds;
|
||||
await repository.saveSelectedWalletIds(clientId, selectedWalletIds);
|
||||
controller.hydrate(selectedWalletIds);
|
||||
});
|
||||
}
|
||||
|
||||
String formatWalletAddress(List<int> bytes) {
|
||||
final hex = bytes
|
||||
.map((byte) => byte.toRadixString(16).padLeft(2, '0'))
|
||||
.join();
|
||||
return '0x$hex';
|
||||
}
|
||||
|
||||
@@ -9,43 +9,272 @@ part of 'wallet_access.dart';
|
||||
// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||
// ignore_for_file: type=lint, type=warning
|
||||
|
||||
@ProviderFor(walletAccess)
|
||||
final walletAccessProvider = WalletAccessProvider._();
|
||||
@ProviderFor(clientWalletAccessRepository)
|
||||
final clientWalletAccessRepositoryProvider =
|
||||
ClientWalletAccessRepositoryProvider._();
|
||||
|
||||
final class WalletAccessProvider
|
||||
final class ClientWalletAccessRepositoryProvider
|
||||
extends
|
||||
$FunctionalProvider<
|
||||
AsyncValue<List<SdkClientWalletAccess>?>,
|
||||
List<SdkClientWalletAccess>?,
|
||||
FutureOr<List<SdkClientWalletAccess>?>
|
||||
ClientWalletAccessRepository,
|
||||
ClientWalletAccessRepository,
|
||||
ClientWalletAccessRepository
|
||||
>
|
||||
with
|
||||
$FutureModifier<List<SdkClientWalletAccess>?>,
|
||||
$FutureProvider<List<SdkClientWalletAccess>?> {
|
||||
WalletAccessProvider._()
|
||||
with $Provider<ClientWalletAccessRepository> {
|
||||
ClientWalletAccessRepositoryProvider._()
|
||||
: super(
|
||||
from: null,
|
||||
argument: null,
|
||||
retry: null,
|
||||
name: r'walletAccessProvider',
|
||||
name: r'clientWalletAccessRepositoryProvider',
|
||||
isAutoDispose: true,
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
);
|
||||
|
||||
@override
|
||||
String debugGetCreateSourceHash() => _$walletAccessHash();
|
||||
String debugGetCreateSourceHash() => _$clientWalletAccessRepositoryHash();
|
||||
|
||||
@$internal
|
||||
@override
|
||||
$FutureProviderElement<List<SdkClientWalletAccess>?> $createElement(
|
||||
$ProviderElement<ClientWalletAccessRepository> $createElement(
|
||||
$ProviderPointer pointer,
|
||||
) => $ProviderElement(pointer);
|
||||
|
||||
@override
|
||||
ClientWalletAccessRepository create(Ref ref) {
|
||||
return clientWalletAccessRepository(ref);
|
||||
}
|
||||
|
||||
/// {@macro riverpod.override_with_value}
|
||||
Override overrideWithValue(ClientWalletAccessRepository value) {
|
||||
return $ProviderOverride(
|
||||
origin: this,
|
||||
providerOverride: $SyncValueProvider<ClientWalletAccessRepository>(value),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
String _$clientWalletAccessRepositoryHash() =>
|
||||
r'bbc332284bc36a8b5d807bd5c45362b6b12b19e7';
|
||||
|
||||
@ProviderFor(clientWalletOptions)
|
||||
final clientWalletOptionsProvider = ClientWalletOptionsProvider._();
|
||||
|
||||
final class ClientWalletOptionsProvider
|
||||
extends
|
||||
$FunctionalProvider<
|
||||
AsyncValue<List<ClientWalletOption>>,
|
||||
List<ClientWalletOption>,
|
||||
FutureOr<List<ClientWalletOption>>
|
||||
>
|
||||
with
|
||||
$FutureModifier<List<ClientWalletOption>>,
|
||||
$FutureProvider<List<ClientWalletOption>> {
|
||||
ClientWalletOptionsProvider._()
|
||||
: super(
|
||||
from: null,
|
||||
argument: null,
|
||||
retry: null,
|
||||
name: r'clientWalletOptionsProvider',
|
||||
isAutoDispose: true,
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
);
|
||||
|
||||
@override
|
||||
String debugGetCreateSourceHash() => _$clientWalletOptionsHash();
|
||||
|
||||
@$internal
|
||||
@override
|
||||
$FutureProviderElement<List<ClientWalletOption>> $createElement(
|
||||
$ProviderPointer pointer,
|
||||
) => $FutureProviderElement(pointer);
|
||||
|
||||
@override
|
||||
FutureOr<List<SdkClientWalletAccess>?> create(Ref ref) {
|
||||
return walletAccess(ref);
|
||||
FutureOr<List<ClientWalletOption>> create(Ref ref) {
|
||||
return clientWalletOptions(ref);
|
||||
}
|
||||
}
|
||||
|
||||
String _$walletAccessHash() => r'954aae12d2d18999efaa97d01be983bf849f2296';
|
||||
String _$clientWalletOptionsHash() =>
|
||||
r'32183c2b281e2a41400de07f2381132a706815ab';
|
||||
|
||||
@ProviderFor(clientWalletAccessSelection)
|
||||
final clientWalletAccessSelectionProvider =
|
||||
ClientWalletAccessSelectionFamily._();
|
||||
|
||||
final class ClientWalletAccessSelectionProvider
|
||||
extends
|
||||
$FunctionalProvider<AsyncValue<Set<int>>, Set<int>, FutureOr<Set<int>>>
|
||||
with $FutureModifier<Set<int>>, $FutureProvider<Set<int>> {
|
||||
ClientWalletAccessSelectionProvider._({
|
||||
required ClientWalletAccessSelectionFamily super.from,
|
||||
required int super.argument,
|
||||
}) : super(
|
||||
retry: null,
|
||||
name: r'clientWalletAccessSelectionProvider',
|
||||
isAutoDispose: true,
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
);
|
||||
|
||||
@override
|
||||
String debugGetCreateSourceHash() => _$clientWalletAccessSelectionHash();
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return r'clientWalletAccessSelectionProvider'
|
||||
''
|
||||
'($argument)';
|
||||
}
|
||||
|
||||
@$internal
|
||||
@override
|
||||
$FutureProviderElement<Set<int>> $createElement($ProviderPointer pointer) =>
|
||||
$FutureProviderElement(pointer);
|
||||
|
||||
@override
|
||||
FutureOr<Set<int>> create(Ref ref) {
|
||||
final argument = this.argument as int;
|
||||
return clientWalletAccessSelection(ref, argument);
|
||||
}
|
||||
|
||||
@override
|
||||
bool operator ==(Object other) {
|
||||
return other is ClientWalletAccessSelectionProvider &&
|
||||
other.argument == argument;
|
||||
}
|
||||
|
||||
@override
|
||||
int get hashCode {
|
||||
return argument.hashCode;
|
||||
}
|
||||
}
|
||||
|
||||
String _$clientWalletAccessSelectionHash() =>
|
||||
r'f33705ee7201cd9b899cc058d6642de85a22b03e';
|
||||
|
||||
final class ClientWalletAccessSelectionFamily extends $Family
|
||||
with $FunctionalFamilyOverride<FutureOr<Set<int>>, int> {
|
||||
ClientWalletAccessSelectionFamily._()
|
||||
: super(
|
||||
retry: null,
|
||||
name: r'clientWalletAccessSelectionProvider',
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
isAutoDispose: true,
|
||||
);
|
||||
|
||||
ClientWalletAccessSelectionProvider call(int clientId) =>
|
||||
ClientWalletAccessSelectionProvider._(argument: clientId, from: this);
|
||||
|
||||
@override
|
||||
String toString() => r'clientWalletAccessSelectionProvider';
|
||||
}
|
||||
|
||||
@ProviderFor(ClientWalletAccessController)
|
||||
final clientWalletAccessControllerProvider =
|
||||
ClientWalletAccessControllerFamily._();
|
||||
|
||||
final class ClientWalletAccessControllerProvider
|
||||
extends
|
||||
$NotifierProvider<
|
||||
ClientWalletAccessController,
|
||||
ClientWalletAccessState
|
||||
> {
|
||||
ClientWalletAccessControllerProvider._({
|
||||
required ClientWalletAccessControllerFamily super.from,
|
||||
required int super.argument,
|
||||
}) : super(
|
||||
retry: null,
|
||||
name: r'clientWalletAccessControllerProvider',
|
||||
isAutoDispose: true,
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
);
|
||||
|
||||
@override
|
||||
String debugGetCreateSourceHash() => _$clientWalletAccessControllerHash();
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return r'clientWalletAccessControllerProvider'
|
||||
''
|
||||
'($argument)';
|
||||
}
|
||||
|
||||
@$internal
|
||||
@override
|
||||
ClientWalletAccessController create() => ClientWalletAccessController();
|
||||
|
||||
/// {@macro riverpod.override_with_value}
|
||||
Override overrideWithValue(ClientWalletAccessState value) {
|
||||
return $ProviderOverride(
|
||||
origin: this,
|
||||
providerOverride: $SyncValueProvider<ClientWalletAccessState>(value),
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
bool operator ==(Object other) {
|
||||
return other is ClientWalletAccessControllerProvider &&
|
||||
other.argument == argument;
|
||||
}
|
||||
|
||||
@override
|
||||
int get hashCode {
|
||||
return argument.hashCode;
|
||||
}
|
||||
}
|
||||
|
||||
String _$clientWalletAccessControllerHash() =>
|
||||
r'45bff81382fec3e8610190167b55667a7dfc1111';
|
||||
|
||||
final class ClientWalletAccessControllerFamily extends $Family
|
||||
with
|
||||
$ClassFamilyOverride<
|
||||
ClientWalletAccessController,
|
||||
ClientWalletAccessState,
|
||||
ClientWalletAccessState,
|
||||
ClientWalletAccessState,
|
||||
int
|
||||
> {
|
||||
ClientWalletAccessControllerFamily._()
|
||||
: super(
|
||||
retry: null,
|
||||
name: r'clientWalletAccessControllerProvider',
|
||||
dependencies: null,
|
||||
$allTransitiveDependencies: null,
|
||||
isAutoDispose: true,
|
||||
);
|
||||
|
||||
ClientWalletAccessControllerProvider call(int clientId) =>
|
||||
ClientWalletAccessControllerProvider._(argument: clientId, from: this);
|
||||
|
||||
@override
|
||||
String toString() => r'clientWalletAccessControllerProvider';
|
||||
}
|
||||
|
||||
abstract class _$ClientWalletAccessController
|
||||
extends $Notifier<ClientWalletAccessState> {
|
||||
late final _$args = ref.$arg as int;
|
||||
int get clientId => _$args;
|
||||
|
||||
ClientWalletAccessState build(int clientId);
|
||||
@$mustCallSuper
|
||||
@override
|
||||
void runBuild() {
|
||||
final ref =
|
||||
this.ref as $Ref<ClientWalletAccessState, ClientWalletAccessState>;
|
||||
final element =
|
||||
ref.element
|
||||
as $ClassProviderElement<
|
||||
AnyNotifier<ClientWalletAccessState, ClientWalletAccessState>,
|
||||
ClientWalletAccessState,
|
||||
Object?,
|
||||
Object?
|
||||
>;
|
||||
element.handleCreate(ref, () => build(_$args));
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user