mirror of
https://github.com/pese-git/cherrypick.git
synced 2026-01-23 21:13:35 +00:00
Applied consistent code formatting across all packages using \$ melos format
└> dart format .
└> RUNNING (in 8 packages)
--------------------------------------------------------------------------------
benchmark_di:
Formatted 18 files (0 changed) in 0.30 seconds.
benchmark_di: SUCCESS
--------------------------------------------------------------------------------
cherrypick:
Formatted 24 files (0 changed) in 0.34 seconds.
cherrypick: SUCCESS
--------------------------------------------------------------------------------
cherrypick_annotations:
Formatted 11 files (0 changed) in 0.14 seconds.
cherrypick_annotations: SUCCESS
--------------------------------------------------------------------------------
cherrypick_flutter:
Formatted 3 files (0 changed) in 0.15 seconds.
cherrypick_flutter: SUCCESS
--------------------------------------------------------------------------------
cherrypick_generator:
Formatted 17 files (0 changed) in 0.27 seconds.
cherrypick_generator: SUCCESS
--------------------------------------------------------------------------------
client_app:
Formatted 4 files (0 changed) in 0.14 seconds.
client_app: SUCCESS
--------------------------------------------------------------------------------
postly:
Formatted lib/router/app_router.gr.dart
Formatted 23 files (1 changed) in 0.33 seconds.
postly: SUCCESS
--------------------------------------------------------------------------------
talker_cherrypick_logger:
Formatted 4 files (0 changed) in 0.18 seconds.
talker_cherrypick_logger: SUCCESS
--------------------------------------------------------------------------------
$ melos format
└> dart format .
└> SUCCESS. No functional or logic changes included.
157 lines
5.7 KiB
Dart
157 lines
5.7 KiB
Dart
import 'package:benchmark_di/scenarios/universal_binding_mode.dart';
|
||
import 'package:benchmark_di/scenarios/universal_scenario.dart';
|
||
import 'package:benchmark_di/scenarios/universal_service.dart';
|
||
import 'package:riverpod/riverpod.dart' as rp;
|
||
import 'di_adapter.dart';
|
||
|
||
/// Унифицированный DIAdapter для Riverpod с поддержкой scopes и строгой типизацией.
|
||
class RiverpodAdapter extends DIAdapter<Map<String, rp.ProviderBase<Object?>>> {
|
||
rp.ProviderContainer? _container;
|
||
final Map<String, rp.ProviderBase<Object?>> _namedProviders;
|
||
final rp.ProviderContainer? _parent;
|
||
|
||
RiverpodAdapter({
|
||
rp.ProviderContainer? container,
|
||
Map<String, rp.ProviderBase<Object?>>? providers,
|
||
rp.ProviderContainer? parent,
|
||
bool isSubScope = false,
|
||
}) : _container = container,
|
||
_namedProviders = providers ?? <String, rp.ProviderBase<Object?>>{},
|
||
_parent = parent;
|
||
|
||
@override
|
||
void setupDependencies(
|
||
void Function(Map<String, rp.ProviderBase<Object?>> container)
|
||
registration) {
|
||
_container ??= _parent == null
|
||
? rp.ProviderContainer()
|
||
: rp.ProviderContainer(parent: _parent);
|
||
registration(_namedProviders);
|
||
}
|
||
|
||
@override
|
||
T resolve<T extends Object>({String? named}) {
|
||
final key = named ?? T.toString();
|
||
final provider = _namedProviders[key];
|
||
if (provider == null) {
|
||
throw Exception('Provider not found for $key');
|
||
}
|
||
return _container!.read(provider) as T;
|
||
}
|
||
|
||
@override
|
||
Future<T> resolveAsync<T extends Object>({String? named}) async {
|
||
final key = named ?? T.toString();
|
||
final provider = _namedProviders[key];
|
||
if (provider == null) {
|
||
throw Exception('Provider not found for $key');
|
||
}
|
||
// Если это FutureProvider — используем .future
|
||
if (provider.runtimeType.toString().contains('FutureProvider')) {
|
||
return await _container!.read((provider as dynamic).future) as T;
|
||
}
|
||
return resolve<T>(named: named);
|
||
}
|
||
|
||
@override
|
||
void teardown() {
|
||
_container?.dispose();
|
||
_container = null;
|
||
_namedProviders.clear();
|
||
}
|
||
|
||
@override
|
||
RiverpodAdapter openSubScope(String name) {
|
||
final newContainer = rp.ProviderContainer(parent: _container);
|
||
return RiverpodAdapter(
|
||
container: newContainer,
|
||
providers: Map.of(_namedProviders),
|
||
parent: _container,
|
||
isSubScope: true,
|
||
);
|
||
}
|
||
|
||
@override
|
||
Future<void> waitForAsyncReady() async {
|
||
// Riverpod синхронный по умолчанию.
|
||
return;
|
||
}
|
||
|
||
@override
|
||
Registration<Map<String, rp.ProviderBase<Object?>>>
|
||
universalRegistration<S extends Enum>({
|
||
required S scenario,
|
||
required int chainCount,
|
||
required int nestingDepth,
|
||
required UniversalBindingMode bindingMode,
|
||
}) {
|
||
if (scenario is UniversalScenario) {
|
||
return (providers) {
|
||
switch (scenario) {
|
||
case UniversalScenario.register:
|
||
providers['UniversalService'] = rp.Provider<UniversalService>(
|
||
(ref) => UniversalServiceImpl(value: 'reg', dependency: null));
|
||
break;
|
||
case UniversalScenario.named:
|
||
providers['impl1'] = rp.Provider<UniversalService>(
|
||
(ref) => UniversalServiceImpl(value: 'impl1'));
|
||
providers['impl2'] = rp.Provider<UniversalService>(
|
||
(ref) => UniversalServiceImpl(value: 'impl2'));
|
||
break;
|
||
case UniversalScenario.chain:
|
||
for (int chain = 1; chain <= chainCount; chain++) {
|
||
for (int level = 1; level <= nestingDepth; level++) {
|
||
final prevDepName = '${chain}_${level - 1}';
|
||
final depName = '${chain}_$level';
|
||
providers[depName] =
|
||
rp.Provider<UniversalService>((ref) => UniversalServiceImpl(
|
||
value: depName,
|
||
dependency: level > 1
|
||
? ref.watch(providers[prevDepName]
|
||
as rp.ProviderBase<UniversalService>)
|
||
: null,
|
||
));
|
||
}
|
||
}
|
||
final depName = '${chainCount}_$nestingDepth';
|
||
providers['UniversalService'] = rp.Provider<UniversalService>(
|
||
(ref) => ref.watch(
|
||
providers[depName] as rp.ProviderBase<UniversalService>));
|
||
break;
|
||
case UniversalScenario.override:
|
||
// handled at benchmark level
|
||
break;
|
||
case UniversalScenario.asyncChain:
|
||
for (int chain = 1; chain <= chainCount; chain++) {
|
||
for (int level = 1; level <= nestingDepth; level++) {
|
||
final prevDepName = '${chain}_${level - 1}';
|
||
final depName = '${chain}_$level';
|
||
providers[depName] =
|
||
rp.FutureProvider<UniversalService>((ref) async {
|
||
return UniversalServiceImpl(
|
||
value: depName,
|
||
dependency: level > 1
|
||
? await ref.watch((providers[prevDepName]
|
||
as rp.FutureProvider<UniversalService>)
|
||
.future) as UniversalService?
|
||
: null,
|
||
);
|
||
});
|
||
}
|
||
}
|
||
final depName = '${chainCount}_$nestingDepth';
|
||
providers['UniversalService'] =
|
||
rp.FutureProvider<UniversalService>((ref) async {
|
||
return await ref.watch(
|
||
(providers[depName] as rp.FutureProvider<UniversalService>)
|
||
.future);
|
||
});
|
||
break;
|
||
}
|
||
};
|
||
}
|
||
throw UnsupportedError(
|
||
'Scenario $scenario not supported by RiverpodAdapter');
|
||
}
|
||
}
|