gitee.com/yfmps/gopher-lua@v0.0.3/cmd/glua/glua.go (about)

     1  package main
     2  
     3  import (
     4  	"flag"
     5  	"fmt"
     6  	"github.com/chzyer/readline"
     7  	"github.com/yuin/gopher-lua"
     8  	"github.com/yuin/gopher-lua/parse"
     9  	"os"
    10  	"runtime/pprof"
    11  )
    12  
    13  func main() {
    14  	os.Exit(mainAux())
    15  }
    16  
    17  func mainAux() int {
    18  	var opt_e, opt_l, opt_p string
    19  	var opt_i, opt_v, opt_dt, opt_dc bool
    20  	var opt_m int
    21  	flag.StringVar(&opt_e, "e", "", "")
    22  	flag.StringVar(&opt_l, "l", "", "")
    23  	flag.StringVar(&opt_p, "p", "", "")
    24  	flag.IntVar(&opt_m, "mx", 0, "")
    25  	flag.BoolVar(&opt_i, "i", false, "")
    26  	flag.BoolVar(&opt_v, "v", false, "")
    27  	flag.BoolVar(&opt_dt, "dt", false, "")
    28  	flag.BoolVar(&opt_dc, "dc", false, "")
    29  	flag.Usage = func() {
    30  		fmt.Println(`Usage: glua [options] [script [args]].
    31  Available options are:
    32    -e stat  execute string 'stat'
    33    -l name  require library 'name'
    34    -mx MB   memory limit(default: unlimited)
    35    -dt      dump AST trees
    36    -dc      dump VM codes
    37    -i       enter interactive mode after executing 'script'
    38    -p file  write cpu profiles to the file
    39    -v       show version information`)
    40  	}
    41  	flag.Parse()
    42  	if len(opt_p) != 0 {
    43  		f, err := os.Create(opt_p)
    44  		if err != nil {
    45  			fmt.Println(err.Error())
    46  			os.Exit(1)
    47  		}
    48  		pprof.StartCPUProfile(f)
    49  		defer pprof.StopCPUProfile()
    50  	}
    51  	if len(opt_e) == 0 && !opt_i && !opt_v && flag.NArg() == 0 {
    52  		opt_i = true
    53  	}
    54  
    55  	status := 0
    56  
    57  	L := lua.NewState()
    58  	defer L.Close()
    59  	if opt_m > 0 {
    60  		L.SetMx(opt_m)
    61  	}
    62  
    63  	if opt_v || opt_i {
    64  		fmt.Println(lua.PackageCopyRight)
    65  	}
    66  
    67  	if len(opt_l) > 0 {
    68  		if err := L.DoFile(opt_l); err != nil {
    69  			fmt.Println(err.Error())
    70  		}
    71  	}
    72  
    73  	if nargs := flag.NArg(); nargs > 0 {
    74  		script := flag.Arg(0)
    75  		argtb := L.NewTable()
    76  		for i := 1; i < nargs; i++ {
    77  			L.RawSet(argtb, lua.LNumber(i), lua.LString(flag.Arg(i)))
    78  		}
    79  		L.SetGlobal("arg", argtb)
    80  		if opt_dt || opt_dc {
    81  			file, err := os.Open(script)
    82  			if err != nil {
    83  				fmt.Println(err.Error())
    84  				return 1
    85  			}
    86  			chunk, err2 := parse.Parse(file, script)
    87  			if err2 != nil {
    88  				fmt.Println(err2.Error())
    89  				return 1
    90  			}
    91  			if opt_dt {
    92  				fmt.Println(parse.Dump(chunk))
    93  			}
    94  			if opt_dc {
    95  				proto, err3 := lua.Compile(chunk, script)
    96  				if err3 != nil {
    97  					fmt.Println(err3.Error())
    98  					return 1
    99  				}
   100  				fmt.Println(proto.String())
   101  			}
   102  		}
   103  		if err := L.DoFile(script); err != nil {
   104  			fmt.Println(err.Error())
   105  			status = 1
   106  		}
   107  	}
   108  
   109  	if len(opt_e) > 0 {
   110  		if err := L.DoString(opt_e); err != nil {
   111  			fmt.Println(err.Error())
   112  			status = 1
   113  		}
   114  	}
   115  
   116  	if opt_i {
   117  		doREPL(L)
   118  	}
   119  	return status
   120  }
   121  
   122  // do read/eval/print/loop
   123  func doREPL(L *lua.LState) {
   124  	rl, err := readline.New("> ")
   125  	if err != nil {
   126  		panic(err)
   127  	}
   128  	defer rl.Close()
   129  	for {
   130  		if str, err := loadline(rl, L); err == nil {
   131  			if err := L.DoString(str); err != nil {
   132  				fmt.Println(err)
   133  			}
   134  		} else { // error on loadline
   135  			fmt.Println(err)
   136  			return
   137  		}
   138  	}
   139  }
   140  
   141  func incomplete(err error) bool {
   142  	if lerr, ok := err.(*lua.ApiError); ok {
   143  		if perr, ok := lerr.Cause.(*parse.Error); ok {
   144  			return perr.Pos.Line == parse.EOF
   145  		}
   146  	}
   147  	return false
   148  }
   149  
   150  func loadline(rl *readline.Instance, L *lua.LState) (string, error) {
   151  	rl.SetPrompt("> ")
   152  	if line, err := rl.Readline(); err == nil {
   153  		if _, err := L.LoadString("return " + line); err == nil { // try add return <...> then compile
   154  			return line, nil
   155  		} else {
   156  			return multiline(line, rl, L)
   157  		}
   158  	} else {
   159  		return "", err
   160  	}
   161  }
   162  
   163  func multiline(ml string, rl *readline.Instance, L *lua.LState) (string, error) {
   164  	for {
   165  		if _, err := L.LoadString(ml); err == nil { // try compile
   166  			return ml, nil
   167  		} else if !incomplete(err) { // syntax error , but not EOF
   168  			return ml, nil
   169  		} else {
   170  			rl.SetPrompt(">> ")
   171  			if line, err := rl.Readline(); err == nil {
   172  				ml = ml + "\n" + line
   173  			} else {
   174  				return "", err
   175  			}
   176  		}
   177  	}
   178  }