package main import ( "bufio" "fmt" "os" _ "os/exec" "os/user" "syscall" "os/signal" "strings" "io" "context" lfs "hilbish/golibs/fs" cmds "hilbish/golibs/commander" "github.com/akamensky/argparse" "github.com/bobappleyard/readline" "github.com/yuin/gopher-lua" "layeh.com/gopher-luar" "mvdan.cc/sh/v3/interp" "mvdan.cc/sh/v3/syntax" ) const version = "0.2.0-dev" var l *lua.LState var prompt string var commands = map[string]bool{} var aliases = map[string]string{} func main() { parser := argparse.NewParser("hilbish", "A shell for lua and flower lovers") verflag := parser.Flag("v", "version", &argparse.Options{ Required: false, Help: "Prints Hilbish version", }) setshflag := parser.Flag("S", "set-shell-env", &argparse.Options{ Required: false, Help: "Sets $SHELL to Hilbish's executed path", }) err := parser.Parse(os.Args) if err != nil { fmt.Print(parser.Usage(err)) os.Exit(0) } if *verflag { fmt.Printf("Hilbish v%s\n", version) os.Exit(0) } if *setshflag { os.Setenv("SHELL", os.Args[0]) } input, err := os.ReadFile(".hilbishrc.lua") if err != nil { input, err = os.ReadFile("/usr/share/hilbish/.hilbishrc.lua") if err != nil { fmt.Println("could not find .hilbishrc.lua or /usr/share/hilbish/.hilbishrc.lua") return } } homedir, _ := os.UserHomeDir() if _, err := os.Stat(homedir + "/.hilbishrc.lua"); os.IsNotExist(err) { err = os.WriteFile(homedir + "/.hilbishrc.lua", input, 0644) if err != nil { fmt.Println("Error creating config file") fmt.Println(err) return } } HandleSignals() LuaInit() readline.Completer = readline.FilenameCompleter for { //dir, _ := os.Getwd() //reader := bufio.NewReader(os.Stdin) //fmt.Printf(prompt) cmdString, err := readline.String(fmtPrompt()) if err == io.EOF { fmt.Println("") break } if err != nil { fmt.Fprintln(os.Stderr, err) } cmdString = strings.TrimSuffix(cmdString, "\n") err = l.DoString(cmdString) if err == nil { readline.AddHistory(cmdString) continue } cmdArgs, cmdString := splitInput(cmdString) if len(cmdArgs) == 0 { continue } if aliases[cmdArgs[0]] != "" { cmdString = aliases[cmdArgs[0]] + strings.Trim(cmdString, cmdArgs[0]) //cmdArgs := splitInput(cmdString) execCommand(cmdString) continue } if commands[cmdArgs[0]] { err := l.CallByParam(lua.P{ Fn: l.GetField( l.GetTable( l.GetGlobal("commanding"), lua.LString("__commands")), cmdArgs[0]), NRet: 0, Protect: true, }, luar.New(l, cmdArgs[1:])) if err != nil { // TODO: dont panic panic(err) } readline.AddHistory(cmdString) continue } switch cmdArgs[0] { case "exit": os.Exit(0) default: err := execCommand(cmdString) if err != nil { if syntax.IsIncomplete(err) { sb := &strings.Builder{} for { done := StartMultiline(cmdString, sb) if done { break } } } else { fmt.Fprintln(os.Stderr, err) } } } } } func fmtPrompt() string { user, _ := user.Current() host, _ := os.Hostname() cwd, _ := os.Getwd() cwd = strings.Replace(cwd, user.HomeDir, "~", 1) args := []string{ "d", cwd, "h", host, "u", user.Name, } for i, v := range args { if i % 2 == 0 { args[i] = "%" + v } } r := strings.NewReplacer(args...) nprompt := r.Replace(prompt) return nprompt } func StartMultiline(prev string, sb *strings.Builder) bool { if sb.String() == "" { sb.WriteString(prev + "\n") } fmt.Printf("... ") reader := bufio.NewReader(os.Stdin) cont, err := reader.ReadString('\n') if err == io.EOF { fmt.Println("") return true } sb.WriteString(cont) err = execCommand(sb.String()) if err != nil && syntax.IsIncomplete(err) { return false } return true } func splitInput(input string) ([]string, string) { // end my suffering // TODO: refactor this garbage quoted := false startlastcmd := false lastcmddone := false cmdArgs := []string{} sb := &strings.Builder{} cmdstr := &strings.Builder{} lastcmd := readline.GetHistory(readline.HistorySize() - 1) for _, r := range input { if r == '"' { quoted = !quoted // dont add back quotes //sb.WriteRune(r) } else if !quoted && r == '~' { sb.WriteString(os.Getenv("HOME")) } else if !quoted && r == ' ' { cmdArgs = append(cmdArgs, sb.String()) sb.Reset() } else if !quoted && r == '^' && startlastcmd && !lastcmddone { cmdstr.WriteString(lastcmd) sb.WriteString(lastcmd) startlastcmd = !startlastcmd lastcmddone = !lastcmddone continue } else if !quoted && r == '^' && !lastcmddone { startlastcmd = !startlastcmd continue } else { sb.WriteRune(r) } cmdstr.WriteRune(r) } if sb.Len() > 0 { cmdArgs = append(cmdArgs, sb.String()) } readline.AddHistory(input) return cmdArgs, cmdstr.String() } func execCommand(cmd string) error { file, err := syntax.NewParser().Parse(strings.NewReader(cmd), "") if err != nil { return err } runner, _ := interp.New( interp.StdIO(os.Stdin, os.Stdout, os.Stderr), ) runner.Run(context.TODO(), file) return nil } func HandleSignals() { c := make(chan os.Signal) signal.Notify(c, os.Interrupt, syscall.SIGTERM) go func() { <-c }() } func LuaInit() { l = lua.NewState() l.OpenLibs() l.SetGlobal("prompt", l.NewFunction(hshprompt)) l.SetGlobal("alias", l.NewFunction(hshalias)) l.PreloadModule("fs", lfs.Loader) commander := cmds.New() commander.Events.On("commandRegister", func (cmdName string, cmd *lua.LFunction) { commands[cmdName] = true l.SetField( l.GetTable(l.GetGlobal("commanding"), lua.LString("__commands")), cmdName, cmd) }) l.PreloadModule("commander", commander.Loader) l.DoString("package.path = package.path .. ';./libs/?/init.lua;/usr/share/hilbish/libs/?/init.lua'") err := l.DoFile("/usr/share/hilbish/preload.lua") if err != nil { err = l.DoFile("preload.lua") if err != nil { fmt.Fprintln(os.Stderr, "Missing preload file, builtins may be missing.") } } homedir, _ := os.UserHomeDir() err = l.DoFile(homedir + "/.hilbishrc.lua") if err != nil { panic(err) } }