Documentation
¶
Overview ¶
Package executil provides utilities for working with os/exec.
Example ¶
package main
import (
"bytes"
"context"
"fmt"
"os"
"os/exec"
"path/filepath"
"regexp"
"runtime"
"sync"
"time"
"cloudeng.io/os/executil"
)
func main() {
go func() {
time.Sleep(time.Second * 5)
buf := make([]byte, 1024*1024)
n := runtime.Stack(buf, true)
fmt.Fprintf(os.Stderr, "%s\n", string(buf[:n]))
os.Exit(1)
}()
ctx := context.Background()
all := &bytes.Buffer{}
// Use go run testdata/cat.go for compatibility across windows and unix.
cmd := exec.CommandContext(ctx, "go", "run", filepath.Join("testdata", "cat.go"), filepath.Join("testdata", "example")) // #nosec G204
ch := make(chan []byte, 1)
filter := executil.NewLineFilter(all, ch, regexp.MustCompile("filter me:"))
cmd.Stdout = filter
var wg sync.WaitGroup
wg.Go(func() {
if err := cmd.Start(); err != nil {
panic(err)
}
})
buf := <-ch
fmt.Println("filtered output")
fmt.Println(string(buf))
fmt.Println("all of the output")
wg.Wait()
if err := filter.Close(); err != nil {
fmt.Println(err)
}
fmt.Println(all.String())
}
Output: filtered output filter me: 33 all of the output some words some more words filter me: 33 and again more words
Index ¶
- func ExecName(path string) string
- func Getenv(env []string, key string) (string, bool)
- func GoBuild(ctx context.Context, binary string, args ...string) (string, error)
- func IsStopped(pid int) bool
- func NewLabelingWriter(w io.Writer, prefix []byte, separator rune) io.Writer
- func NewLineFilter(forward io.Writer, ch chan<- []byte, res ...*regexp.Regexp) io.WriteCloser
- func ReplaceEnvVar(env []string, key, value string) []string
- func SignalAndWait(ctx context.Context, perSignalOrWait time.Duration, cmd *exec.Cmd, ...) error
- func WaitFor(ctx context.Context, interval time.Duration, ...) error
- func WaitForStopped(ctx context.Context, pid int, waitFor time.Duration) error
- type AsyncWait
- type LabelingWriter
- type TailWriter
Examples ¶
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
func ExecName ¶
ExecName returns path in a form suitable for use as an executable. For unix systems the path is unchanged. For windows a '.exe' suffix is added if not already present.
func IsStopped ¶
IsStopped returns true if the process with the specified pid has stopped or does not exist. Wait must have been called on the process otherwise this function will return true on some systems since the process may still exist as a defunct process.
func NewLabelingWriter ¶
func NewLineFilter ¶
NewLineFilter returns an io.WriteCloser that scans the contents of the supplied io.Writer and sends lines that match the regexp to the supplied channel. It can be used to filter the output of a command started by the exec package for example for specific output. If no regexps are supplied, all lines are sent. Close must be called on the returned io.WriteCloser to ensure that all resources are reclaimed.
func ReplaceEnvVar ¶
ReplaceEnvVar replaces the value of an environment variable in the provided slice. If the variable does not exist, it is added to the slice.
func SignalAndWait ¶
func SignalAndWait(ctx context.Context, perSignalOrWait time.Duration, cmd *exec.Cmd, sigs ...os.Signal) error
SignalAndWait provides a convenience function to signal a process to terminate by sending it one or more signals and waiting for it to terminate but with a timeout on calling Wait and on waiting for the process to stop after each signal. The perSignalOrWait duration is used as the timeout for both calling Wait and for waiting for the process to stop after each signal, hence the total time spent waiting may be up to len(sigs)+1 times perSignalOrWait. If the process stops after any signal, SignalAndWait returns immediately.
func WaitFor ¶
func WaitFor(ctx context.Context, interval time.Duration, check func(ctx context.Context) (done bool, err error)) error
WaitFor repeatedly calls the provided check function until it returns done=true, or the context is done. It waits for the specified interval between calls. If check returns an error, it is ignored unless done=true, in which case it is returned immediately.
Types ¶
type AsyncWait ¶
type AsyncWait struct {
// contains filtered or unexported fields
}
AsyncWait simplifies implementing asynchronous waiting for an exec.Cmd.
func NewAsyncWait ¶
NewAsyncWait creates a new AsyncWait for the given exec.Cmd. It immediately starts a goroutine to wait for the cmd to complete.
type LabelingWriter ¶
type LabelingWriter struct {
// contains filtered or unexported fields
}
LabelingWriter is an io.Writer that prepends prefix to the data written from the underlying writer. It prepends the prefix to the beginning of the stream and after every separator character. For example, it can be used to insert labels in the output of an exec.Cmd without modifying the command itself when working with multiple outstanding commands.
func (*LabelingWriter) Write ¶
func (pr *LabelingWriter) Write(p []byte) (n int, err error)
Write implements io.Writer. It writes the data to the underlying writer, inserting the prefix at the beginning of the stream and after every separator character. It returns the number of bytes from p that were written rather than the total number of bytes including the label.
type TailWriter ¶
type TailWriter struct {
// contains filtered or unexported fields
}
TailWriter is an io.Writer that keeps only the last N bytes written to it. It is useful for capturing the output of a command while limiting memory usage.
func NewTailWriter ¶
func NewTailWriter(n int) *TailWriter
NewTailWriter creates a new TailWriter that keeps the last n bytes.
func (*TailWriter) Bytes ¶
func (w *TailWriter) Bytes() []byte
Bytes returns the contents of the TailWriter.