2024-01-01 14:58:21 -05:00
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
2023-07-28 17:27:10 +02:00
use serde ::Serialize ;
2024-08-20 10:27:36 +09:00
use crate ::tools ::test ::TestFailureFormatOptions ;
2024-08-15 22:47:16 +01:00
use crate ::version ;
2023-07-28 17:27:10 +02:00
use super ::* ;
pub trait BenchReporter {
fn report_group_summary ( & mut self ) ;
fn report_plan ( & mut self , plan : & BenchPlan ) ;
fn report_end ( & mut self , report : & BenchReport ) ;
fn report_register ( & mut self , desc : & BenchDescription ) ;
fn report_wait ( & mut self , desc : & BenchDescription ) ;
fn report_output ( & mut self , output : & str ) ;
fn report_result ( & mut self , desc : & BenchDescription , result : & BenchResult ) ;
2023-12-05 09:26:06 -07:00
fn report_uncaught_error ( & mut self , origin : & str , error : Box < JsError > ) ;
2023-07-28 17:27:10 +02:00
}
2024-09-05 10:51:40 +02:00
const JSON_SCHEMA_VERSION : u8 = 1 ;
2023-07-28 17:27:10 +02:00
#[ derive(Debug, Serialize) ]
struct JsonReporterOutput {
2024-09-05 10:51:40 +02:00
version : u8 ,
2023-07-28 17:27:10 +02:00
runtime : String ,
cpu : String ,
benches : Vec < JsonReporterBench > ,
}
impl Default for JsonReporterOutput {
fn default ( ) -> Self {
Self {
2024-09-05 10:51:40 +02:00
version : JSON_SCHEMA_VERSION ,
2024-08-15 22:47:16 +01:00
runtime : format ! (
" {} {} " ,
version ::DENO_VERSION_INFO . user_agent ,
env! ( " TARGET " )
) ,
2023-07-28 17:27:10 +02:00
cpu : mitata ::cpu ::name ( ) ,
benches : vec ! [ ] ,
}
}
}
#[ derive(Debug, Serialize) ]
struct JsonReporterBench {
origin : String ,
group : Option < String > ,
name : String ,
baseline : bool ,
results : Vec < BenchResult > ,
}
#[ derive(Debug, Serialize) ]
pub struct JsonReporter ( JsonReporterOutput ) ;
impl JsonReporter {
pub fn new ( ) -> Self {
Self ( Default ::default ( ) )
}
}
2024-05-08 22:45:06 -04:00
#[ allow(clippy::print_stdout) ]
2023-07-28 17:27:10 +02:00
impl BenchReporter for JsonReporter {
fn report_group_summary ( & mut self ) { }
#[ cold ]
fn report_plan ( & mut self , _plan : & BenchPlan ) { }
fn report_end ( & mut self , _report : & BenchReport ) {
match write_json_to_stdout ( self ) {
Ok ( _ ) = > ( ) ,
2024-05-08 22:45:06 -04:00
Err ( e ) = > println! ( " {} " , e ) ,
2023-07-28 17:27:10 +02:00
}
}
fn report_register ( & mut self , _desc : & BenchDescription ) { }
fn report_wait ( & mut self , _desc : & BenchDescription ) { }
fn report_output ( & mut self , _output : & str ) { }
fn report_result ( & mut self , desc : & BenchDescription , result : & BenchResult ) {
if desc . warmup {
return ;
}
let maybe_bench = self . 0. benches . iter_mut ( ) . find ( | bench | {
bench . origin = = desc . origin
& & bench . group = = desc . group
& & bench . name = = desc . name
& & bench . baseline = = desc . baseline
} ) ;
if let Some ( bench ) = maybe_bench {
bench . results . push ( result . clone ( ) ) ;
} else {
self . 0. benches . push ( JsonReporterBench {
origin : desc . origin . clone ( ) ,
group : desc . group . clone ( ) ,
name : desc . name . clone ( ) ,
baseline : desc . baseline ,
results : vec ! [ result . clone ( ) ] ,
} ) ;
}
}
2023-12-05 09:26:06 -07:00
fn report_uncaught_error ( & mut self , _origin : & str , _error : Box < JsError > ) { }
2023-07-28 17:27:10 +02:00
}
pub struct ConsoleReporter {
name : String ,
show_output : bool ,
group : Option < String > ,
baseline : bool ,
group_measurements : Vec < ( BenchDescription , BenchStats ) > ,
options : Option < mitata ::reporter ::Options > ,
}
impl ConsoleReporter {
pub fn new ( show_output : bool ) -> Self {
Self {
show_output ,
group : None ,
options : None ,
baseline : false ,
name : String ::new ( ) ,
group_measurements : Vec ::new ( ) ,
}
}
}
2024-05-08 22:45:06 -04:00
#[ allow(clippy::print_stdout) ]
2023-07-28 17:27:10 +02:00
impl BenchReporter for ConsoleReporter {
#[ cold ]
fn report_plan ( & mut self , plan : & BenchPlan ) {
use std ::sync ::atomic ::AtomicBool ;
use std ::sync ::atomic ::Ordering ;
static FIRST_PLAN : AtomicBool = AtomicBool ::new ( true ) ;
self . report_group_summary ( ) ;
self . group = None ;
self . baseline = false ;
self . name = String ::new ( ) ;
self . group_measurements . clear ( ) ;
self . options = Some ( mitata ::reporter ::Options ::new (
& plan . names . iter ( ) . map ( | x | x . as_str ( ) ) . collect ::< Vec < & str > > ( ) ,
) ) ;
let options = self . options . as_mut ( ) . unwrap ( ) ;
options . percentiles = true ;
if FIRST_PLAN
. compare_exchange ( true , false , Ordering ::SeqCst , Ordering ::SeqCst )
. is_ok ( )
{
2024-08-26 10:56:04 +01:00
println! (
" {} " ,
colors ::gray ( format! ( " CPU | {} " , mitata ::cpu ::name ( ) ) )
) ;
2023-07-28 17:27:10 +02:00
println! (
" {} \n " ,
colors ::gray ( format! (
2024-08-26 10:56:04 +01:00
" Runtime | Deno {} ({}) " ,
2024-08-15 22:47:16 +01:00
crate ::version ::DENO_VERSION_INFO . deno ,
2023-07-28 17:27:10 +02:00
env! ( " TARGET " )
) )
) ;
} else {
println! ( ) ;
}
println! (
2024-08-26 10:56:04 +01:00
" {} \n \n {} \n {} " ,
2023-07-28 17:27:10 +02:00
colors ::gray ( & plan . origin ) ,
mitata ::reporter ::header ( options ) ,
mitata ::reporter ::br ( options )
) ;
}
fn report_register ( & mut self , _desc : & BenchDescription ) { }
fn report_wait ( & mut self , desc : & BenchDescription ) {
2024-06-14 17:10:57 +05:30
self . name . clone_from ( & desc . name ) ;
2023-07-28 17:27:10 +02:00
match & desc . group {
fix(bench): Fix group header printing logic + don't filter out the warmup benchmark (#23083)
Fixes #23053.
Two small bugs here:
- the existing condition for printing out the group header was broken.
it worked in the reproducer (in the issue above) without filtering only
by accident, due to setting `self.has_ungrouped = true` once we see the
warmup bench. Knowing that we sort benchmarks to put ungrouped benches
first, there are only two cases: 1) we are starting the first group 2)
we are ending the previous group and starting a new group
- when you passed `--filter` we were applying that filter to the warmup
bench (which is not visible to users), so we suffered from jit bias if
you were filtering (unless your filter was `<warmup>`)
TLDR;
Running
```bash
deno bench main.js --filter="G"
```
```js
// main.js
Deno.bench({
group: "G1",
name: "G1-A",
fn() {},
});
Deno.bench({
group: "G1",
name: "G1-B",
fn() {},
});
```
Before this PR:
```
benchmark time (avg) iter/s (min … max) p75 p99 p995
--------------------------------------------------------------- -----------------------------
G1-A 303.52 ps/iter3,294,726,102.1 (254.2 ps … 7.8 ns) 287.5 ps 391.7 ps 437.5 ps
G1-B 3.8 ns/iter 263,360,635.9 (2.24 ns … 8.36 ns) 3.84 ns 4.73 ns 4.94 ns
summary
G1-A
12.51x faster than G1-B
```
After this PR:
```
benchmark time (avg) iter/s (min … max) p75 p99 p995
--------------------------------------------------------------- -----------------------------
group G1
G1-A 3.85 ns/iter 259,822,096.0 (2.42 ns … 9.03 ns) 3.83 ns 4.62 ns 4.83 ns
G1-B 3.84 ns/iter 260,458,274.5 (3.55 ns … 7.05 ns) 3.83 ns 4.45 ns 4.7 ns
summary
G1-B
1x faster than G1-A
```
2024-03-26 09:19:24 -07:00
None = > { }
2023-07-28 17:27:10 +02:00
Some ( group ) = > {
if self . group . is_none ( ) | | group ! = self . group . as_ref ( ) . unwrap ( ) {
self . report_group_summary ( ) ;
2023-12-18 04:27:14 +05:30
println! ( " {} {} " , colors ::gray ( " group " ) , colors ::green ( group ) ) ;
2023-07-28 17:27:10 +02:00
}
self . group = Some ( group . clone ( ) ) ;
}
}
}
fn report_output ( & mut self , output : & str ) {
if self . show_output {
print! ( " {} {} " , colors ::gray ( format! ( " {} : " , self . name ) ) , output )
}
}
fn report_result ( & mut self , desc : & BenchDescription , result : & BenchResult ) {
if desc . warmup {
return ;
}
let options = self . options . as_ref ( ) . unwrap ( ) ;
match result {
BenchResult ::Ok ( stats ) = > {
let mut desc = desc . clone ( ) ;
if desc . baseline & & ! self . baseline {
self . baseline = true ;
} else {
desc . baseline = false ;
}
println! (
" {} " ,
mitata ::reporter ::benchmark (
& desc . name ,
& mitata ::reporter ::BenchmarkStats {
avg : stats . avg ,
min : stats . min ,
max : stats . max ,
p75 : stats . p75 ,
p99 : stats . p99 ,
p995 : stats . p995 ,
} ,
options
)
) ;
fix(bench): explicit timers don't force high precision measurements (#20272)
Disables `BenchContext::start()` and `BenchContext::end()` for low
precision benchmarks (less than 0.01s per iteration). Prints a warning
when they are used in such benchmarks, suggesting to remove them.
```ts
Deno.bench("noop", { group: "noops" }, () => {});
Deno.bench("noop with start/end", { group: "noops" }, (b) => {
b.start();
b.end();
});
```
Before:
```
cpu: 12th Gen Intel(R) Core(TM) i9-12900K
runtime: deno 1.36.2 (x86_64-unknown-linux-gnu)
file:///home/nayeem/projects/deno/temp3.ts
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------- -----------------------------
noop 2.63 ns/iter 380,674,131.4 (2.45 ns … 27.78 ns) 2.55 ns 4.03 ns 5.33 ns
noop with start and end 302.47 ns/iter 3,306,146.0 (200 ns … 151.2 µs) 300 ns 400 ns 400 ns
summary
noop
115.14x faster than noop with start and end
```
After:
```
cpu: 12th Gen Intel(R) Core(TM) i9-12900K
runtime: deno 1.36.1 (x86_64-unknown-linux-gnu)
file:///home/nayeem/projects/deno/temp3.ts
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------- -----------------------------
noop 3.01 ns/iter 332,565,561.7 (2.73 ns … 29.54 ns) 2.93 ns 5.29 ns 7.45 ns
noop with start and end 7.73 ns/iter 129,291,091.5 (6.61 ns … 46.76 ns) 7.87 ns 13.12 ns 15.32 ns
Warning start() and end() calls in "noop with start and end" are ignored because it averages less than 0.01s per iteration. Remove them for better results.
summary
noop
2.57x faster than noop with start and end
```
2023-08-26 10:29:45 +01:00
if ! stats . high_precision & & stats . used_explicit_timers {
2023-10-10 14:40:36 +01:00
println! ( " {} " , colors ::yellow ( format! ( " Warning: start() and end() calls in \" {} \" are ignored because it averages less \n than 10µs per iteration. Remove them for better results. " , & desc . name ) ) ) ;
fix(bench): explicit timers don't force high precision measurements (#20272)
Disables `BenchContext::start()` and `BenchContext::end()` for low
precision benchmarks (less than 0.01s per iteration). Prints a warning
when they are used in such benchmarks, suggesting to remove them.
```ts
Deno.bench("noop", { group: "noops" }, () => {});
Deno.bench("noop with start/end", { group: "noops" }, (b) => {
b.start();
b.end();
});
```
Before:
```
cpu: 12th Gen Intel(R) Core(TM) i9-12900K
runtime: deno 1.36.2 (x86_64-unknown-linux-gnu)
file:///home/nayeem/projects/deno/temp3.ts
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------- -----------------------------
noop 2.63 ns/iter 380,674,131.4 (2.45 ns … 27.78 ns) 2.55 ns 4.03 ns 5.33 ns
noop with start and end 302.47 ns/iter 3,306,146.0 (200 ns … 151.2 µs) 300 ns 400 ns 400 ns
summary
noop
115.14x faster than noop with start and end
```
After:
```
cpu: 12th Gen Intel(R) Core(TM) i9-12900K
runtime: deno 1.36.1 (x86_64-unknown-linux-gnu)
file:///home/nayeem/projects/deno/temp3.ts
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------- -----------------------------
noop 3.01 ns/iter 332,565,561.7 (2.73 ns … 29.54 ns) 2.93 ns 5.29 ns 7.45 ns
noop with start and end 7.73 ns/iter 129,291,091.5 (6.61 ns … 46.76 ns) 7.87 ns 13.12 ns 15.32 ns
Warning start() and end() calls in "noop with start and end" are ignored because it averages less than 0.01s per iteration. Remove them for better results.
summary
noop
2.57x faster than noop with start and end
```
2023-08-26 10:29:45 +01:00
}
2023-07-28 17:27:10 +02:00
self . group_measurements . push ( ( desc , stats . clone ( ) ) ) ;
}
BenchResult ::Failed ( js_error ) = > {
println! (
" {} " ,
mitata ::reporter ::benchmark_error (
& desc . name ,
& mitata ::reporter ::Error {
stack : None ,
2024-08-20 10:27:36 +09:00
message : format_test_error (
js_error ,
& TestFailureFormatOptions ::default ( )
) ,
2023-07-28 17:27:10 +02:00
} ,
options
)
)
}
} ;
}
fn report_group_summary ( & mut self ) {
2024-04-29 09:27:55 -04:00
if self . options . is_none ( ) {
return ;
}
2023-07-28 17:27:10 +02:00
if 2 < = self . group_measurements . len ( )
& & ( self . group . is_some ( ) | | ( self . group . is_none ( ) & & self . baseline ) )
{
println! (
" \n {} " ,
mitata ::reporter ::summary (
& self
. group_measurements
. iter ( )
. map ( | ( d , s ) | mitata ::reporter ::GroupBenchmark {
name : d . name . clone ( ) ,
baseline : d . baseline ,
group : d . group . as_deref ( ) . unwrap_or ( " " ) . to_owned ( ) ,
stats : mitata ::reporter ::BenchmarkStats {
avg : s . avg ,
min : s . min ,
max : s . max ,
p75 : s . p75 ,
p99 : s . p99 ,
p995 : s . p995 ,
} ,
} )
. collect ::< Vec < mitata ::reporter ::GroupBenchmark > > ( ) ,
)
) ;
}
2023-12-18 04:27:14 +05:30
println! ( ) ;
2023-07-28 17:27:10 +02:00
self . baseline = false ;
self . group_measurements . clear ( ) ;
}
fn report_end ( & mut self , _ : & BenchReport ) {
self . report_group_summary ( ) ;
}
2023-12-05 09:26:06 -07:00
fn report_uncaught_error ( & mut self , _origin : & str , error : Box < JsError > ) {
println! (
" {}: {} " ,
colors ::red_bold ( " error " ) ,
2024-08-20 10:27:36 +09:00
format_test_error ( & error , & TestFailureFormatOptions ::default ( ) )
2023-12-05 09:26:06 -07:00
) ;
println! ( " This error was not caught from a benchmark and caused the bench runner to fail on the referenced module. " ) ;
println! ( " It most likely originated from a dangling promise, event/timeout handler or top-level code. " ) ;
println! ( ) ;
}
2023-07-28 17:27:10 +02:00
}