// Copyright (c) 2019-2020 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 term readline os.cmdline ) struct Repl { mut: indent int in_func bool line string lines []string temp_lines []string functions_name []string functions []string } fn (r mut Repl) checks() bool { mut in_string := false mut is_cut := false was_indent := r.indent > 0 for i := 0; i < r.line.len; i++ { if r.line[i] == `\'` && (i == 0 || r.line[i - 1] != `\\`) { in_string = !in_string } if r.line[i] == `{` && !in_string { r.line = r.line[..i + 1] + '\n' + r.line[i + 1..] is_cut = true i++ r.indent++ } if r.line[i] == `}` && !in_string { r.line = r.line[..i] + '\n' + r.line[i..] is_cut = true i++ r.indent-- if r.indent == 0 { r.in_func = false } } if i + 2 < r.line.len && r.indent == 0 && r.line[i + 1] == `f` && r.line[i + 2] == `n` { r.in_func = true } } return r.in_func || (was_indent && r.indent <= 0) || r.indent > 0 || is_cut } fn (r &Repl) function_call(line string) bool { for function in r.functions_name { if line.starts_with(function) { return true } } return false } pub fn repl_help() { version := v_version() println(version) println(' help Displays this information. Ctrl-C, Ctrl-D, exit Exits the REPL. clear Clears the screen. ') } pub fn run_repl(workdir string, vrepl_prefix string) []string { version := v_version() println(version) println('Use Ctrl-C or `exit` to exit') file := os.join( workdir, '.${vrepl_prefix}vrepl.v' ) temp_file := os.join( workdir, '.${vrepl_prefix}vrepl_temp.v') mut prompt := '>>> ' defer { println('') os.rm(file) os.rm(temp_file) os.rm(file[..file.len - 2]) os.rm(temp_file[..temp_file.len - 2]) } mut r := Repl{} mut readline := readline.Readline{} vexe := os.getenv('VEXE') for { if r.indent == 0 { prompt = '>>> ' } else { prompt = '... ' } mut line := readline.read_line(prompt) or { break } if line.trim_space() == '' && line.ends_with('\n') { continue } line = line.trim_space() if line.len <= -1 || line == '' || line == 'exit' { break } r.line = line if r.line == '\n' { continue } if r.line == 'clear' { term.erase_display('2') continue } if r.line == 'help' { repl_help() continue } if r.line.starts_with('fn') { r.in_func = true r.functions_name << r.line.all_after('fn').all_before('(').trim_space() } was_func := r.in_func if r.checks() { for rline in r.line.split('\n') { if r.in_func || was_func { r.functions << rline } else { r.temp_lines << rline } } if r.indent > 0 { continue } r.line = '' } // Save the source only if the user is printing something, // but don't add this print call to the `lines` array, // so that it doesn't get called during the next print. if r.line.starts_with('print') { source_code := r.functions.join('\n') + r.lines.join('\n') + '\n' + r.line os.write_file(file, source_code) s := os.exec('"$vexe" -repl run $file') or { rerror(err) return [] } print_output(s) } else { mut temp_line := r.line mut temp_flag := false func_call := r.function_call(r.line) if !( r.line.contains(' ') || r.line.contains(':') || r.line.contains('=') || r.line.contains(',') || r.line.ends_with('++') || r.line.ends_with('--') || r.line == '') && !func_call { temp_line = 'println($r.line)' temp_flag = true } temp_source_code := r.functions.join('\n') + r.lines.join('\n') + '\n' + r.temp_lines.join('\n') + '\n' + temp_line os.write_file(temp_file, temp_source_code) s := os.exec('"$vexe" -repl run $temp_file') or { println("SDFSDF") rerror(err) return [] } if !func_call && s.exit_code == 0 && !temp_flag { for r.temp_lines.len > 0 { if !r.temp_lines[0].starts_with('print') { r.lines << r.temp_lines[0] } r.temp_lines.delete(0) } r.lines << r.line } else { for r.temp_lines.len > 0 { r.temp_lines.delete(0) } } print_output(s) } } return r.lines } fn print_output(s os.Result) { lines := s.output.split('\n') for line in lines { if line.contains('.vrepl_temp.v:') { // Hide the temporary file name sline := line.all_after('.vrepl_temp.v:') idx := sline.index(' ') or { println(sline) return } println(sline[idx+1..]) } else if line.contains('.vrepl.v:') { // Ensure that .vrepl.v: is at the start, ignore the path // This is needed to have stable .repl tests. idx := line.index('.vrepl.v:') or { return } println(line[idx..]) } else { println(line) } } } fn main() { // Support for the parameters replfolder and replprefix is needed // so that the repl can be launched in parallel by several different // threads by the REPL test runner. args := cmdline.options_after(os.args, ['repl']) replfolder := os.realpath( cmdline.option(args, '-replfolder', '.') ) replprefix := cmdline.option(args, '-replprefix', 'noprefix.') os.chdir( replfolder ) if !os.exists(os.getenv('VEXE')) { println('Usage:') println(' VEXE=vexepath vrepl\n') println(' ... where vexepath is the full path to the v executable file') return } run_repl( replfolder, replprefix ) } pub fn rerror(s string) { println('V repl error: $s') os.flush() exit(1) } fn v_version() string { vexe := os.getenv('VEXE') vversion_res := os.exec('$vexe -version') or { panic('"$vexe -version" is not working') } return vversion_res.output }