module main import os import testing struct Config { run_just_essential bool = '${os.getenv('VTEST_JUST_ESSENTIAL')}${os.getenv('VTEST_SANDBOXED_PACKAGING')}' != '' run_slow_sanitize bool = os.getenv('VTEST_RUN_FSANITIZE_TOO_SLOW') != '' is_musl_ci bool = os.getenv('V_CI_MUSL') != '' is_ubuntu_musl_ci bool = os.getenv('V_CI_UBUNTU_MUSL') != '' is_sandboxed_packaging bool = os.getenv('VTEST_SANDBOXED_PACKAGING') != '' github_job string = os.getenv('GITHUB_JOB') mut: test_dirs []string = ['cmd', 'vlib'] is_asan_compiler bool is_msan_compiler bool is_ubsan_compiler bool // Options relating to the v command itself (passed in the prefix) `v [...args] test-self`. werror bool sanitize_memory bool sanitize_address bool sanitize_undefined bool } const vroot = os.dir(os.real_path(os.getenv_opt('VEXE') or { @VEXE })) const essential_list = [ 'cmd/tools/vvet/vet_test.v', 'cmd/tools/vdoc/document/doc_test.v', 'vlib/arrays/arrays_test.v', 'vlib/bitfield/bitfield_test.v', // 'vlib/builtin/int_test.v', 'vlib/builtin/array_test.v', 'vlib/builtin/float_test.v', 'vlib/builtin/byte_test.v', 'vlib/builtin/rune_test.v', 'vlib/builtin/builtin_test.c.v', 'vlib/builtin/map_of_floats_test.v', 'vlib/builtin/string_int_test.v', 'vlib/builtin/utf8_test.v', 'vlib/builtin/map_test.v', 'vlib/builtin/string_test.v', 'vlib/builtin/sorting_test.v', 'vlib/builtin/gated_array_string_test.v', 'vlib/builtin/isnil_test.v', 'vlib/builtin/string_match_glob_test.v', 'vlib/builtin/string_strip_margin_test.v', // 'vlib/cli/command_test.v', 'vlib/crypto/md5/md5_test.v', 'vlib/dl/dl_test.v', 'vlib/encoding/base64/base64_test.v', 'vlib/encoding/utf8/validate/encoding_utf8_test.v', 'vlib/encoding/utf8/utf8_util_test.v', 'vlib/flag/flag_test.v', 'vlib/json/tests/json_decode_test.v', 'vlib/math/math_test.v', 'vlib/net/tcp_test.v', 'vlib/net/http/http_test.v', 'vlib/net/http/server_test.v', 'vlib/net/http/request_test.v', 'vlib/io/io_test.v', 'vlib/io/os_file_reader_test.v', 'vlib/os/process_test.v', 'vlib/os/file_test.v', 'vlib/os/notify/notify_test.c.v', 'vlib/os/filepath_test.v', 'vlib/os/environment_test.v', 'vlib/os/glob_test.v', 'vlib/os/os_test.c.v', 'vlib/rand/random_numbers_test.v', 'vlib/rand/wyrand/wyrand_test.v', 'vlib/runtime/runtime_test.v', 'vlib/semver/semver_test.v', 'vlib/sync/stdatomic/atomic_test.v', 'vlib/sync/thread_test.v', 'vlib/sync/waitgroup_test.v', 'vlib/sync/pool/pool_test.v', 'vlib/strings/builder_test.v', 'vlib/strconv/atof_test.c.v', 'vlib/strconv/atoi_test.v', 'vlib/strconv/f32_f64_to_string_test.v', 'vlib/strconv/format_test.v', 'vlib/strconv/number_to_base_test.v', 'vlib/time/time_test.v', 'vlib/toml/tests/toml_test.v', 'vlib/v/compiler_errors_test.v', 'vlib/v/eval/interpret_test.v', 'vlib/v/fmt/fmt_keep_test.v', 'vlib/v/fmt/fmt_test.v', 'vlib/v/gen/c/coutput_test.v', 'vlib/v/gen/js/program_test.v', 'vlib/v/gen/native/macho_test.v', 'vlib/v/gen/native/tests/native_test.v', 'vlib/v/pkgconfig/pkgconfig_test.v', 'vlib/v/slow_tests/inout/compiler_test.v', 'vlib/x/json2/tests/json2_test.v', ] const skip_test_files = [ 'do_not_remove', 'cmd/tools/vdoc/vdoc_test.v', // markdown not installed 'vlib/context/deadline_test.v', // sometimes blocks 'vlib/context/onecontext/onecontext_test.v', // backtrace_symbols is missing 'vlib/db/mysql/mysql_orm_test.v', // mysql not installed 'vlib/db/mysql/mysql_test.v', // mysql not installed 'vlib/db/mysql/prepared_stmt_test.v', // mysql not installed 'vlib/db/pg/pg_orm_test.v', // pg not installed 'vlib/db/pg/pg_test.v', // pg not installed 'vlib/db/pg/pg_double_test.v', // pg not installed 'vlib/net/ftp/ftp_test.v', // currently broken ] // These tests are too slow to be run in the CI on each PR/commit // in the sanitized modes: const skip_fsanitize_too_slow = [ 'do_not_remove', 'vlib/v/compiler_errors_test.v', 'vlib/v/fmt/fmt_test.v', 'vlib/v/fmt/fmt_keep_test.v', 'vlib/v/fmt/fmt_vlib_test.v', 'vlib/v/live/live_test.v', 'vlib/v/parser/v_parser_test.v', 'vlib/v/scanner/scanner_test.v', 'vlib/v/slow_tests/inout/compiler_test.v', 'vlib/v/slow_tests/prod_test.v', 'vlib/v/slow_tests/profile/profile_test.v', 'vlib/v/slow_tests/repl/repl_test.v', 'vlib/v/slow_tests/valgrind/valgrind_test.v', 'cmd/tools/vpm/dependency_test.v', 'cmd/tools/vpm/install_test.v', 'cmd/tools/vpm/install_version_input_test.v', 'cmd/tools/vpm/install_version_test.v', 'cmd/tools/vpm/update_test.v', 'cmd/tools/vdoc/document/doc_test.v', ] const skip_with_fsanitize_memory = [ 'do_not_remove', 'cmd/tools/vpm/dependency_test.v', // known flaky, for fsanitize_memory, due to using mbedtls 'cmd/tools/vpm/install_test.v', // known flaky, for fsanitize_memory, due to using mbedtls 'cmd/tools/vpm/install_version_input_test.v', // known flaky, for fsanitize_memory, due to using mbedtls 'cmd/tools/vpm/install_version_test.v', // known flaky, for fsanitize_memory, due to using mbedtls 'cmd/tools/vpm/update_test.v', // known flaky, for fsanitize_memory, due to using mbedtls 'vlib/net/tcp_simple_client_server_test.v', 'vlib/net/http/cookie_test.v', 'vlib/net/http/http_test.v', 'vlib/net/http/status_test.v', 'vlib/net/http/header_test.v', 'vlib/net/http/server_test.v', 'vlib/net/udp_test.v', 'vlib/net/tcp_test.v', 'vlib/orm/orm_test.v', 'vlib/orm/orm_sql_or_blocks_test.v', 'vlib/orm/orm_create_and_drop_test.v', 'vlib/orm/orm_insert_test.v', 'vlib/orm/orm_insert_reserved_name_test.v', 'vlib/orm/orm_sum_type_insert_test.v', 'vlib/orm/orm_fn_calls_test.v', 'vlib/orm/orm_last_id_test.v', 'vlib/orm/orm_string_interpolation_in_where_test.v', 'vlib/orm/orm_interface_test.v', 'vlib/orm/orm_mut_db_test.v', 'vlib/orm/orm_null_test.v', 'vlib/orm/orm_result_test.v', 'vlib/orm/orm_custom_operators_test.v', 'vlib/orm/orm_fk_test.v', 'vlib/orm/orm_references_test.v', 'vlib/orm/orm_option_array_test.v', 'vlib/orm/orm_option_time_test.v', 'vlib/orm/orm_order_by_custom_field_test.v', 'vlib/orm/orm_serial_attribute_test.v', 'vlib/orm/orm_option_subselect_test.v', 'vlib/db/sqlite/sqlite_test.v', 'vlib/db/sqlite/sqlite_orm_test.v', 'vlib/db/sqlite/sqlite_comptime_field_test.v', 'vlib/db/sqlite/parent_child_test.v', 'vlib/db/sqlite/sqlite_vfs_lowlevel_test.v', 'vlib/v/tests/orm_enum_test.v', 'vlib/v/tests/orm_sub_struct_test.v', 'vlib/v/tests/orm_sub_array_struct_test.v', 'vlib/v/tests/orm_joined_tables_select_test.v', 'vlib/v/tests/sql_statement_inside_fn_call_test.v', 'vlib/v/tests/orm_stmt_wrong_return_checking_test.v', 'vlib/v/tests/orm_table_name_test.v', 'vlib/v/tests/orm_array_field_test.v', 'vlib/v/tests/orm_handle_error_for_select_from_not_created_table_test.v', 'vlib/v/tests/orm_create_several_tables_test.v', 'vlib/v/tests/orm_update_test.v', 'vlib/vweb/tests/vweb_test.v', 'vlib/vweb/csrf/csrf_test.v', 'vlib/net/http/request_test.v', 'vlib/net/http/response_test.v', 'vlib/vweb/route_test.v', 'vlib/net/websocket/websocket_test.v', 'vlib/net/smtp/smtp_test.v', 'vlib/v/tests/websocket_logger_interface_should_compile_test.v', 'vlib/v/tests/fns/fn_literal_type_test.v', 'vlib/x/sessions/tests/db_store_test.v', ] const skip_with_fsanitize_address = [ 'do_not_remove', 'vlib/net/websocket/websocket_test.v', 'vlib/orm/orm_create_and_drop_test.v', 'vlib/orm/orm_insert_test.v', 'vlib/orm/orm_insert_reserved_name_test.v', 'vlib/orm/orm_sum_type_insert_test.v', 'vlib/orm/orm_references_test.v', 'vlib/v/tests/websocket_logger_interface_should_compile_test.v', 'vlib/v/tests/orm_enum_test.v', 'vlib/v/tests/orm_sub_array_struct_test.v', 'vlib/v/tests/orm_handle_error_for_select_from_not_created_table_test.v', 'vlib/v/tests/orm_create_several_tables_test.v', 'vlib/v/tests/orm_update_test.v', ] const skip_with_fsanitize_undefined = [ 'do_not_remove', 'vlib/orm/orm_create_and_drop_test.v', 'vlib/orm/orm_insert_test.v', 'vlib/orm/orm_insert_reserved_name_test.v', 'vlib/orm/orm_sum_type_insert_test.v', 'vlib/orm/orm_references_test.v', 'vlib/v/tests/orm_enum_test.v', 'vlib/v/tests/orm_sub_array_struct_test.v', 'vlib/v/tests/orm_handle_error_for_select_from_not_created_table_test.v', 'vlib/v/tests/orm_create_several_tables_test.v', 'vlib/v/tests/orm_update_test.v', 'vlib/v/tests/project_with_cpp_code/compiling_cpp_files_with_a_cplusplus_compiler_test.c.v', // fails compilation with: undefined reference to vtable for __cxxabiv1::__function_type_info' ] const skip_with_werror = [ 'do_not_remove', 'vlib/v/embed_file/tests/embed_file_test.v', ] const skip_with_asan_compiler = [ 'do_not_remove', ] const skip_with_msan_compiler = [ 'do_not_remove', ] const skip_with_ubsan_compiler = [ 'do_not_remove', ] const skip_on_musl = [ 'do_not_remove', 'vlib/v/slow_tests/profile/profile_test.v', 'vlib/gg/draw_fns_api_test.v', 'vlib/v/tests/skip_unused/gg_code.vv', 'vlib/v/tests/c_struct_with_reserved_field_name_test.v', 'vlib/arrays/parallel/parallel_test.v', ] const skip_on_ubuntu_musl = [ 'do_not_remove', 'vlib/arrays/parallel/parallel_test.v', //'vlib/v/gen/js/jsgen_test.v', 'vlib/net/http/cookie_test.v', 'vlib/net/http/status_test.v', 'vlib/db/sqlite/sqlite_test.v', 'vlib/db/sqlite/sqlite_orm_test.v', 'vlib/db/sqlite/sqlite_comptime_field_test.v', 'vlib/db/sqlite/sqlite_vfs_lowlevel_test.v', 'vlib/db/sqlite/parent_child_test.v', 'vlib/orm/orm_test.v', 'vlib/orm/orm_sql_or_blocks_test.v', 'vlib/orm/orm_create_and_drop_test.v', 'vlib/orm/orm_insert_test.v', 'vlib/orm/orm_insert_reserved_name_test.v', 'vlib/orm/orm_sum_type_insert_test.v', 'vlib/orm/orm_fn_calls_test.v', 'vlib/orm/orm_null_test.v', 'vlib/orm/orm_last_id_test.v', 'vlib/orm/orm_string_interpolation_in_where_test.v', 'vlib/orm/orm_interface_test.v', 'vlib/orm/orm_mut_db_test.v', 'vlib/orm/orm_result_test.v', 'vlib/orm/orm_custom_operators_test.v', 'vlib/orm/orm_fk_test.v', 'vlib/orm/orm_references_test.v', 'vlib/orm/orm_option_array_test.v', 'vlib/orm/orm_option_time_test.v', 'vlib/orm/orm_order_by_custom_field_test.v', 'vlib/orm/orm_serial_attribute_test.v', 'vlib/orm/orm_option_subselect_test.v', 'vlib/v/tests/orm_enum_test.v', 'vlib/v/tests/orm_sub_struct_test.v', 'vlib/v/tests/orm_sub_array_struct_test.v', 'vlib/v/tests/orm_joined_tables_select_test.v', 'vlib/v/tests/orm_stmt_wrong_return_checking_test.v', 'vlib/v/tests/orm_table_name_test.v', 'vlib/v/tests/orm_array_field_test.v', 'vlib/v/tests/orm_handle_error_for_select_from_not_created_table_test.v', 'vlib/v/tests/orm_create_several_tables_test.v', 'vlib/v/tests/orm_update_test.v', 'vlib/v/tests/sql_statement_inside_fn_call_test.v', 'vlib/clipboard/clipboard_test.v', 'vlib/vweb/tests/vweb_test.v', 'vlib/vweb/csrf/csrf_test.v', 'vlib/net/http/request_test.v', 'vlib/vweb/route_test.v', 'vlib/net/websocket/websocket_test.v', 'vlib/net/http/header_test.v', 'vlib/net/http/server_test.v', 'vlib/net/http/response_test.v', 'vlib/builtin/js/array_test.js.v', 'vlib/net/smtp/smtp_test.v', 'vlib/v/tests/websocket_logger_interface_should_compile_test.v', 'vlib/v/tests/fns/fn_literal_type_test.v', 'vlib/x/sessions/tests/db_store_test.v', 'vlib/veb/tests/veb_app_test.v', ] const skip_on_linux = [ 'do_not_remove', ] const skip_on_non_linux = [ 'do_not_remove', ] const skip_on_windows_msvc = [ 'do_not_remove', 'vlib/v/tests/consts/const_fixed_array_containing_references_to_itself_test.v', // error C2099: initializer is not a constant 'vlib/v/tests/consts/const_and_global_with_same_name_test.v', // error C2099: initializer is not a constant 'vlib/v/tests/sumtypes/sumtype_as_cast_1_test.v', // error: cannot support compound statement expression ({expr; expr; expr;}) 'vlib/v/tests/sumtypes/sumtype_as_cast_2_test.v', // error: cannot support compound statement expression ({expr; expr; expr;}) 'vlib/v/tests/project_with_cpp_code/compiling_cpp_files_with_a_cplusplus_compiler_test.c.v', // TODO ] const skip_on_windows = [ 'do_not_remove', 'vlib/orm/orm_test.v', 'vlib/v/tests/orm_sub_struct_test.v', 'vlib/v/tests/orm_joined_tables_select_test.v', 'vlib/v/tests/orm_handle_error_for_select_from_not_created_table_test.v', 'vlib/net/websocket/websocket_test.v', 'vlib/net/openssl/openssl_compiles_test.c.v', 'vlib/net/http/request_test.v', 'vlib/net/smtp/smtp_test.v', 'vlib/net/ssl/ssl_compiles_test.v', 'vlib/net/mbedtls/mbedtls_compiles_test.v', 'vlib/vweb/tests/vweb_test.v', 'vlib/vweb/route_test.v', 'vlib/sync/many_times_test.v', 'vlib/sync/once_test.v', 'vlib/v/tests/websocket_logger_interface_should_compile_test.v', 'vlib/v/tests/fns/fn_literal_type_test.v', ] const skip_on_non_windows = [ 'do_not_remove', ] const skip_on_macos = [ 'do_not_remove', ] const skip_on_non_macos = [ 'do_not_remove', ] const skip_on_amd64 = [ 'do_not_remove', ] const skip_on_arm64 = [ 'do_not_remove', ] const skip_on_non_amd64_or_arm64 = [ 'do_not_remove', // closures aren't implemented yet: 'vlib/v/tests/fns/closure_test.v', // native aren't implemented: 'vlib/v/gen/native/tests/native_test.v', 'vlib/context/cancel_test.v', 'vlib/context/deadline_test.v', 'vlib/context/empty_test.v', 'vlib/context/value_test.v', 'vlib/context/onecontext/onecontext_test.v', 'vlib/sync/once_test.v', 'vlib/sync/many_times_test.v', 'do_not_remove', ] const skip_on_sandboxed_packaging = [ 'do_not_remove', 'vlib/v/slow_tests/inout/compiler_test.v', 'vlib/v/gen/native/tests/native_test.v', 'vlib/v/compiler_errors_test.v', 'vlib/v/gen/c/coutput_test.v', ] fn Config.init(vargs []string, targs []string) !Config { mut cfg := Config{} for arg in vargs { match arg { '-Werror', '-cstrict' { cfg.werror = true } '-fsanitize=memory' { cfg.sanitize_memory = true } '-fsanitize=address' { cfg.sanitize_address = true } '-fsanitize=undefined' { cfg.sanitize_undefined = true } else {} } } if targs.len == 0 { return cfg } mut tdirs := []string{} mut errs := []string{} for arg in targs { match arg { '-asan-compiler', '--asan-compiler' { cfg.is_asan_compiler = true } '-msan-compiler', '--msan-compiler' { cfg.is_msan_compiler = true } '-ubsan-compiler', '--ubsan-compiler' { cfg.is_ubsan_compiler = true } else { if arg.starts_with('-') { errs << 'error: unknown flag `${arg}`' continue } if !os.is_dir(os.join_path(vroot, arg)) { errs << 'error: failed to find directory `${arg}`' continue } tdirs << arg } } } if errs.len > 0 { return error(errs.join_lines()) } if tdirs.len > 0 { cfg.test_dirs = tdirs } return cfg } fn main() { os.chdir(vroot) or { panic(err) } args_idx := os.args.index('test-self') vargs := os.args[1..args_idx] targs := os.args#[args_idx + 1..] cfg := Config.init(vargs, targs) or { eprintln(err) exit(1) } // dump(cfg) title := 'testing: ${cfg.test_dirs.join(', ')}' mut tpaths := map[string]bool{} mut tpaths_ref := &tpaths for dir in cfg.test_dirs { os.walk(os.join_path(vroot, dir), fn [mut tpaths_ref] (p string) { if p.ends_with('_test.v') || p.ends_with('_test.c.v') || (testing.is_node_present && p.ends_with('_test.js.v')) { unsafe { tpaths_ref[p] = true } } }) } mut all_test_files := tpaths.keys() if cfg.run_just_essential { all_test_files = essential_list.map(os.join_path(vroot, it)) } mut tsession := testing.new_test_session(vargs.join(' '), true) tsession.exec_mode = .compile_and_run tsession.files << all_test_files.filter(!it.contains('testdata' + os.path_separator)) tsession.skip_files << skip_test_files if !testing.is_go_present { tsession.skip_files << 'vlib/v/gen/golang/tests/golang_test.v' } testing.find_started_process('mysqld') or { tsession.skip_files << 'vlib/db/mysql/mysql_orm_test.v' tsession.skip_files << 'vlib/db/mysql/mysql_test.v' } testing.find_started_process('postgres') or { tsession.skip_files << 'vlib/db/pg/pg_orm_test.v' tsession.skip_files << 'vlib/db/pg/pg_double_test.v' } $if windows && tinyc { tsession.skip_files << 'vlib/v/tests/project_with_cpp_code/compiling_cpp_files_with_a_cplusplus_compiler_test.c.v' } if !cfg.run_slow_sanitize && ((cfg.sanitize_undefined || cfg.sanitize_memory || cfg.sanitize_address) || (cfg.is_msan_compiler || cfg.is_asan_compiler || cfg.is_ubsan_compiler)) { tsession.skip_files << skip_fsanitize_too_slow } if cfg.werror { tsession.skip_files << skip_with_werror } if cfg.sanitize_memory { tsession.skip_files << skip_with_fsanitize_memory } if cfg.sanitize_address { tsession.skip_files << skip_with_fsanitize_address } if cfg.sanitize_undefined { tsession.skip_files << skip_with_fsanitize_undefined } if cfg.is_asan_compiler { tsession.skip_files << skip_with_asan_compiler } if cfg.is_msan_compiler { tsession.skip_files << skip_with_msan_compiler } if cfg.is_ubsan_compiler { tsession.skip_files << skip_with_ubsan_compiler } if cfg.is_musl_ci { tsession.skip_files << skip_on_musl } if cfg.is_ubuntu_musl_ci { tsession.skip_files << skip_on_ubuntu_musl } if cfg.is_sandboxed_packaging { tsession.skip_files << skip_on_sandboxed_packaging } $if !amd64 && !arm64 { tsession.skip_files << skip_on_non_amd64_or_arm64 } $if amd64 { tsession.skip_files << skip_on_amd64 } $if arm64 { tsession.skip_files << skip_on_arm64 } $if !linux { tsession.skip_files << skip_on_non_linux } $if linux { tsession.skip_files << skip_on_linux } $if windows { tsession.skip_files << skip_on_windows $if msvc { tsession.skip_files << skip_on_windows_msvc } } $if !windows { tsession.skip_files << skip_on_non_windows } $if macos { $if arm64 { if cfg.github_job.starts_with('clang-') { tsession.skip_files << 'vlib/net/openssl/openssl_compiles_test.c.v' } } tsession.skip_files << skip_on_macos } $if !macos { tsession.skip_files << skip_on_non_macos } // dump(tsession.skip_files) mut unavailable_files := tsession.files.filter(!os.exists(it)) unavailable_files << tsession.skip_files.filter(it != 'do_not_remove' && !os.exists(it)) if unavailable_files.len > 0 { for f in unavailable_files { eprintln('error: failed to find file: ${f}') } exit(1) } tsession.skip_files = tsession.skip_files.map(os.abs_path) tsession.session_start(title) tsession.test() tsession.session_stop(title) if tsession.benchmark.nfail > 0 { eprintln('\nError: failed ${tsession.benchmark.nfail} times.\n') exit(1) } }