vss/commands/build.v

185 lines
4.3 KiB
V

module commands
import os
import cli
import log
import time
import regex
import markdown
import internal.template
import internal.config
const default_config = 'config.toml'
const default_template = 'layouts/index.html'
const defautl_static = 'static'
const default_index = 'index.md'
const default_dist = 'dist'
struct Builder {
mut:
config config.Config
logger log.Log
dist string
static_dir string
template_content string
config_map map[string]string
}
fn new_builder(logger log.Log) Builder {
return Builder{
logger: logger
dist: commands.default_dist
static_dir: commands.defautl_static
}
}
fn new_build_cmd() cli.Command {
return cli.Command{
name: 'build'
description: 'build your site'
usage: 'vss build'
execute: fn (cmd cli.Command) ! {
mut logger := log.Log{}
logger.set_level(log.Level.info)
conf := load_config(commands.default_config)!
build(conf, mut logger) or {
logger.error(err.msg())
println('Build failed')
}
}
}
}
fn get_html_path(md_path string) string {
mut file_name := os.file_name(md_path)
file_name = file_name.replace('.md', '.html')
dir := os.dir(md_path)
if dir == '.' {
return file_name
}
return os.join_path(dir, file_name)
}
fn normalise_paths(paths []string) []string {
cwd := os.getwd() + os.path_separator
mut res := paths.map(os.abs_path(it).replace(cwd, '').replace(os.path_separator, '/'))
res.sort()
return res
}
fn get_md_content(path string) !string {
return os.read_file(path)!
}
fn get_content(path string) !string {
md := get_md_content(path)!
return markdown.to_html(md)
}
fn check_layout(path string) bool {
// check if layout is specified in front matter
// if not, use default layout
// if specified, check if layout file exists
// if not, return error
return true
}
fn (mut b Builder) md2html(md_path string) ! {
// get html body content from md
b.logger.info('start md to html: ${md_path}')
content := get_content(md_path)!
// want to change from contents to content
b.config_map['contents'] = content
// parse template
html_path := get_html_path(md_path)
dir := os.dir(md_path)
mut template_content := ''
if os.exists('layouts/${html_path}') {
b.logger.info('use custom template: layouts/${html_path}')
template_content = os.read_file('layouts/${html_path}')!
} else if os.exists('layouts/${dir}/index.html') {
b.logger.info('use custom template: layouts/${dir}/index.html')
template_content = os.read_file('layouts/${dir}/index.html')!
} else {
b.logger.info('use default template')
template_content = b.template_content
}
html := template.parse(template_content, b.config_map)
dist_path := os.join_path(b.dist, html_path)
if !os.exists(os.dir(dist_path)) {
os.mkdir_all(os.dir(dist_path))!
}
os.write_file(dist_path, html)!
}
// load_config loads a toml config file
fn load_config(toml_file string) !config.Config {
toml_text := os.read_file(toml_file)!
return config.load(toml_text)
}
// copy_static copy static files to dist
fn (b Builder) copy_static() ! {
if os.exists(b.static_dir) {
os.cp_all(b.static_dir, b.dist, false)!
}
}
// create_dist_dir create build output destination
fn (mut b Builder) create_dist_dir() ! {
if os.exists(b.dist) {
b.logger.info('re-create dist dir')
os.rmdir_all(b.dist)!
os.mkdir_all(b.dist)!
} else {
b.logger.info('create dist dir')
os.mkdir_all(b.dist)!
}
}
fn (mut b Builder) is_ignore(path string) bool {
// e.g. README.md
file_name := os.file_name(path)
// notify user that build was skipped
if file_name in b.config.build.ignore_files {
return true
}
return false
}
fn build(conf config.Config, mut logger log.Log) ! {
println('Start building')
mut sw := time.new_stopwatch()
mut b := new_builder(logger)
template_content := os.read_file(commands.default_template)!
b.template_content = template_content
b.config = conf
b.config_map = conf.as_map()
b.create_dist_dir()!
// copy static dir files
logger.info('copy static files')
b.copy_static()!
mds := normalise_paths(os.walk_ext('.', '.md'))
logger.info('start md to html')
for path in mds {
if b.is_ignore(path) {
logger.info('${path} is included in ignore_files, skip build')
continue
}
b.md2html(path)!
}
logger.info('end md to html')
sw.stop()
println('Total in ' + sw.elapsed().milliseconds().str() + ' ms')
return
}