pinebook-tests/random-output/shell.go

248 lines
5.5 KiB
Go

package main
import "fmt"
import "log"
import "strings"
import "io"
import "time"
import "os"
import "os/exec"
import "bufio"
import "reflect"
import "github.com/davecgh/go-spew/spew"
/*
func exampleCommand(a string, b ...string) {
// cmd := exec.Command(range myargs)
// args := []string{"/tmp", "/"}
// cmd := exec.Command(args)
cmd := exec.Command(a, b...)
cmd.Stdin = strings.NewReader("foobar")
var out bytes.Buffer
cmd.Stdout = &out
err := cmd.Run()
if err != nil {
log.Fatal(err)
}
fmt.Println(out.String())
// fmt.Printf("in all caps: %q\n", out.String())
}
func format(mybut *ui.Button) {
log.Println("doButton() mybut =", reflect.ValueOf(mybut).Elem())
log.Println("format and mount here")
exampleCommand("ls", "/tmp")
ping()
}
*/
func ping(hostname string, count int) {
cmdName := "ping " + hostname
cmdArgs := strings.Fields(cmdName)
cmd := exec.Command(cmdArgs[0], cmdArgs[1:len(cmdArgs)]...)
stdout, _ := cmd.StdoutPipe()
cmd.Start()
oneByte := make([]byte, 100)
f := bufio.NewWriter(os.Stdout)
f.Write(oneByte)
f.Flush()
num := 1
for {
/*
stuff, err := stdout.Read(oneByte)
if err != nil {
log.Println("ERROR:", err.Error())
break
} else {
log.Println("stuff =", stuff)
break
}
*/
r := bufio.NewReader(stdout)
line, _, _ := r.ReadLine()
fmt.Println(string(line))
num = num + 1
if num > count {
os.Exit(0)
}
}
cmd.Wait()
}
// this runs a process, then captures both the stdout
// and stderr in raw byte (so ncurses-like things work)
// the does unbuffered writes to stdout by running
// a flush() on each write to stdout
func shellRAW(hostname string, count int) {
cmdName := "ping " + hostname
cmdArgs := strings.Fields(cmdName)
cmd := exec.Command(cmdArgs[0], cmdArgs[1:len(cmdArgs)]...)
// merge stdout and stderr into a combined buffer
stdout, _ := cmd.StdoutPipe()
stderr, _ := cmd.StderrPipe()
merged := io.MultiReader(stderr, stdout)
scanner := bufio.NewScanner(merged)
cmd.Start()
// oneByte := make([]byte, 100)
num := 1
for {
// stuff := scanner.Bytes()
stuff := scanner.Text()
if stuff == "" {
// log.Println("START ", stuff)
// fmt.Printf(err.Error())
/*
f := bufio.NewWriter(os.Stdout)
f.Write(stuff)
f.Flush()
*/
// break
} else {
log.Println("START is nil", stuff)
}
/*
r := bufio.NewReader(stdout)
line, _, _ := r.ReadLine()
fmt.Println(string(line))
*/
num = num + 1
/*
if num > count {
os.Exit(0)
}
*/
}
cmd.Wait()
}
func script(cmds string) int {
// split on new lines (while we are at it, handle stupid windows text files
lines := strings.Split(strings.Replace(cmds, "\r\n", "\n", -1), "\n")
for _, line := range lines {
line = strings.TrimSpace(line) // this is like 'chomp' in perl
fmt.Println("LINE:", line)
time.Sleep(1)
// simpleProcess("ls /")
simpleProcess(line)
}
return 0
}
// Spawns a process and captures stdout and stderr
// this has to internally handle 'cd' and call os.Chdir()
func simpleProcess(cmd string) int {
log.Println("START " + cmd)
// cmd := exec.Command("ls", "/tmp", "/ballon", "/")
cmd = strings.TrimSpace(cmd) // this is like 'chomp' in perl
cmdArgs := strings.Fields(cmd)
if (len(cmdArgs) == 0) {
log.Println("END " + cmd)
return 0 // nothing to do
}
if (cmdArgs[0] == "cd") {
if (len(cmdArgs) > 1) {
log.Println("os.Chdir()", cmd)
os.Chdir(cmdArgs[1])
}
log.Println("END " + cmd)
return 0 // nothing to do
}
process := exec.Command(cmdArgs[0], cmdArgs[1:len(cmdArgs)]...)
stdout, _ := process.StdoutPipe()
stderr, _ := process.StderrPipe()
process.Start()
pid := process.Process.Pid
log.Println("pid =", pid)
log.Println(reflect.ValueOf(process.Process).Elem())
log.Println(reflect.TypeOf(process.Process))
merged := io.MultiReader(stderr, stdout)
scanner := bufio.NewScanner(merged)
for scanner.Scan() {
line := scanner.Text()
fmt.Println("NOT " + line) // Println will add back the final '\n'
}
if err := scanner.Err(); err != nil {
// fmt.Fprintln(os.Stderr, "reading standard input:", err)
fmt.Println("reading standard input:", err)
}
// One must wait for the process to formally finish and flush all buffer output
err := process.Wait()
// try to process the return value from the OS. As of golang 1.12,
// this should be cross platform (linux,macos,windows)
if err != nil {
spew.Dump(err.(*exec.ExitError))
spew.Dump(process.ProcessState)
stuff := err.(*exec.ExitError)
log.Println("ERROR ", stuff)
log.Println("END ", cmd)
return -1
}
log.Println("END " + cmd)
return 0
}
func exampleScanner() {
// cmd := exec.Command("ls", "/tmp", "/ballon", "/")
cmd := exec.Command("ls", "/ballon")
stdout, _ := cmd.StdoutPipe()
cmd.Start()
scanner := bufio.NewScanner(stdout)
for scanner.Scan() {
line := scanner.Text()
fmt.Println("NOT " + line) // Println will add back the final '\n'
}
if err := scanner.Err(); err != nil {
// fmt.Fprintln(os.Stderr, "reading standard input:", err)
fmt.Println("reading standard input:", err)
}
}
// prompt = whatever you would type at the command line "ls /tmp"
// timeout = number of lines to read in until exit
func stdoutExec(prompt string, timeout int) {
cmdArgs := strings.Fields(prompt)
cmd := exec.Command(cmdArgs[0], cmdArgs[1:len(cmdArgs)]...)
stdout, _ := cmd.StdoutPipe()
cmd.Start()
// oneByte := make([]byte, 100)
num := 1
for {
/*
_, err := stdout.Read(oneByte)
if err != nil {
fmt.Printf(err.Error())
break
}
*/
r := bufio.NewReader(stdout)
line, _, _ := r.ReadLine()
fmt.Println(string(line))
num = num + 1
if num > timeout {
os.Exit(0)
}
}
cmd.Wait()
}