mirror of
https://github.com/pese-git/cherrypick.git
synced 2026-01-23 21:13:35 +00:00
refactor(benchmarks): introduce DIAdapter abstraction, migrate all scenarios to use DIAdapter
This commit is contained in:
@@ -1,14 +1,19 @@
|
||||
import 'package:benchmark_cherrypick/cherrypick_benchmark.dart';
|
||||
import 'package:benchmark_cherrypick/complex_bindings_benchmark.dart';
|
||||
import 'package:benchmark_cherrypick/async_chain_benchmark.dart';
|
||||
import 'package:benchmark_cherrypick/di_adapter.dart';
|
||||
import 'package:benchmark_cherrypick/scope_override_benchmark.dart';
|
||||
import 'package:args/args.dart';
|
||||
import 'dart:io';
|
||||
import 'dart:math';
|
||||
|
||||
Future<void> main(List<String> args) async {
|
||||
final parser = ArgParser()
|
||||
..addOption('benchmark', abbr: 'b', help: 'Benchmark name (register, chain_singleton, chain_factory, named, override, async_chain, all)', defaultsTo: 'all')
|
||||
..addOption('chainCount', abbr: 'c', help: 'Comma-separated chainCounts (используется в chain_singleton/factory)', defaultsTo: '100')
|
||||
..addOption('nestingDepth', abbr: 'd', help: 'Comma-separated depths (используется в chain_singleton/factory)', defaultsTo: '100')
|
||||
..addOption('repeat', abbr: 'r', help: 'Repeats for each run (statistical run, >=2)', defaultsTo: '5')
|
||||
..addOption('warmup', abbr: 'w', help: 'Warmup runs before timing', defaultsTo: '2')
|
||||
..addOption('format', abbr: 'f', help: 'Output format (pretty, csv, json)', defaultsTo: 'pretty')
|
||||
..addFlag('help', abbr: 'h', negatable: false, help: 'Show help');
|
||||
|
||||
@@ -17,66 +22,116 @@ Future<void> main(List<String> args) async {
|
||||
if (result['help'] == true) {
|
||||
print('Dart DI benchmarks');
|
||||
print(parser.usage);
|
||||
print('\nExamples:\n'
|
||||
' dart run bin/main.dart --benchmark=chain_singleton --chainCount=10,100 --nestingDepth=5,10 --format=csv\n'
|
||||
' dart run bin/main.dart --benchmark=named\n'
|
||||
'Extra: --repeat=7 --warmup=3\n');
|
||||
return;
|
||||
}
|
||||
|
||||
final di = CherrypickDIAdapter();
|
||||
|
||||
final benchmark = result['benchmark'] as String;
|
||||
final format = result['format'] as String;
|
||||
|
||||
final chainCounts = _parseIntList(result['chainCount'] as String);
|
||||
final nestDepths = _parseIntList(result['nestingDepth'] as String);
|
||||
final repeats = int.tryParse(result['repeat'] as String? ?? "") ?? 5;
|
||||
final warmups = int.tryParse(result['warmup'] as String? ?? "") ?? 2;
|
||||
|
||||
final results = <Map<String, dynamic>>[];
|
||||
void addResult(String name, int? chainCount, int? nestingDepth, num elapsed) {
|
||||
void addResult(
|
||||
String name,
|
||||
int? chainCount,
|
||||
int? nestingDepth,
|
||||
List<num> timings,
|
||||
int? memoryDiffKb,
|
||||
int? deltaPeakKb,
|
||||
int? peakRssKb,
|
||||
) {
|
||||
timings.sort();
|
||||
|
||||
var mean = timings.reduce((a, b) => a + b) / timings.length;
|
||||
var median = timings[timings.length ~/ 2];
|
||||
var minVal = timings.first;
|
||||
var maxVal = timings.last;
|
||||
var stddev = sqrt(timings.map((x) => pow(x - mean, 2)).reduce((a, b) => a + b) / timings.length);
|
||||
results.add({
|
||||
'benchmark': name,
|
||||
'chainCount': chainCount,
|
||||
'nestingDepth': nestingDepth,
|
||||
'elapsed_us': elapsed.round()
|
||||
'mean_us': mean.round(),
|
||||
'median_us': median.round(),
|
||||
'stddev_us': stddev.round(),
|
||||
'min_us': minVal.round(),
|
||||
'max_us': maxVal.round(),
|
||||
'trials': timings.length,
|
||||
'timings_us': timings.map((t) => t.round()).toList(),
|
||||
'memory_diff_kb': memoryDiffKb,
|
||||
'delta_peak_kb': deltaPeakKb,
|
||||
'peak_rss_kb': peakRssKb,
|
||||
});
|
||||
}
|
||||
|
||||
Future<void> runAndCollect(String name, Future<num> Function() fn, {int? chainCount, int? nestingDepth}) async {
|
||||
final elapsed = await fn();
|
||||
addResult(name, chainCount, nestingDepth, elapsed);
|
||||
Future<void> runAndCollect(
|
||||
String name,
|
||||
Future<num> Function() fn, {
|
||||
int? chainCount,
|
||||
int? nestingDepth,
|
||||
}) async {
|
||||
for (int i = 0; i < warmups; i++) {
|
||||
await fn();
|
||||
}
|
||||
final timings = <num>[];
|
||||
final rssValues = <int>[];
|
||||
final memBefore = ProcessInfo.currentRss;
|
||||
for (int i = 0; i < repeats; i++) {
|
||||
timings.add(await fn());
|
||||
rssValues.add(ProcessInfo.currentRss);
|
||||
}
|
||||
final memAfter = ProcessInfo.currentRss;
|
||||
final memDiffKB = ((memAfter - memBefore) / 1024).round();
|
||||
final peakRss = [...rssValues, memBefore].reduce(max);
|
||||
final deltaPeakKb = ((peakRss - memBefore) / 1024).round();
|
||||
addResult(name, chainCount, nestingDepth, timings, memDiffKB, deltaPeakKb, (peakRss/1024).round());
|
||||
}
|
||||
|
||||
if (benchmark == 'all' || benchmark == 'register') {
|
||||
await runAndCollect('RegisterAndResolve', () async {
|
||||
return _captureReport(RegisterAndResolveBenchmark().report);
|
||||
return _captureReport(RegisterAndResolveBenchmark(di).report);
|
||||
});
|
||||
}
|
||||
if (benchmark == 'all' || benchmark == 'chain_singleton') {
|
||||
for (final c in chainCounts) {
|
||||
for (final c in chainCounts) {
|
||||
for (final d in nestDepths) {
|
||||
await runAndCollect('ChainSingleton', () async {
|
||||
return _captureReport(() => ChainSingletonBenchmark(chainCount: c, nestingDepth: d).report());
|
||||
return _captureReport(() => ChainSingletonBenchmark(di,chainCount: c, nestingDepth: d).report());
|
||||
}, chainCount: c, nestingDepth: d);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (benchmark == 'all' || benchmark == 'chain_factory') {
|
||||
for (final c in chainCounts) {
|
||||
for (final c in chainCounts) {
|
||||
for (final d in nestDepths) {
|
||||
await runAndCollect('ChainFactory', () async {
|
||||
return _captureReport(() => ChainFactoryBenchmark(chainCount: c, nestingDepth: d).report());
|
||||
return _captureReport(() => ChainFactoryBenchmark(di, chainCount: c, nestingDepth: d).report());
|
||||
}, chainCount: c, nestingDepth: d);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (benchmark == 'all' || benchmark == 'named') {
|
||||
await runAndCollect('NamedResolve', () async {
|
||||
return _captureReport(NamedResolveBenchmark().report);
|
||||
return _captureReport(NamedResolveBenchmark(di).report);
|
||||
});
|
||||
}
|
||||
if (benchmark == 'all' || benchmark == 'override') {
|
||||
await runAndCollect('ScopeOverride', () async {
|
||||
return _captureReport(ScopeOverrideBenchmark().report);
|
||||
return _captureReport(ScopeOverrideBenchmark(di).report);
|
||||
});
|
||||
}
|
||||
if (benchmark == 'all' || benchmark == 'async_chain') {
|
||||
await runAndCollect('AsyncChain', () async {
|
||||
return _captureReportAsync(AsyncChainBenchmark().report);
|
||||
return _captureReportAsync(AsyncChainBenchmark(di).report);
|
||||
});
|
||||
}
|
||||
|
||||
@@ -106,16 +161,28 @@ Future<num> _captureReportAsync(Future<void> Function() fn) async {
|
||||
}
|
||||
|
||||
String _toPretty(List<Map<String, dynamic>> rows) {
|
||||
final keys = ['benchmark','chainCount','nestingDepth','elapsed_us'];
|
||||
final keys = [
|
||||
'benchmark','chainCount','nestingDepth','mean_us','median_us','stddev_us',
|
||||
'min_us','max_us','trials','memory_diff_kb','delta_peak_kb','peak_rss_kb'
|
||||
];
|
||||
final header = keys.join('\t');
|
||||
final lines = rows.map((r) => keys.map((k) => (r[k] ?? '').toString()).join('\t')).toList();
|
||||
return ([header] + lines).join('\n');
|
||||
}
|
||||
|
||||
String _toCsv(List<Map<String, dynamic>> rows) {
|
||||
final keys = ['benchmark','chainCount','nestingDepth','elapsed_us'];
|
||||
final keys = [
|
||||
'benchmark','chainCount','nestingDepth','mean_us','median_us','stddev_us',
|
||||
'min_us','max_us','trials','timings_us','memory_diff_kb','delta_peak_kb','peak_rss_kb'
|
||||
];
|
||||
final header = keys.join(',');
|
||||
final lines = rows.map((r) => keys.map((k) => (r[k] ?? '').toString()).join(',')).toList();
|
||||
final lines = rows.map((r) =>
|
||||
keys.map((k) {
|
||||
final v = r[k];
|
||||
if (v is List) return '"${v.join(';')}"';
|
||||
return (v ?? '').toString();
|
||||
}).join(',')
|
||||
).toList();
|
||||
return ([header] + lines).join('\n');
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user