v/vlib/v/vmod/vmod.v

172 lines
4.5 KiB
V
Raw Normal View History

2020-04-10 12:01:06 +02:00
module vmod
2020-02-29 14:23:45 +01:00
import os
// This file provides a caching mechanism for seeking quickly whether a
// given folder has a v.mod file in it or in any of its parent folders.
//
// ModFileCacher.get(folder) works in such a way, that given this tree:
// examples/hanoi.v
// vlib/v.mod
// vlib/v/tests/project_with_c_code/mod1/v.mod
// vlib/v/tests/project_with_c_code/mod1/wrapper.v
2020-02-29 14:23:45 +01:00
// -----------------
// ModFileCacher.get('examples')
// => ModFileAndFolder{'', 'examples'}
// ModFileCacher.get('vlib/v/tests')
2020-02-29 14:23:45 +01:00
// => ModFileAndFolder{'vlib/v.mod', 'vlib'}
// ModFileCacher.get('vlib/v')
2020-02-29 14:23:45 +01:00
// => ModFileAndFolder{'vlib/v.mod', 'vlib'}
// ModFileCacher.get('vlib/v/test/project_with_c_code/mod1')
// => ModFileAndFolder{'vlib/v/test/project_with_c_code/mod1/v.mod', 'vlib/v/test/project_with_c_code/mod1'}
pub struct ModFileAndFolder {
pub:
2020-02-29 14:23:45 +01:00
// vmod_file contains the full path of the found 'v.mod' file, or ''
// if no 'v.mod' file was found in file_path_dir, or in its parent folders.
vmod_file string
2020-02-29 14:23:45 +01:00
// vmod_folder contains the file_path_dir, if there is no 'v.mod' file in
// *any* of the parent folders, otherwise it is the first parent folder,
// where a v.mod file was found.
vmod_folder string
}
[heap]
pub struct ModFileCacher {
2020-02-29 14:23:45 +01:00
mut:
cache map[string]ModFileAndFolder
2020-02-29 14:23:45 +01:00
// folder_files caches os.ls(key)
folder_files map[string][]string
}
pub fn new_mod_file_cacher() &ModFileCacher {
2020-02-29 14:23:45 +01:00
return &ModFileCacher{}
}
2021-03-06 18:09:28 +01:00
pub fn (mcache &ModFileCacher) debug() {
2020-02-29 14:23:45 +01:00
$if debug {
eprintln('ModFileCacher DUMP:')
eprintln(' ModFileCacher.cache:')
for k, v in mcache.cache {
2020-02-29 14:23:45 +01:00
eprintln(' K: ${k:-32s} | V: "${v.vmod_file:32s}" | "${v.vmod_folder:32s}" ')
}
eprintln(' ModFileCacher.folder_files:')
for k, v in mcache.folder_files {
eprintln(' K: ${k:-32s} | V: $v.str()')
2020-02-29 14:23:45 +01:00
}
}
}
pub fn (mut mcache ModFileCacher) get_by_file(vfile string) ModFileAndFolder {
return mcache.get_by_folder(os.dir(vfile))
}
pub fn (mut mcache ModFileCacher) get_by_folder(vfolder string) ModFileAndFolder {
mfolder := os.real_path(vfolder)
2020-02-29 14:23:45 +01:00
if mfolder in mcache.cache {
return mcache.cache[mfolder]
2020-02-29 14:23:45 +01:00
}
traversed_folders, res := mcache.traverse(mfolder)
2020-02-29 14:23:45 +01:00
for tfolder in traversed_folders {
mcache.add(tfolder, res)
2020-02-29 14:23:45 +01:00
}
return res
}
2020-05-17 13:51:18 +02:00
fn (mut cacher ModFileCacher) add(path string, result ModFileAndFolder) {
cacher.cache[path] = result
2020-02-29 14:23:45 +01:00
}
2020-05-17 13:51:18 +02:00
fn (mut mcache ModFileCacher) traverse(mfolder string) ([]string, ModFileAndFolder) {
2020-02-29 14:23:45 +01:00
mut cfolder := mfolder
mut folders_so_far := [cfolder]
mut levels := 0
for {
if levels > 255 {
break
}
if cfolder == '/' || cfolder == '' {
break
}
if cfolder in mcache.cache {
res := mcache.cache[cfolder]
2020-02-29 14:23:45 +01:00
if res.vmod_file.len == 0 {
mcache.mark_folders_as_vmod_free(folders_so_far)
} else {
mcache.mark_folders_with_vmod(folders_so_far, res)
2020-02-29 14:23:45 +01:00
}
return []string{}, res
2020-02-29 14:23:45 +01:00
}
files := mcache.get_files(cfolder)
2020-02-29 14:23:45 +01:00
if 'v.mod' in files {
// TODO: actually read the v.mod file and parse its contents to see
// if its source folder is different
res := ModFileAndFolder{
vmod_file: os.join_path(cfolder, 'v.mod')
vmod_folder: cfolder
}
2020-02-29 14:23:45 +01:00
return folders_so_far, res
}
if mcache.check_for_stop(cfolder, files) {
2020-02-29 14:23:45 +01:00
break
}
cfolder = os.dir(cfolder)
2020-02-29 14:23:45 +01:00
folders_so_far << cfolder
levels++
}
mcache.mark_folders_as_vmod_free(folders_so_far)
return [mfolder], ModFileAndFolder{
vmod_file: ''
vmod_folder: mfolder
}
2020-02-29 14:23:45 +01:00
}
fn (mut mcache ModFileCacher) mark_folders_with_vmod(folders_so_far []string, vmod ModFileAndFolder) {
2020-02-29 14:23:45 +01:00
for f in folders_so_far {
mcache.add(f, vmod)
2020-02-29 14:23:45 +01:00
}
}
fn (mut mcache ModFileCacher) mark_folders_as_vmod_free(folders_so_far []string) {
2020-02-29 14:23:45 +01:00
// No need to check these folders anymore,
// because their parents do not contain v.mod files
for f in folders_so_far {
mcache.add(f, vmod_file: '', vmod_folder: f)
2020-02-29 14:23:45 +01:00
}
}
const (
mod_file_stop_paths = ['.git', '.hg', '.svn', '.v.mod.stop']
)
2020-02-29 14:23:45 +01:00
fn (mcache &ModFileCacher) check_for_stop(cfolder string, files []string) bool {
2021-01-25 10:26:20 +01:00
for i in vmod.mod_file_stop_paths {
2020-02-29 14:23:45 +01:00
if i in files {
return true
}
}
return false
}
2020-05-17 13:51:18 +02:00
fn (mut mcache ModFileCacher) get_files(cfolder string) []string {
2020-02-29 14:23:45 +01:00
if cfolder in mcache.folder_files {
return mcache.folder_files[cfolder]
2020-02-29 14:23:45 +01:00
}
mut files := []string{}
if os.exists(cfolder) && os.is_dir(cfolder) {
2020-04-25 22:43:46 +02:00
if listing := os.ls(cfolder) {
2020-12-20 10:42:46 +01:00
files = listing.clone()
2020-04-25 22:43:46 +02:00
}
}
mcache.folder_files[cfolder] = files
2020-02-29 14:23:45 +01:00
return files
}
2020-04-10 12:01:06 +02:00
// used during lookup for v.mod to support @VROOT
const (
private_file_cacher = new_mod_file_cacher()
)
pub fn get_cache() &ModFileCacher {
2021-01-25 10:26:20 +01:00
return vmod.private_file_cacher
}