v/cmd/tools/vpm/install.v

171 lines
4.4 KiB
V

module main
import os
import v.vmod
import v.help
enum InstallResult {
installed
failed
skipped
}
fn vpm_install(query []string) {
if settings.is_help {
help.print_and_exit('vpm')
}
mut modules := parse_query(if query.len == 0 {
if os.exists('./v.mod') {
// Case: `v install` was run in a directory of another V-module to install its dependencies
// - without additional module arguments.
println('Detected v.mod file inside the project directory. Using it...')
manifest := vmod.from_file('./v.mod') or { panic(err) }
if manifest.dependencies.len == 0 {
println('Nothing to install.')
exit(0)
}
manifest.dependencies
} else {
vpm_error('specify at least one module for installation.',
details: 'example: `v install publisher.package` or `v install https://github.com/owner/repository`'
)
exit(2)
}
} else {
query
})
installed_modules := get_installed_modules()
vpm_log(@FILE_LINE, @FN, 'Queried Modules: ${modules}')
vpm_log(@FILE_LINE, @FN, 'Installed modules: ${installed_modules}')
if installed_modules.len > 0 && settings.is_once {
num_to_install := modules.len
mut already_installed := []string{}
if modules.len > 0 {
mut i_deleted := []int{}
for i, m in modules {
if m.name in installed_modules {
already_installed << m.name
i_deleted << i
}
}
for i in i_deleted.reverse() {
modules.delete(i)
}
}
if already_installed.len > 0 {
verbose_println('Already installed modules: ${already_installed}')
if already_installed.len == num_to_install {
println('All modules are already installed.')
exit(0)
}
}
}
install_modules(modules)
}
fn install_modules(modules []Module) {
vpm_log(@FILE_LINE, @FN, 'modules: ${modules}')
mut errors := 0
for m in modules {
vpm_log(@FILE_LINE, @FN, 'module: ${m}')
match m.install() {
.installed {}
.failed {
errors++
continue
}
.skipped {
continue
}
}
if !m.is_external {
increment_module_download_count(m.name) or {
vpm_error('failed to increment the download count for `${m.name}`',
details: err.msg()
)
errors++
}
}
println('Installed `${m.name}`.')
}
if errors > 0 {
exit(1)
}
}
fn (m Module) install() InstallResult {
defer {
os.rmdir_all(m.tmp_path) or {}
}
if m.is_installed {
// Case: installed, but not an explicit version. Update instead of continuing the installation.
if m.version == '' && m.installed_version == '' {
if m.is_external && m.url.starts_with('http://') {
vpm_update([m.install_path.all_after(settings.vmodules_path).trim_left(os.path_separator).replace(os.path_separator,
'.')])
} else {
vpm_update([m.name])
}
return .skipped
}
// Case: installed, but conflicting. Confirmation or -[-f]orce flag required.
if settings.is_force || m.confirm_install() {
m.remove() or {
vpm_error('failed to remove `${m.name}`.', details: err.msg())
return .failed
}
} else {
return .skipped
}
}
println('Installing `${m.name}`...')
// When the module should be relocated into a subdirectory we need to make sure
// it exists to not run into permission errors.
parent_dir := m.install_path.all_before_last(os.path_separator)
if !os.exists(parent_dir) {
os.mkdir_all(parent_dir) or {
vpm_error('failed to create module directory for `${m.name}`.', details: err.msg())
return .failed
}
}
os.mv(m.tmp_path, m.install_path) or {
vpm_error('failed to install `${m.name}`.', details: err.msg())
return .failed
}
return .installed
}
fn (m Module) confirm_install() bool {
if m.installed_version == m.version {
println('Module `${m.name}${at_version(m.installed_version)}` is already installed, use --force to overwrite.')
return false
} else {
println('Module `${m.name}${at_version(m.installed_version)}` is already installed at `${m.install_path_fmted}`.')
if settings.fail_on_prompt {
vpm_error('VPM should not have entered a confirmation prompt.')
exit(1)
}
install_version := at_version(if m.version == '' { 'latest' } else { m.version })
input := os.input('Replace it with `${m.name}${install_version}`? [Y/n]: ')
match input.trim_space().to_lower() {
'', 'y' {
return true
}
else {
verbose_println('Skipping `${m.name}`.')
return false
}
}
}
}
fn (m Module) remove() ! {
verbose_println('Removing `${m.name}` from `${m.install_path_fmted}`...')
rmdir_all(m.install_path)!
verbose_println('Removed `${m.name}`.')
}