2022-09-07 01:55:32 +03:00
|
|
|
module main
|
|
|
|
|
|
|
|
import vweb
|
|
|
|
import os
|
|
|
|
import json
|
|
|
|
import arrays
|
|
|
|
import net.http
|
|
|
|
import math
|
|
|
|
import v.util.version
|
|
|
|
|
|
|
|
[table: 'benchmark']
|
|
|
|
struct Task {
|
|
|
|
mut:
|
|
|
|
id u32 [primary; serial; sql: serial]
|
|
|
|
title string
|
|
|
|
status string
|
|
|
|
}
|
|
|
|
|
|
|
|
struct FrameworkBenchmarkResponse {
|
|
|
|
insert []int
|
|
|
|
@select []int
|
|
|
|
update []int
|
|
|
|
}
|
|
|
|
|
|
|
|
struct FrameworkPlatform {
|
|
|
|
mut:
|
|
|
|
v_sqlite_memory []int
|
|
|
|
// v_sqlite_file []int
|
|
|
|
typescript_sqlite_memory []int
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (framework_platform FrameworkPlatform) to_map() map[string][]int {
|
|
|
|
mut mapa := map[string][]int{}
|
|
|
|
|
|
|
|
mapa['v_sqlite_memory'] = framework_platform.v_sqlite_memory
|
|
|
|
// mapa['v_sqlite_file'] = framework_platform.v_sqlite_file
|
|
|
|
mapa['typescript_sqlite_memory'] = framework_platform.typescript_sqlite_memory
|
|
|
|
return mapa
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
http_port = 3001
|
|
|
|
benchmark_loop_length = 20
|
|
|
|
)
|
|
|
|
|
|
|
|
struct App {
|
|
|
|
vweb.Context
|
|
|
|
}
|
|
|
|
|
|
|
|
enum SqliteDbConnection {
|
|
|
|
sqlite_memory
|
|
|
|
sqlite_file
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
vweb.run(new_app(), http_port)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn (mut app App) before_request() {
|
|
|
|
os.execute_or_panic('v -b js_browser draw.js.v ')
|
|
|
|
}
|
|
|
|
|
|
|
|
fn new_app() &App {
|
|
|
|
mut app := &App{}
|
|
|
|
app.serve_static('/favicon.ico', 'favicon.ico')
|
|
|
|
app.serve_static('/draw.js', 'draw.js')
|
|
|
|
app.mount_static_folder_at(os.resource_abs_path('.'), '/')
|
|
|
|
return app
|
|
|
|
}
|
|
|
|
|
|
|
|
['/'; get]
|
|
|
|
pub fn (mut app App) controller_get_all_task() ?vweb.Result {
|
|
|
|
v_version := version.full_v_version(true)
|
|
|
|
orm_stmt_kinds := ['insert', 'select', 'update']
|
|
|
|
|
|
|
|
mut attribute_names := map[string][]string{}
|
|
|
|
// Used to garante the chart proposionalite
|
|
|
|
mut max_benchmark := map[string]int{}
|
|
|
|
mut from_framework := map[string]string{}
|
|
|
|
mut maxs := map[string][]int{}
|
|
|
|
mut framework_platform := map[string]map[string][]int{}
|
|
|
|
mut table := map[string]map[string]map[string]string{}
|
|
|
|
|
|
|
|
chart_colors := ['gray', 'red', 'orange', 'purple', 'red', 'orange', 'purple']
|
|
|
|
for orm_stmt_kind in orm_stmt_kinds {
|
|
|
|
match orm_stmt_kind {
|
|
|
|
'insert' {
|
2022-09-19 14:49:23 +03:00
|
|
|
framework_platform[orm_stmt_kind] = insert_framework_benchmark_times() or {
|
|
|
|
return error('')
|
|
|
|
}.to_map()
|
2022-09-07 01:55:32 +03:00
|
|
|
}
|
|
|
|
'select' {
|
2022-09-19 14:49:23 +03:00
|
|
|
framework_platform[orm_stmt_kind] = select_framework_benchmark_times() or {
|
|
|
|
return error('')
|
|
|
|
}.to_map()
|
2022-09-07 01:55:32 +03:00
|
|
|
}
|
|
|
|
'update' {
|
2022-09-19 14:49:23 +03:00
|
|
|
framework_platform[orm_stmt_kind] = update_framework_benchmark_times() or {
|
|
|
|
return error('')
|
|
|
|
}.to_map()
|
2022-09-07 01:55:32 +03:00
|
|
|
}
|
|
|
|
else {}
|
|
|
|
}
|
|
|
|
|
|
|
|
for key, values in framework_platform[orm_stmt_kind] {
|
|
|
|
attribute_names[orm_stmt_kind] << key
|
2022-10-06 17:20:32 +03:00
|
|
|
maxs[orm_stmt_kind] << arrays.max(values) or { continue }
|
2022-09-07 01:55:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
from_framework[orm_stmt_kind] = json.encode(framework_platform[orm_stmt_kind])
|
|
|
|
table[orm_stmt_kind] = gen_table_info(attribute_names[orm_stmt_kind], framework_platform[orm_stmt_kind])
|
2022-10-06 17:20:32 +03:00
|
|
|
max_benchmark[orm_stmt_kind] = arrays.max(maxs[orm_stmt_kind]) or { continue }
|
2022-09-07 01:55:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return $vweb.html()
|
|
|
|
}
|
|
|
|
|
2022-09-19 14:49:23 +03:00
|
|
|
fn insert_framework_benchmark_times() !FrameworkPlatform {
|
2022-09-07 01:55:32 +03:00
|
|
|
numbers := FrameworkPlatform{
|
|
|
|
v_sqlite_memory: v_sqlite_memory()!.insert
|
|
|
|
// v_sqlite_file: v_sqlite_file()!.insert
|
|
|
|
typescript_sqlite_memory: typescript_sqlite_memory()!.insert
|
|
|
|
}
|
|
|
|
|
|
|
|
return numbers
|
|
|
|
}
|
|
|
|
|
2022-09-19 14:49:23 +03:00
|
|
|
fn select_framework_benchmark_times() !FrameworkPlatform {
|
2022-09-07 01:55:32 +03:00
|
|
|
numbers := FrameworkPlatform{
|
|
|
|
v_sqlite_memory: v_sqlite_memory()!.@select
|
|
|
|
// v_sqlite_file: v_sqlite_file()!.@select
|
|
|
|
typescript_sqlite_memory: typescript_sqlite_memory()!.@select
|
|
|
|
}
|
|
|
|
|
|
|
|
return numbers
|
|
|
|
}
|
|
|
|
|
2022-09-19 14:49:23 +03:00
|
|
|
fn update_framework_benchmark_times() !FrameworkPlatform {
|
2022-09-07 01:55:32 +03:00
|
|
|
numbers := FrameworkPlatform{
|
|
|
|
v_sqlite_memory: v_sqlite_memory()!.update
|
|
|
|
// v_sqlite_file: v_sqlite_file()!.@select
|
|
|
|
typescript_sqlite_memory: typescript_sqlite_memory()!.update
|
|
|
|
}
|
|
|
|
|
|
|
|
return numbers
|
|
|
|
}
|
|
|
|
|
|
|
|
fn typescript_sqlite_memory() ?FrameworkBenchmarkResponse {
|
|
|
|
url := 'http://localhost:3000/sqlite-memory/$benchmark_loop_length'
|
|
|
|
res := http.get(url) or { panic(err) }
|
|
|
|
framework_benchmark_response := json.decode(FrameworkBenchmarkResponse, res.body)?
|
|
|
|
return framework_benchmark_response
|
|
|
|
}
|
|
|
|
|
|
|
|
fn v_sqlite_memory() ?FrameworkBenchmarkResponse {
|
|
|
|
url := 'http://localhost:4000/sqlite-memory/$benchmark_loop_length'
|
|
|
|
res := http.get(url) or { panic(err) }
|
|
|
|
framework_benchmark_response := json.decode(FrameworkBenchmarkResponse, res.body)?
|
|
|
|
return framework_benchmark_response
|
|
|
|
}
|
|
|
|
|
|
|
|
fn v_sqlite_file() ?FrameworkBenchmarkResponse {
|
|
|
|
// url := 'http://localhost:3000/sqlite-memory/$benchmark_loop_length'
|
|
|
|
// res := http.get(url) or { panic(err) }
|
|
|
|
// framework_benchmark_response := json.decode(FrameworkBenchmarkResponse, res.body)?
|
|
|
|
framework_benchmark_response := FrameworkBenchmarkResponse{
|
|
|
|
insert: []
|
|
|
|
@select: []
|
|
|
|
update: []
|
|
|
|
}
|
|
|
|
return framework_benchmark_response
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gen_table_info(attribute_names []string, framework_platform map[string][]int) map[string]map[string]string {
|
|
|
|
mut table := map[string]map[string]string{}
|
|
|
|
|
|
|
|
// nanoseconds
|
|
|
|
mut max_times := map[string]int{}
|
|
|
|
mut ten_perc_max_times := map[string]int{}
|
|
|
|
mut min_times := map[string]int{}
|
|
|
|
mut ten_perc_min_times := map[string]int{}
|
|
|
|
|
|
|
|
// bigger to calculate percent
|
|
|
|
mut max := 0.0
|
|
|
|
mut ten_perc_max := 0.0
|
|
|
|
mut min := 0.0
|
|
|
|
mut ten_perc_min := 0.0
|
|
|
|
|
|
|
|
// percentes
|
|
|
|
mut max_fast := map[string]int{}
|
|
|
|
mut ten_perc_max_fast := map[string]int{}
|
|
|
|
mut min_fast := map[string]int{}
|
|
|
|
mut ten_perc_min_fast := map[string]int{}
|
|
|
|
|
|
|
|
// nanoseconds
|
|
|
|
for idx, name in attribute_names {
|
|
|
|
// qtd. of values in 10 % of arrays
|
|
|
|
ten_perc := int(framework_platform[name].len / 10)
|
|
|
|
|
|
|
|
// get 10% highter
|
|
|
|
mut min_ten_array := framework_platform[name].clone()
|
|
|
|
min_ten_array.sort()
|
|
|
|
min_ten_array.trim(ten_perc)
|
|
|
|
|
|
|
|
// get 10% lower
|
|
|
|
mut max_ten_array := framework_platform[name].clone()
|
|
|
|
max_ten_array.sort(a > b)
|
|
|
|
max_ten_array.trim(ten_perc)
|
|
|
|
|
|
|
|
// popule array with nanoseconds to which benchmark
|
|
|
|
max_times[name] = arrays.max(framework_platform[name]) or { 0 } // int
|
|
|
|
ten_perc_max_times[name] = arrays.sum(max_ten_array) or { 0 } / ten_perc // int
|
|
|
|
min_times[name] = arrays.min(framework_platform[name]) or { 0 } // int
|
|
|
|
ten_perc_min_times[name] = arrays.sum(min_ten_array) or { 0 } / ten_perc // int
|
|
|
|
|
|
|
|
// set bigger values
|
|
|
|
if idx < 1 {
|
|
|
|
max = f64(max_times[name])
|
|
|
|
ten_perc_max = f64(ten_perc_max_times[name])
|
|
|
|
min = f64(min_times[name])
|
|
|
|
ten_perc_min = f64(ten_perc_min_times[name])
|
|
|
|
} else {
|
|
|
|
if max < f64(max_times[name]) {
|
|
|
|
max = f64(max_times[name])
|
|
|
|
}
|
|
|
|
if ten_perc_max < f64(ten_perc_max_times[name]) {
|
|
|
|
ten_perc_max = f64(ten_perc_max_times[name])
|
|
|
|
}
|
|
|
|
if min < f64(min_times[name]) {
|
|
|
|
min = f64(min_times[name])
|
|
|
|
}
|
|
|
|
if ten_perc_min < f64(ten_perc_min_times[name]) {
|
|
|
|
ten_perc_min = f64(ten_perc_min_times[name])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// percents
|
|
|
|
for name in attribute_names {
|
|
|
|
max_fast[name] = int(max / f64(max_times[name]))
|
|
|
|
ten_perc_max_fast[name] = int(ten_perc_max / f64(ten_perc_max_times[name]))
|
|
|
|
min_fast[name] = int(min / f64(min_times[name]))
|
|
|
|
ten_perc_min_fast[name] = int(ten_perc_min / f64(ten_perc_min_times[name]))
|
|
|
|
}
|
|
|
|
|
|
|
|
for name in attribute_names {
|
|
|
|
table[name]['max.'] = '${math.round_sig(f64(max_times[name]) / 1000000, 2)} ms (${max_fast[name]}x faster)'
|
|
|
|
table[name]['10% max.'] = '${math.round_sig(f64(ten_perc_max_times[name]) / 1000000,
|
|
|
|
2)} ms (${ten_perc_max_fast[name]}x faster)'
|
|
|
|
table[name]['min.'] = '${math.round_sig(f64(min_times[name]) / 1000000, 2)} ms (${min_fast[name]}x faster)'
|
|
|
|
table[name]['10% min.'] = '${math.round_sig(f64(ten_perc_min_times[name]) / 1000000,
|
|
|
|
2)} ms (${ten_perc_min_fast[name]}x faster)'
|
|
|
|
}
|
|
|
|
return table
|
|
|
|
}
|