v/cmd/tools/vup.v
2025-02-20 03:07:30 +03:00

207 lines
6.1 KiB
V

module main
import os
import v.util.version
import v.util.recompilation
const vexe = os.real_path(os.getenv_opt('VEXE') or { @VEXE })
const vroot = os.dir(vexe)
struct App {
is_verbose bool
is_prod bool
vexe string
vroot string
skip_v_self bool // do not run `v self`, effectively enforcing the running of `make` or `make.bat`
skip_current bool // skip the current hash check, enabling easier testing on the same commit, without using docker etc
}
const args = arguments()
fn new_app() App {
return App{
is_verbose: '-v' in args
is_prod: '-prod' in args
vexe: vexe
vroot: vroot
skip_v_self: '-skip_v_self' in args
skip_current: '-skip_current' in args
}
}
fn main() {
app := new_app()
recompilation.must_be_enabled(app.vroot, 'Please install V from source, to use `v up` .')
os.chdir(app.vroot)!
println('Updating V...')
app.update_from_master()
hash_when_vup_was_compiled := @VCURRENTHASH
current_hash_from_filesystem := version.githash(vroot) or { hash_when_vup_was_compiled }
if !app.skip_current && hash_when_vup_was_compiled == current_hash_from_filesystem {
println('V is already updated.')
app.show_current_v_version()
return
}
if os.user_os() == 'windows' {
app.backup('cmd/tools/vup.exe')
}
if !app.recompile_v() {
app.show_current_v_version()
eprintln('Recompiling V *failed*.')
eprintln('Try running `${get_make_cmd_name()}` .')
exit(1)
}
app.recompile_vup()
app.show_current_v_version()
}
fn (app App) vprintln(s string) {
if app.is_verbose {
println(s)
}
}
fn (app App) update_from_master() {
app.vprintln('> updating from master ...')
if !os.exists('.git') {
// initialize the folder, as if it had been cloned:
app.git_command('git init')
app.git_command('git remote add origin https://github.com/vlang/v')
app.git_command('git fetch')
app.git_command('git remote set-head origin master')
app.git_command('git reset --hard origin/master')
// Note 1: patterns starting with /, will match only against the root;
// `--exclude v` will match also vlib/v/ in addition to ./v; `--exclude /v` will only match ./v
// Note 2: patterns ending with / are treated as folders.
app.git_command('git clean -xfd --exclude /thirdparty/tcc/ --exclude /v --exclude /v.exe --exclude /cmd/tools/vup --exclude /cmd/tools/vup.exe')
} else {
// pull latest
app.git_command('git pull https://github.com/vlang/v master')
}
}
fn (app App) recompile_v() bool {
// Note: app.vexe is more reliable than just v (which may be a symlink)
opts := if app.is_prod { '-prod' } else { '' }
vself := '${os.quoted_path(app.vexe)} ${opts} self'
if app.skip_v_self {
return app.make(vself)
}
self_result := os.execute(vself)
if self_result.exit_code == 0 {
println(self_result.output.trim_space())
println('> Done recompiling.')
return true
} else {
println('> `${vself}` failed, running `make`...')
app.vprintln(self_result.output.trim_space())
}
return app.make(vself)
}
fn (app App) recompile_vup() bool {
eprintln('> Recompiling vup.v ...')
vup_result := os.execute('${os.quoted_path(app.vexe)} -g cmd/tools/vup.v')
if vup_result.exit_code != 0 {
eprintln('> Failed recompiling vup.v .')
eprintln(vup_result.output)
return false
}
return true
}
fn (app App) make(vself string) bool {
println('> running make ...')
make := get_make_cmd_name()
make_result := os.execute(make)
if make_result.exit_code != 0 {
eprintln('> ${make} failed:')
eprintln('> make output:')
eprintln(make_result.output)
return false
}
app.vprintln(make_result.output)
println('> done running make.')
return true
}
fn (app App) show_current_v_version() {
vout := os.execute('${os.quoted_path(app.vexe)} version')
if vout.exit_code >= 0 {
mut vversion := vout.output.trim_space()
if vout.exit_code == 0 {
latest_v_commit := vversion.split(' ').last().all_after('.')
latest_v_commit_time := os.execute('git show -s --format=%ci ${latest_v_commit}')
if latest_v_commit_time.exit_code == 0 {
vversion += ', timestamp: ' + latest_v_commit_time.output.trim_space()
}
}
println('Current V version: ${vversion}')
}
}
fn (app App) backup(file string) {
backup_file := '${file}_old.exe'
println('> backing up `${file}` to `${backup_file}` ...')
if os.exists(backup_file) {
os.rm(backup_file) or { eprintln('failed removing ${backup_file}: ${err.msg()}') }
}
os.mv(file, backup_file) or { eprintln('failed moving ${file}: ${err.msg()}') }
}
fn (app App) git_command(command string) {
println('> git_command: ${command}')
git_result := os.execute(command)
if git_result.exit_code < 0 {
app.install_git()
// Try it again with (maybe) git installed
os.execute_or_exit(command)
}
if git_result.exit_code != 0 {
eprintln('Failed git command: ${command}')
eprintln(git_result.output)
exit(1)
}
app.vprintln(git_result.output)
}
fn (app App) install_git() {
if os.user_os() != 'windows' {
// Probably some kind of *nix, usually need to get using a package manager.
eprintln("error: Install `git` using your system's package manager")
}
println('Downloading git 32 bit for Windows, please wait.')
// We'll use 32 bit because maybe someone out there is using 32-bit windows
res_download := os.execute('bitsadmin.exe /transfer "vgit" https://github.com/git-for-windows/git/releases/download/v2.30.0.windows.2/Git-2.30.0.2-32-bit.exe "${os.getwd()}/git32.exe"')
if res_download.exit_code != 0 {
eprintln('Unable to install git automatically: please install git manually')
panic(res_download.output)
}
res_git32 := os.execute(os.quoted_path(os.join_path_single(os.getwd(), 'git32.exe')))
if res_git32.exit_code != 0 {
eprintln('Unable to install git automatically: please install git manually')
panic(res_git32.output)
}
}
fn get_make_cmd_name() string {
if os.user_os() == 'windows' {
return 'make.bat'
}
cmd := 'make'
make_sure_cmd_is_available(cmd)
cc := os.getenv_opt('CC') or { 'cc' }
make_sure_cmd_is_available(cc)
return cmd
}
fn make_sure_cmd_is_available(cmd string) {
found_path := os.find_abs_path_of_executable(cmd) or {
eprintln('Could not find `${cmd}` in PATH. Please install `${cmd}`, since `v up` needs it.')
exit(1)
}
println('Found `${cmd}` as `${found_path}`.')
}