v/vlib/compiler/modules.v

187 lines
5.0 KiB
V
Raw Normal View History

2019-07-21 17:53:35 +02:00
// Copyright (c) 2019 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 compiler
2019-07-21 17:53:35 +02:00
2019-09-01 21:51:16 +02:00
import os
import filepath
2019-08-04 00:03:52 +02:00
2019-10-30 15:07:41 +01:00
pub const (
v_modules_path = os.home_dir() + '.vmodules'
2019-10-04 14:48:09 +02:00
)
2019-10-25 15:34:12 +02:00
// Holds import information scoped to the parsed file
struct ImportTable {
mut:
imports map[string]string // alias => module
used_imports []string // alias
import_tok_idx map[string]int // module => idx
}
// Once we have a module format we can read from module file instead
// this is not optimal
fn (table &Table) qualify_module(mod string, file_path string) string {
for m in table.imports {
if m.contains('.') && m.contains(mod) {
m_parts := m.split('.')
m_path := m_parts.join(os.path_separator)
if mod == m_parts[m_parts.len-1] && file_path.contains(m_path) {
return m
}
}
}
return mod
}
fn new_import_table() ImportTable {
return ImportTable{
imports: map[string]string
}
}
fn (p mut Parser) register_import(mod string, tok_idx int) {
p.register_import_alias(mod, mod, tok_idx)
}
fn (p mut Parser) register_import_alias(alias string, mod string, tok_idx int) {
// NOTE: come back here
// if alias in it.imports && it.imports[alias] == mod {}
if alias in p.import_table.imports && p.import_table.imports[alias] != mod {
p.error('cannot import $mod as $alias: import name $alias already in use')
2019-10-25 15:34:12 +02:00
}
if mod.contains('.internal.') && !p.is_vgen {
2019-10-25 15:34:12 +02:00
mod_parts := mod.split('.')
mut internal_mod_parts := []string
for part in mod_parts {
if part == 'internal' { break }
internal_mod_parts << part
}
internal_parent := internal_mod_parts.join('.')
if !p.mod.starts_with(internal_parent) {
p.error('module $mod can only be imported internally by libs')
}
}
p.import_table.imports[alias] = mod
p.import_table.import_tok_idx[mod] = tok_idx
}
fn (it &ImportTable) get_import_tok_idx(mod string) int {
return it.import_tok_idx[mod]
}
fn (it &ImportTable) known_import(mod string) bool {
return mod in it.imports || it.is_aliased(mod)
}
fn (it &ImportTable) known_alias(alias string) bool {
return alias in it.imports
}
fn (it &ImportTable) is_aliased(mod string) bool {
for _, val in it.imports {
if val == mod {
return true
}
}
return false
}
fn (it &ImportTable) resolve_alias(alias string) string {
return it.imports[alias]
}
fn (it mut ImportTable) register_used_import(alias string) {
if !(alias in it.used_imports) {
it.used_imports << alias
}
}
fn (it &ImportTable) is_used_import(alias string) bool {
return alias in it.used_imports
}
// should module be accessable
pub fn (p &Parser) is_mod_in_scope(mod string) bool {
mut mods_in_scope := ['', 'builtin', 'main', p.mod]
for _, m in p.import_table.imports {
mods_in_scope << m
}
return mod in mods_in_scope
}
2019-10-25 15:34:12 +02:00
// return resolved dep graph (order deps)
pub fn (v &V) resolve_deps() &DepGraph {
graph := v.import_graph()
deps_resolved := graph.resolve()
if !deps_resolved.acyclic {
verror('import cycle detected between the following modules: \n' + deps_resolved.display_cycles())
}
return deps_resolved
}
// graph of all imported modules
pub fn(v &V) import_graph() &DepGraph {
mut graph := new_dep_graph()
for p in v.parsers {
2019-07-21 17:53:35 +02:00
mut deps := []string
2019-10-25 15:34:12 +02:00
for _, m in p.import_table.imports {
2019-07-21 17:53:35 +02:00
deps << m
}
2019-10-25 15:34:12 +02:00
graph.add(p.mod, deps)
2019-07-21 17:53:35 +02:00
}
2019-10-25 15:34:12 +02:00
return graph
2019-07-21 17:53:35 +02:00
}
// get ordered imports (module speficic dag method)
pub fn(graph &DepGraph) imports() []string {
2019-07-21 17:53:35 +02:00
mut mods := []string
for node in graph.nodes {
mods << node.name
}
return mods
}
[inline] fn (v &V) module_path(mod string) string {
2019-10-04 14:48:09 +02:00
// submodule support
return mod.replace('.', os.path_separator)
2019-10-04 14:48:09 +02:00
}
2019-09-01 21:51:16 +02:00
// 'strings' => 'VROOT/vlib/strings'
// 'installed_mod' => '~/.vmodules/installed_mod'
// 'local_mod' => '/path/to/current/dir/local_mod'
fn (v &V) find_module_path(mod string) ?string {
// Module search order:
// 1) search in the *same* directory, as the compiled final v program source
// (i.e. the . in `v .` or file.v in `v file.v`)
// 2) search in the modules/ in the same directory.
// 3) search in vlib/
// 4.1) search in -vpath (if given)
// 4.2) search in ~/.vmodules/ (i.e. modules installed with vpm) (no -vpath)
modules_lookup_path := if v.pref.vpath.len > 0 { v.pref.vpath } else { v_modules_path }
2019-08-04 00:03:52 +02:00
mod_path := v.module_path(mod)
mut tried_paths := []string
tried_paths << filepath.join(v.compiled_dir, mod_path)
tried_paths << filepath.join(v.compiled_dir, 'modules', mod_path)
tried_paths << filepath.join(v.pref.vlib_path, mod_path)
tried_paths << filepath.join(modules_lookup_path, mod_path)
if v.pref.user_mod_path.len > 0 {
tried_paths << v.pref.user_mod_path
}
for try_path in tried_paths {
if v.pref.is_verbose { println(' >> trying to find $mod in $try_path ...') }
if os.dir_exists(try_path) {
return try_path
2019-09-01 21:51:16 +02:00
}
2019-08-04 00:03:52 +02:00
}
return error('module "$mod" not found')
2019-09-01 21:51:16 +02:00
}
2019-10-13 02:05:11 +02:00
[inline] fn mod_gen_name(mod string) string {
return mod.replace('.', '_dot_')
}
2019-10-13 02:05:11 +02:00
[inline] fn mod_gen_name_rev(mod string) string {
return mod.replace('_dot_', '.')
2019-10-12 21:31:05 +02:00
}