v0.0.72
This commit is contained in:
parent
72d6b538f7
commit
ff08d5f180
@ -10,6 +10,7 @@ type CommandRunner struct {
|
|||||||
args []string
|
args []string
|
||||||
timeout *time.Duration
|
timeout *time.Duration
|
||||||
env []string
|
env []string
|
||||||
|
listener []CommandListener
|
||||||
}
|
}
|
||||||
|
|
||||||
func Runner(program string) *CommandRunner {
|
func Runner(program string) *CommandRunner {
|
||||||
@ -18,6 +19,7 @@ func Runner(program string) *CommandRunner {
|
|||||||
args: make([]string, 0),
|
args: make([]string, 0),
|
||||||
timeout: nil,
|
timeout: nil,
|
||||||
env: make([]string, 0),
|
env: make([]string, 0),
|
||||||
|
listener: make([]CommandListener, 0),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -51,6 +53,21 @@ func (r *CommandRunner) Envs(env []string) *CommandRunner {
|
|||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *CommandRunner) Listen(lstr CommandListener) *CommandRunner {
|
||||||
|
r.listener = append(r.listener, lstr)
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *CommandRunner) ListenStdout(lstr func(string)) *CommandRunner {
|
||||||
|
r.listener = append(r.listener, genericCommandListener{_readStdoutLine: &lstr})
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *CommandRunner) ListenStderr(lstr func(string)) *CommandRunner {
|
||||||
|
r.listener = append(r.listener, genericCommandListener{_readStderrLine: &lstr})
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
func (r *CommandRunner) Run() (CommandResult, error) {
|
func (r *CommandRunner) Run() (CommandResult, error) {
|
||||||
return run(*r)
|
return run(*r)
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ package cmdext
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
|
"io"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@ -34,48 +35,114 @@ func run(opt CommandRunner) (CommandResult, error) {
|
|||||||
return CommandResult{}, err
|
return CommandResult{}, err
|
||||||
}
|
}
|
||||||
|
|
||||||
errch := make(chan error, 1)
|
errch := make(chan error, 3)
|
||||||
go func() { errch <- cmd.Wait() }()
|
go func() { errch <- cmd.Wait() }()
|
||||||
|
|
||||||
|
// [1] read raw stdout
|
||||||
|
|
||||||
|
stdoutBufferReader, stdoutBufferWriter := io.Pipe()
|
||||||
|
stdout := ""
|
||||||
|
go func() {
|
||||||
|
buf := make([]byte, 128)
|
||||||
|
for true {
|
||||||
|
n, out := stdoutPipe.Read(buf)
|
||||||
|
|
||||||
|
if n > 0 {
|
||||||
|
txt := string(buf[:n])
|
||||||
|
stdout += txt
|
||||||
|
_, _ = stdoutBufferWriter.Write(buf[:n])
|
||||||
|
for _, lstr := range opt.listener {
|
||||||
|
lstr.ReadRawStdout(buf[:n])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if out == io.EOF {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if out != nil {
|
||||||
|
errch <- out
|
||||||
|
_ = cmd.Process.Kill()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ = stdoutBufferWriter.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
// [2] read raw stderr
|
||||||
|
|
||||||
|
stderrBufferReader, stderrBufferWriter := io.Pipe()
|
||||||
|
stderr := ""
|
||||||
|
go func() {
|
||||||
|
buf := make([]byte, 128)
|
||||||
|
for true {
|
||||||
|
n, err := stderrPipe.Read(buf)
|
||||||
|
|
||||||
|
if n > 0 {
|
||||||
|
txt := string(buf[:n])
|
||||||
|
stderr += txt
|
||||||
|
_, _ = stderrBufferWriter.Write(buf[:n])
|
||||||
|
for _, lstr := range opt.listener {
|
||||||
|
lstr.ReadRawStderr(buf[:n])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err == io.EOF {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
errch <- err
|
||||||
|
_ = cmd.Process.Kill()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ = stderrBufferWriter.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
combch := make(chan string, 32)
|
combch := make(chan string, 32)
|
||||||
stopCombch := make(chan bool)
|
stopCombch := make(chan bool)
|
||||||
|
|
||||||
stdout := ""
|
// [3] collect stdout line-by-line
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
scanner := bufio.NewScanner(stdoutPipe)
|
scanner := bufio.NewScanner(stdoutBufferReader)
|
||||||
for scanner.Scan() {
|
for scanner.Scan() {
|
||||||
txt := scanner.Text()
|
txt := scanner.Text()
|
||||||
stdout += txt
|
for _, lstr := range opt.listener {
|
||||||
|
lstr.ReadStdoutLine(txt)
|
||||||
|
}
|
||||||
combch <- txt
|
combch <- txt
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
stderr := ""
|
// [4] collect stderr line-by-line
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
scanner := bufio.NewScanner(stderrPipe)
|
scanner := bufio.NewScanner(stderrBufferReader)
|
||||||
for scanner.Scan() {
|
for scanner.Scan() {
|
||||||
txt := scanner.Text()
|
txt := scanner.Text()
|
||||||
stderr += txt
|
for _, lstr := range opt.listener {
|
||||||
|
lstr.ReadStderrLine(txt)
|
||||||
|
}
|
||||||
combch <- txt
|
combch <- txt
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
defer func() {
|
defer func() { stopCombch <- true }()
|
||||||
stopCombch <- true
|
|
||||||
}()
|
// [5] combine stdcombined
|
||||||
|
|
||||||
stdcombined := ""
|
stdcombined := ""
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case txt := <-combch:
|
case txt := <-combch:
|
||||||
stdcombined += txt
|
stdcombined += txt + "\n" // this comes from bufio.Scanner and has no newlines...
|
||||||
case <-stopCombch:
|
case <-stopCombch:
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
// [6] run
|
||||||
|
|
||||||
var timeoutChan <-chan time.Time = make(chan time.Time, 1)
|
var timeoutChan <-chan time.Time = make(chan time.Time, 1)
|
||||||
if opt.timeout != nil {
|
if opt.timeout != nil {
|
||||||
timeoutChan = time.After(*opt.timeout)
|
timeoutChan = time.After(*opt.timeout)
|
||||||
@ -85,6 +152,9 @@ func run(opt CommandRunner) (CommandResult, error) {
|
|||||||
|
|
||||||
case <-timeoutChan:
|
case <-timeoutChan:
|
||||||
_ = cmd.Process.Kill()
|
_ = cmd.Process.Kill()
|
||||||
|
for _, lstr := range opt.listener {
|
||||||
|
lstr.Timeout()
|
||||||
|
}
|
||||||
return CommandResult{
|
return CommandResult{
|
||||||
StdOut: stdout,
|
StdOut: stdout,
|
||||||
StdErr: stderr,
|
StdErr: stderr,
|
||||||
@ -95,16 +165,23 @@ func run(opt CommandRunner) (CommandResult, error) {
|
|||||||
|
|
||||||
case err := <-errch:
|
case err := <-errch:
|
||||||
if exiterr, ok := err.(*exec.ExitError); ok {
|
if exiterr, ok := err.(*exec.ExitError); ok {
|
||||||
|
excode := exiterr.ExitCode()
|
||||||
|
for _, lstr := range opt.listener {
|
||||||
|
lstr.Finished(excode)
|
||||||
|
}
|
||||||
return CommandResult{
|
return CommandResult{
|
||||||
StdOut: stdout,
|
StdOut: stdout,
|
||||||
StdErr: stderr,
|
StdErr: stderr,
|
||||||
StdCombined: stdcombined,
|
StdCombined: stdcombined,
|
||||||
ExitCode: exiterr.ExitCode(),
|
ExitCode: excode,
|
||||||
CommandTimedOut: false,
|
CommandTimedOut: false,
|
||||||
}, nil
|
}, nil
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
return CommandResult{}, err
|
return CommandResult{}, err
|
||||||
} else {
|
} else {
|
||||||
|
for _, lstr := range opt.listener {
|
||||||
|
lstr.Finished(0)
|
||||||
|
}
|
||||||
return CommandResult{
|
return CommandResult{
|
||||||
StdOut: stdout,
|
StdOut: stdout,
|
||||||
StdErr: stderr,
|
StdErr: stderr,
|
||||||
|
59
cmdext/cmdrunner_test.go
Normal file
59
cmdext/cmdrunner_test.go
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
package cmdext
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestStdout(t *testing.T) {
|
||||||
|
|
||||||
|
res1, err := Runner("printf").Arg("hello").Run()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v", err)
|
||||||
|
}
|
||||||
|
if res1.StdErr != "" {
|
||||||
|
t.Errorf("res1.StdErr == '%v'", res1.StdErr)
|
||||||
|
}
|
||||||
|
if res1.StdOut != "hello" {
|
||||||
|
t.Errorf("res1.StdOut == '%v'", res1.StdOut)
|
||||||
|
}
|
||||||
|
if res1.StdCombined != "hello\n" {
|
||||||
|
t.Errorf("res1.StdCombined == '%v'", res1.StdCombined)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStderr(t *testing.T) {
|
||||||
|
|
||||||
|
res1, err := Runner("python").Arg("-c").Arg("import sys; print(\"error\", file=sys.stderr, end='')").Run()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v", err)
|
||||||
|
}
|
||||||
|
if res1.StdErr != "error" {
|
||||||
|
t.Errorf("res1.StdErr == '%v'", res1.StdErr)
|
||||||
|
}
|
||||||
|
if res1.StdOut != "" {
|
||||||
|
t.Errorf("res1.StdOut == '%v'", res1.StdOut)
|
||||||
|
}
|
||||||
|
if res1.StdCombined != "error\n" {
|
||||||
|
t.Errorf("res1.StdCombined == '%v'", res1.StdCombined)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStdcombined(t *testing.T) {
|
||||||
|
res1, err := Runner("python").
|
||||||
|
Arg("-c").
|
||||||
|
Arg("import sys; import time; print(\"1\", file=sys.stderr, flush=True); time.sleep(0.1); print(\"2\", file=sys.stdout, flush=True); time.sleep(0.1); print(\"3\", file=sys.stderr, flush=True)").
|
||||||
|
Run()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v", err)
|
||||||
|
}
|
||||||
|
if res1.StdErr != "1\n3\n" {
|
||||||
|
t.Errorf("res1.StdErr == '%v'", res1.StdErr)
|
||||||
|
}
|
||||||
|
if res1.StdOut != "2\n" {
|
||||||
|
t.Errorf("res1.StdOut == '%v'", res1.StdOut)
|
||||||
|
}
|
||||||
|
if res1.StdCombined != "1\n2\n3\n" {
|
||||||
|
t.Errorf("res1.StdCombined == '%v'", res1.StdCombined)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
57
cmdext/listener.go
Normal file
57
cmdext/listener.go
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
package cmdext
|
||||||
|
|
||||||
|
type CommandListener interface {
|
||||||
|
ReadRawStdout([]byte)
|
||||||
|
ReadRawStderr([]byte)
|
||||||
|
|
||||||
|
ReadStdoutLine(string)
|
||||||
|
ReadStderrLine(string)
|
||||||
|
|
||||||
|
Finished(int)
|
||||||
|
Timeout()
|
||||||
|
}
|
||||||
|
|
||||||
|
type genericCommandListener struct {
|
||||||
|
_readRawStdout *func([]byte)
|
||||||
|
_readRawStderr *func([]byte)
|
||||||
|
_readStdoutLine *func(string)
|
||||||
|
_readStderrLine *func(string)
|
||||||
|
_finished *func(int)
|
||||||
|
_timeout *func()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g genericCommandListener) ReadRawStdout(v []byte) {
|
||||||
|
if g._readRawStdout != nil {
|
||||||
|
(*g._readRawStdout)(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g genericCommandListener) ReadRawStderr(v []byte) {
|
||||||
|
if g._readRawStderr != nil {
|
||||||
|
(*g._readRawStderr)(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g genericCommandListener) ReadStdoutLine(v string) {
|
||||||
|
if g._readStdoutLine != nil {
|
||||||
|
(*g._readStdoutLine)(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g genericCommandListener) ReadStderrLine(v string) {
|
||||||
|
if g._readStderrLine != nil {
|
||||||
|
(*g._readStderrLine)(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g genericCommandListener) Finished(v int) {
|
||||||
|
if g._finished != nil {
|
||||||
|
(*g._finished)(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g genericCommandListener) Timeout() {
|
||||||
|
if g._timeout != nil {
|
||||||
|
(*g._timeout)()
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user