1
0
mirror of https://github.com/vlang/v.git synced 2023-08-10 21:13:21 +03:00
v/cmd/v/v.v
Alexander Medvednikov 6756d28595 all: 2023 copyright
2023-03-28 22:55:57 +02:00

211 lines
5.7 KiB
V
Executable File

// Copyright (c) 2019-2023 Alexander Medvednikov. All rights reserved.
// Use of this source code is governed by an MIT license
// that can be found in the LICENSE file.
module main
import os
import term
import v.help
import v.pref
import v.util
import v.util.version
import v.builder
import v.builder.cbuilder
const (
external_tools = [
'ast',
'bin2v',
'bug',
'build-examples',
'build-tools',
'build-vbinaries',
'bump',
'check-md',
'complete',
'compress',
'doc',
'doctor',
'fmt',
'gret',
'ls',
'missdoc',
'repl',
'self',
'setup-freetype',
'shader',
'share',
'should-compile-all',
'symlink',
'scan',
'test',
'test-all', // runs most of the tests and other checking tools, that will be run by the CI
'test-cleancode',
'test-fmt',
'test-parser',
'test-self',
'tracev',
'up',
'vet',
'wipe-cache',
'watch',
'where',
]
list_of_flags_that_allow_duplicates = ['cc', 'd', 'define', 'cf', 'cflags']
)
fn main() {
mut timers_should_print := false
$if time_v ? {
timers_should_print = true
}
mut timers := util.new_timers(should_print: timers_should_print, label: 'main')
timers.start('v total')
defer {
timers.show('v total')
}
timers.start('v start')
timers.show('v start')
timers.start('parse_CLI_args')
args := os.args[1..]
if args.len == 0 || args[0] in ['-', 'repl'] {
if args.len == 0 {
// Running `./v` without args launches repl
if os.is_atty(0) == 0 {
mut args_and_flags := util.join_env_vflags_and_os_args()[1..].clone()
args_and_flags << ['run', '-']
pref.parse_args_and_show_errors(external_tools, args_and_flags, true)
}
}
util.launch_tool(false, 'vrepl', os.args[1..])
return
}
mut args_and_flags := util.join_env_vflags_and_os_args()[1..]
prefs, command := pref.parse_args_and_show_errors(external_tools, args_and_flags,
true)
if prefs.use_cache && os.user_os() == 'windows' {
eprintln('-usecache is currently disabled on windows')
exit(1)
}
timers.show('parse_CLI_args')
// Start calling the correct functions/external tools
// Note for future contributors: Please add new subcommands in the `match` block below.
if command in external_tools {
// External tools
util.launch_tool(prefs.is_verbose, 'v' + command, os.args[1..])
return
}
match command {
'run', 'crun', 'build', 'build-module' {
rebuild(prefs)
return
}
'help' {
invoke_help_and_exit(args)
}
'version' {
println(version.full_v_version(prefs.is_verbose))
return
}
'new', 'init' {
util.launch_tool(prefs.is_verbose, 'vcreate', os.args[1..])
return
}
'install', 'list', 'outdated', 'remove', 'search', 'show', 'update', 'upgrade' {
util.launch_tool(prefs.is_verbose, 'vpm', os.args[1..])
return
}
'vlib-docs' {
util.launch_tool(prefs.is_verbose, 'vdoc', ['doc', 'vlib'])
}
'interpret' {
util.launch_tool(prefs.is_verbose, 'builders/interpret_builder', os.args[1..])
}
'get' {
eprintln('V Error: Use `v install` to install modules from vpm.vlang.io')
exit(1)
}
'translate' {
util.launch_tool(prefs.is_verbose, 'translate', os.args[1..])
// exit(1)
// return
}
else {
if command.ends_with('.v') || os.exists(command) {
// println('command')
// println(prefs.path)
rebuild(prefs)
return
}
}
}
if prefs.is_help {
invoke_help_and_exit(args)
}
other_commands := ['run', 'crun', 'build', 'build-module', 'help', 'version', 'new', 'init',
'install', 'list', 'outdated', 'remove', 'search', 'show', 'update', 'upgrade', 'vlib-docs',
'interpret', 'translate']
mut all_commands := []string{}
all_commands << external_tools
all_commands << other_commands
all_commands.sort()
eprintln(util.new_suggestion(command, all_commands).say('v: unknown command `${command}`'))
eprintln('Run ${term.highlight_command('v help')} for usage.')
exit(1)
}
fn invoke_help_and_exit(remaining []string) {
match remaining.len {
0, 1 { help.print_and_exit('default') }
2 { help.print_and_exit(remaining[1]) }
else {}
}
eprintln('${term.highlight_command('v help')}: provide only one help topic.')
eprintln('For usage information, use ${term.highlight_command('v help')}.')
exit(1)
}
fn rebuild(prefs &pref.Preferences) {
match prefs.backend {
.c {
$if no_bootstrapv ? {
// TODO: improve the bootstrapping with a split C backend here.
// C code generated by `VEXE=v cmd/tools/builders/c_builder -os cross -o c.c cmd/tools/builders/c_builder.v`
// is enough to bootstrap the C backend, and thus the rest, but currently bootstrapping relies on
// `v -os cross -o v.c cmd/v` having a functional C codegen inside instead.
util.launch_tool(prefs.is_verbose, 'builders/c_builder', os.args[1..])
}
builder.compile('build', prefs, cbuilder.compile_c)
}
.js_node, .js_freestanding, .js_browser {
util.launch_tool(prefs.is_verbose, 'builders/js_builder', os.args[1..])
}
.native {
util.launch_tool(prefs.is_verbose, 'builders/native_builder', os.args[1..])
}
.interpret {
util.launch_tool(prefs.is_verbose, 'builders/interpret_builder', os.args[1..])
}
.golang {
println('using Go WIP backend...')
util.launch_tool(prefs.is_verbose, 'builders/golang_builder', os.args[1..])
}
.wasm {
assert_wasm_backend_thirdparty()
util.launch_tool(prefs.is_verbose, 'builders/wasm_builder', os.args[1..])
}
}
}
fn assert_wasm_backend_thirdparty() {
vroot := os.dir(pref.vexe_path())
if !os.exists('${vroot}/cmd/tools/builders/wasm_builder')
&& !os.exists('${vroot}/thirdparty/binaryen') {
eprintln('The WebAssembly backend requires `binaryen`, an external library dependency')
eprintln('This can be installed with `./cmd/tools/install_binaryen.vsh`, to download prebuilt libraries for your platform')
exit(1)
}
}