import os import time import term import v.util.version import runtime import encoding.iconv struct App { mut: report_lines []string cached_cpuinfo map[string]string vexe string } fn (mut a App) println(s string) { a.report_lines << s } fn (mut a App) collect_info() { a.line('V full version', version.full_v_version(true)) a.line(':-------------------', ':-------------------') mut os_kind := os.user_os() mut arch_details := []string{} arch_details << '${runtime.nr_cpus()} cpus' if runtime.is_32bit() { arch_details << '32bit' } if runtime.is_64bit() { arch_details << '64bit' } if runtime.is_big_endian() { arch_details << 'big endian' } if runtime.is_little_endian() { arch_details << 'little endian' } if os_kind == 'macos' { arch_details << a.cmd(command: 'sysctl -n machdep.cpu.brand_string') } if os_kind == 'linux' { mut cpu_details := '' if cpu_details == '' { cpu_details = a.cpu_info('model name') } if cpu_details == '' { cpu_details = a.cpu_info('hardware') } if cpu_details == '' { cpu_details = os.uname().machine } arch_details << cpu_details } if os_kind == 'windows' { arch_details << a.cmd( command: 'wmic cpu get name /format:table' line: 2 ) } mut os_details := '' wsl_check := a.cmd(command: 'cat /proc/sys/kernel/osrelease') if os_kind == 'linux' { os_details = a.get_linux_os_name() if a.cpu_info('flags').contains('hypervisor') { if wsl_check.contains('microsoft') { // WSL 2 is a Managed VM and Full Linux Kernel // See https://docs.microsoft.com/en-us/windows/wsl/compare-versions os_details += ' (WSL 2)' } else { os_details += ' (VM)' } } // WSL 1 is NOT a Managed VM and Full Linux Kernel // See https://docs.microsoft.com/en-us/windows/wsl/compare-versions if wsl_check.contains('Microsoft') { os_details += ' (WSL)' } // From https://unix.stackexchange.com/a/14346 awk_cmd := '[ "$(awk \'\$5=="/" {print \$1}\' 0 && output.len > c.line { return output[c.line] } } return 'Error: ${x.output}' } fn (mut a App) line(label string, value string) { a.println('|${label:-20}|${term.colorize(term.bold, value)}') } fn (mut a App) line2(label string, value string, value2 string) { a.println('|${label:-20}|${term.colorize(term.bold, value)}, value: ${term.colorize(term.bold, value2)}') } fn (mut a App) line_env(env_var string) { value := os.getenv(env_var) if value != '' { a.line('env ${env_var}', '"${value}"') } } fn (app &App) parse(config string, sep string) map[string]string { mut m := map[string]string{} lines := config.split_into_lines() for line in lines { sline := line.trim_space() if sline.len == 0 || sline[0] == `#` { continue } x := sline.split(sep) if x.len < 2 { continue } m[x[0].trim_space().to_lower()] = x[1].trim_space().trim('"') } return m } fn (mut a App) get_linux_os_name() string { if os.is_file('/etc/os-release') { if lines := os.read_file('/etc/os-release') { vals := a.parse(lines, '=') if vals['PRETTY_NAME'] != '' { return vals['PRETTY_NAME'] } } } if !a.cmd(command: 'type lsb_release').starts_with('Error') { return a.cmd(command: 'lsb_release -d -s') } if os.is_file('/proc/version') { return a.cmd(command: 'cat /proc/version') } ouname := os.uname() return '${ouname.release}, ${ouname.version}' } fn (mut a App) cpu_info(key string) string { if a.cached_cpuinfo.len > 0 { return a.cached_cpuinfo[key] } info := os.execute('cat /proc/cpuinfo') if info.exit_code != 0 { return '`cat /proc/cpuinfo` could not run' } a.cached_cpuinfo = a.parse(info.output, ':') return a.cached_cpuinfo[key] } fn (mut a App) git_info() string { // Check if in a Git repository x := os.execute('git rev-parse --is-inside-work-tree') if x.exit_code != 0 || x.output.trim_space() != 'true' { return 'N/A' } mut out := a.cmd(command: 'git -C . describe --abbrev=8 --dirty --always --tags').trim_space() os.execute('git -C . remote add V_REPO https://github.com/vlang/v') // ignore failure (i.e. remote exists) if '-skip-github' !in os.args { os.execute('${os.quoted_path(a.vexe)} timeout 5.1 "git -C . fetch V_REPO"') // usually takes ~0.6s; 5 seconds should be enough for even the slowest networks } commit_count := a.cmd(command: 'git rev-list @{0}...V_REPO/master --right-only --count').int() if commit_count > 0 { out += ' (${commit_count} commit(s) behind V master)' } return out } fn (mut a App) report_tcc_version(tccfolder string) { cmd := os.join_path(tccfolder, 'tcc.exe') + ' -v' x := os.execute(cmd) if x.exit_code == 0 { a.line('tcc version', '${x.output.trim_space()}') } else { a.line('tcc version', 'N/A') } if !os.is_file(os.join_path(tccfolder, '.git', 'config')) { a.line('tcc git status', 'N/A') } else { tcc_branch_name := a.cmd( command: 'git -C ${os.quoted_path(tccfolder)} rev-parse --abbrev-ref HEAD' ) tcc_commit := a.cmd( command: 'git -C ${os.quoted_path(tccfolder)} describe --abbrev=8 --dirty --always --tags' ) a.line('tcc git status', '${tcc_branch_name} ${tcc_commit}') } } fn (mut a App) report_info() { for x in a.report_lines { println(x) } } fn is_writable_dir(path string) bool { os.ensure_folder_is_writable(path) or { return false } return true } fn diagnose_dir(path string) string { mut diagnostics := []string{} if !is_writable_dir(path) { diagnostics << 'NOT writable' } if path.contains(' ') { diagnostics << 'contains spaces' } path_non_ascii_runes := path.runes().filter(it > 255) if path_non_ascii_runes.len > 0 { diagnostics << 'contains these non ASCII characters: ${path_non_ascii_runes}' } if diagnostics.len == 0 { diagnostics << 'OK' } return diagnostics.join(', ') } fn main() { mut app := App{} app.vexe = os.getenv('VEXE') app.collect_info() app.report_info() }