2016-08-13 12:49:04 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2016-10-29 23:20:35 +00:00
|
|
|
"net"
|
2016-08-13 12:49:04 +00:00
|
|
|
"os"
|
|
|
|
"os/exec"
|
2016-11-06 15:19:48 +00:00
|
|
|
"strconv"
|
2016-08-13 12:49:04 +00:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
type app struct {
|
|
|
|
ui *ui
|
|
|
|
nav *nav
|
2016-10-27 19:24:42 +00:00
|
|
|
quit chan bool
|
|
|
|
}
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
func newApp() *app {
|
2016-10-27 19:24:42 +00:00
|
|
|
ui := newUI()
|
|
|
|
nav := newNav(ui.wins[0].h)
|
2016-11-22 17:42:37 +00:00
|
|
|
quit := make(chan bool, 1)
|
2016-10-29 23:20:35 +00:00
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
return &app{
|
2016-10-27 19:24:42 +00:00
|
|
|
ui: ui,
|
|
|
|
nav: nav,
|
|
|
|
quit: quit,
|
|
|
|
}
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func waitKey() error {
|
|
|
|
// TODO: this should be done with termbox somehow
|
|
|
|
|
2016-08-28 13:58:24 +00:00
|
|
|
c := `echo
|
|
|
|
echo -n 'Press any key to continue'
|
|
|
|
old=$(stty -g)
|
|
|
|
stty raw -echo
|
|
|
|
eval "ignore=\$(dd bs=1 count=1 2> /dev/null)"
|
|
|
|
stty $old
|
|
|
|
echo`
|
|
|
|
|
|
|
|
cmd := exec.Command(gOpts.shell, "-c", c)
|
2016-08-13 12:49:04 +00:00
|
|
|
|
|
|
|
cmd.Stdin = os.Stdin
|
|
|
|
cmd.Stdout = os.Stdout
|
|
|
|
cmd.Stderr = os.Stderr
|
|
|
|
|
2016-08-17 20:28:42 +00:00
|
|
|
if err := cmd.Run(); err != nil {
|
2016-08-13 12:49:04 +00:00
|
|
|
return fmt.Errorf("waiting key: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-11-06 20:25:59 +00:00
|
|
|
// This is the main event loop of the application. There are two channels to
|
|
|
|
// read expressions from client and server. Reading and evaluation are done on
|
2016-12-15 09:26:06 +00:00
|
|
|
// separate goroutines.
|
2016-12-17 21:47:37 +00:00
|
|
|
func (app *app) handleInp() {
|
2016-12-15 09:26:06 +00:00
|
|
|
clientChan := app.ui.readExpr()
|
2016-10-29 23:20:35 +00:00
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
var serverChan chan expr
|
2016-11-06 14:08:16 +00:00
|
|
|
|
2016-10-29 23:20:35 +00:00
|
|
|
c, err := net.Dial("unix", gSocketPath)
|
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("connecting server: %s", err)
|
|
|
|
app.ui.message = msg
|
|
|
|
log.Printf(msg)
|
2016-11-06 14:08:16 +00:00
|
|
|
} else {
|
|
|
|
serverChan = readExpr(c)
|
2016-10-29 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
2016-08-13 12:49:04 +00:00
|
|
|
for {
|
2016-10-27 19:24:42 +00:00
|
|
|
select {
|
|
|
|
case <-app.quit:
|
2016-08-13 12:49:04 +00:00
|
|
|
log.Print("bye!")
|
2016-08-14 12:45:24 +00:00
|
|
|
|
|
|
|
if gLastDirPath != "" {
|
|
|
|
f, err := os.Create(gLastDirPath)
|
|
|
|
if err != nil {
|
2016-08-17 19:09:34 +00:00
|
|
|
log.Printf("opening last dir file: %s", err)
|
2016-08-14 12:45:24 +00:00
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
dir := app.nav.currDir()
|
|
|
|
|
|
|
|
_, err = f.WriteString(dir.path)
|
|
|
|
if err != nil {
|
2016-08-17 19:09:34 +00:00
|
|
|
log.Printf("writing last dir file: %s", err)
|
2016-08-14 12:45:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-13 12:49:04 +00:00
|
|
|
return
|
2016-10-29 23:20:35 +00:00
|
|
|
case e := <-clientChan:
|
|
|
|
for i := 0; i < e.count; i++ {
|
|
|
|
e.expr.eval(app, nil)
|
2016-10-27 19:24:42 +00:00
|
|
|
}
|
|
|
|
app.ui.draw(app.nav)
|
2016-10-29 23:20:35 +00:00
|
|
|
case e := <-serverChan:
|
|
|
|
e.eval(app, nil)
|
|
|
|
app.ui.draw(app.nav)
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
func (app *app) exportVars() {
|
2016-08-13 12:49:04 +00:00
|
|
|
var envFile string
|
2016-10-24 19:18:31 +00:00
|
|
|
if f, err := app.nav.currFile(); err == nil {
|
|
|
|
envFile = f.Path
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
marks := app.nav.currMarks()
|
|
|
|
|
|
|
|
envFiles := strings.Join(marks, ":")
|
|
|
|
|
|
|
|
os.Setenv("f", envFile)
|
|
|
|
os.Setenv("fs", envFiles)
|
|
|
|
|
|
|
|
if len(marks) == 0 {
|
|
|
|
os.Setenv("fx", envFile)
|
|
|
|
} else {
|
|
|
|
os.Setenv("fx", envFiles)
|
|
|
|
}
|
2016-11-06 15:19:48 +00:00
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
os.Setenv("id", strconv.Itoa(gClientID))
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This function is used to run a command in shell. Following modes are used:
|
|
|
|
//
|
2016-08-26 21:20:08 +00:00
|
|
|
// Prefix Wait Async Stdin/Stdout/Stderr UI action
|
|
|
|
// $ No No Yes Pause and then resume
|
|
|
|
// ! Yes No Yes Pause and then resume
|
2016-08-13 12:49:04 +00:00
|
|
|
// & No Yes No Do nothing
|
|
|
|
//
|
|
|
|
// Waiting async commands are not used for now.
|
2016-12-17 21:47:37 +00:00
|
|
|
func (app *app) runShell(s string, args []string, wait bool, async bool) {
|
2016-08-13 12:49:04 +00:00
|
|
|
app.exportVars()
|
|
|
|
|
|
|
|
if len(gOpts.ifs) != 0 {
|
|
|
|
s = fmt.Sprintf("IFS='%s'; %s", gOpts.ifs, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
args = append([]string{"-c", s, "--"}, args...)
|
2016-08-27 11:12:03 +00:00
|
|
|
cmd := exec.Command(gOpts.shell, args...)
|
2016-08-13 12:49:04 +00:00
|
|
|
|
|
|
|
if !async {
|
|
|
|
cmd.Stdin = os.Stdin
|
|
|
|
cmd.Stdout = os.Stdout
|
|
|
|
cmd.Stderr = os.Stderr
|
|
|
|
|
|
|
|
app.ui.pause()
|
|
|
|
defer app.ui.resume()
|
2016-08-26 21:20:08 +00:00
|
|
|
defer app.nav.renew(app.ui.wins[0].h)
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var err error
|
|
|
|
if async {
|
|
|
|
err = cmd.Start()
|
|
|
|
} else {
|
|
|
|
err = cmd.Run()
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("running shell: %s", err)
|
|
|
|
app.ui.message = msg
|
|
|
|
log.Print(msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
if wait {
|
|
|
|
if err := waitKey(); err != nil {
|
|
|
|
msg := fmt.Sprintf("waiting shell: %s", err)
|
|
|
|
app.ui.message = msg
|
|
|
|
log.Print(msg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|