// Copyright (c) 2019-2022 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 // Note: this program follows a similar convention to Rust: `init` makes the // structure of the program in the _current_ directory, while `new` // makes the program structure in a _sub_ directory. Besides that, the // functionality is essentially the same. // Note: here are the currently supported invokations so far: // 1) `v init` -> create a new project in the current folder // 2) `v new abc` -> create a new project in the new folder `abc`, by default a "hello world" project. // 3) `v new abcd web` -> create a new project in the new folder `abcd`, using the vweb template. // 4) `v new abcde gg` -> create a new project in the new folder `abcde`, using the gg template. // Note: run `v cmd/tools/vcreate_test.v` after changes to this program, to avoid regressions. struct Create { mut: name string description string version string license string } fn cerror(e string) { eprintln('\nerror: ${e}') } fn check_name(name string) string { if name.trim_space().len == 0 { cerror('project name cannot be empty') exit(1) } if name.is_title() { mut cname := name.to_lower() if cname.contains(' ') { cname = cname.replace(' ', '_') } eprintln('warning: the project name cannot be capitalized, the name will be changed to `${cname}`') return cname } if name.contains(' ') { cname := name.replace(' ', '_') eprintln('warning: the project name cannot contain spaces, the name will be changed to `${cname}`') return cname } return name } fn vmod_content(c Create) string { return "Module { name: '${c.name}' description: '${c.description}' version: '${c.version}' license: '${c.license}' dependencies: [] } " } fn new_project_content() string { if os.args.len == 2 && os.args[1] == 'init' { return main_content() } if os.args.len == 3 { return main_content() } if os.args.len == 4 { kind := os.args.last() return match kind { 'web' { simple_web_app() } 'gg' { main_content() // TODO } else { '' } } } return '' } fn main_content() string { return "module main fn main() { println('Hello World!') } " } fn gen_gitignore(name string) string { return '# Binaries for programs and plugins main ${name} *.exe *.exe~ *.so *.dylib *.dll # Ignore binary output folders bin/ # Ignore common editor/system specific metadata .DS_Store .idea/ .vscode/ *.iml ' } fn gitattributes_content() string { return '* text=auto eol=lf *.bat eol=crlf **/*.v linguist-language=V **/*.vv linguist-language=V **/*.vsh linguist-language=V **/v.mod linguist-language=V ' } fn editorconfig_content() string { return '[*] charset = utf-8 end_of_line = lf insert_final_newline = true trim_trailing_whitespace = true [*.v] indent_style = tab indent_size = 4 ' } fn (c &Create) write_vmod(new bool) { vmod_path := if new { '${c.name}/v.mod' } else { 'v.mod' } os.write_file(vmod_path, vmod_content(c)) or { panic(err) } } fn (c &Create) write_main(new bool) { if !new && (os.exists('${c.name}.v') || os.exists('src/${c.name}.v')) { return } main_path := if new { '${c.name}/${c.name}.v' } else { '${c.name}.v' } os.write_file(main_path, new_project_content()) or { panic(err) } } fn (c &Create) write_gitattributes(new bool) { gitattributes_path := if new { '${c.name}/.gitattributes' } else { '.gitattributes' } if !new && os.exists(gitattributes_path) { return } os.write_file(gitattributes_path, gitattributes_content()) or { panic(err) } } fn (c &Create) write_editorconfig(new bool) { editorconfig_path := if new { '${c.name}/.editorconfig' } else { '.editorconfig' } if !new && os.exists(editorconfig_path) { return } os.write_file(editorconfig_path, editorconfig_content()) or { panic(err) } } fn (c &Create) create_git_repo(dir string) { // Create Git Repo and .gitignore file if !os.is_dir('${dir}/.git') { res := os.execute('git init ${dir}') if res.exit_code != 0 { cerror('Unable to create git repo') exit(4) } } gitignore_path := '${dir}/.gitignore' if !os.exists(gitignore_path) { os.write_file(gitignore_path, gen_gitignore(c.name)) or {} } } fn create(args []string) { if os.args.len == 4 { template := os.args.last() if template !in ['web', 'gg'] { eprintln('uknown template "${template}", possible templates: web, gg') exit(1) } } mut c := Create{} c.name = check_name(if args.len > 0 { args[0] } else { os.input('Input your project name: ') }) if c.name == '' { cerror('project name cannot be empty') exit(1) } if c.name.contains('-') { cerror('"${c.name}" should not contain hyphens') exit(1) } if os.is_dir(c.name) { cerror('${c.name} folder already exists') exit(3) } c.description = if args.len > 1 { args[1] } else { os.input('Input your project description: ') } default_version := '0.0.0' c.version = os.input('Input your project version: (${default_version}) ') if c.version == '' { c.version = default_version } default_license := os.getenv_opt('VLICENSE') or { 'MIT' } c.license = os.input('Input your project license: (${default_license}) ') if c.license == '' { c.license = default_license } println('Initialising ...') os.mkdir(c.name) or { panic(err) } c.write_vmod(true) c.write_main(true) c.write_gitattributes(true) c.write_editorconfig(true) c.create_git_repo(c.name) } fn init_project() { mut c := Create{} c.name = check_name(os.file_name(os.getwd())) if !os.exists('v.mod') { c.description = '' c.write_vmod(false) println('Change the description of your project in `v.mod`') } c.write_main(false) c.write_gitattributes(false) c.write_editorconfig(false) c.create_git_repo('.') } fn main() { cmd := os.args[1] match cmd { 'new' { create(os.args[2..]) } 'init' { init_project() } else { cerror('unknown command: ${cmd}') exit(1) } } println('Complete!') } fn simple_web_app() string { return "import vweb import sqlite // can change to 'mysql', 'pg' const ( port = 8082 ) struct App { vweb.Context mut: db shared sqlite.DB } pub fn (app App) before_request() { println('[web] before_request: \${app.req.method} \${app.req.url}') } fn main() { vweb.run(&App{ db: sqlite.connect('vweb.sql')! }, port) } ['/users/:name'] pub fn (mut app App) user(name string) vweb.Result { user := sql app.db { select from User where name == name } return \$vweb.html() } ['/api/users/:name'] pub fn (mut app App) user(name string) vweb.Result { user := sql app.db { select from User where name == name } return app.json({ user: id }) } pub fn (mut app App) index() vweb.Result { return \$vweb.html() } [post] ['/register'] pub fn (mut app App) register_user(name string, password string) vweb.Result { user := User{name:name, password, password} sql app.db { insert user into User } return app.redirect('/') } " }