forked from bitfield/script
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathscript.go
203 lines (165 loc) · 5.1 KB
/
script.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
package script
import (
"fmt"
"io"
"net/http"
"os"
"strings"
"github.com/bartdeboer/pipeline"
"github.com/bartdeboer/pipeline/std"
)
type Pipe struct {
std.Pipeline[*Pipe]
stdout io.Writer
httpClient *http.Client
}
func NewPipe() *Pipe {
p := &Pipe{
httpClient: http.DefaultClient,
}
p.Pipeline = std.NewPipeline(p)
p.WithStdout(os.Stdout)
p.SetCombinedOutput(true)
return p
}
// For backwards compatibility
func (p *Pipe) Filter(filter func(r io.Reader, w io.Writer) error) *Pipe {
b := pipeline.NewBaseProgram()
b.StartFn = func() error {
return filter(b.Stdin, b.Stdout)
}
return p.Pipe(b)
}
// For backwards compatibility
func (p *Pipe) FilterScan(filter func(string, io.Writer)) *Pipe {
return p.Scanner(filter)
}
func (p *Pipe) Stdout() (int, error) {
n64, err := p.Pipeline.Run()
n := int(n64)
if int64(n) != n64 {
return 0, fmt.Errorf("length %d overflows int", n64)
}
return n, err
}
// Sources:
// Args creates a pipeline with the command line arguments
func Args() *Pipe {
return Slice(os.Args[1:])
}
// Do creates a pipeline with an HTTP request
func Do(req *http.Request) *Pipe {
return NewPipe().Do(req)
}
// Echo creates a pipeline with the specified string
func Echo(s string) *Pipe {
return NewPipe().Echo(s)
}
// Exec creates a pipeline with the specified command using sh/shell
func Exec(name string, arg ...string) *Pipe {
return NewPipe().Exec(name, arg...)
}
// File creates a pipeline with the file contents
func File(path string) *Pipe {
return NewPipe().Pipe(std.File(path))
}
// FindFiles creates a pipeline with the files found in dir
func FindFiles(dir string) *Pipe {
return NewPipe().Pipe(std.FindFiles(dir))
}
// Do creates a pipeline with a GET HTTP request
func Get(url string) *Pipe {
return NewPipe().Get(url)
}
func IfExists(path string) *Pipe {
p := NewPipe()
p.Pipeline.SetExitOnError(true)
return p.Pipe(std.IfExists(path))
}
// ListFiles creates a pipeline with the file listing of path
func ListFiles(path string) *Pipe {
return NewPipe().Pipe(std.ListFiles(path))
}
// Do creates a pipeline with a POST HTTP request
func Post(url string) *Pipe {
return NewPipe().Post(url)
}
// Slice creates a pipeline with a new line for each slice item
func Slice(s []string) *Pipe {
return Echo(strings.Join(s, "\n") + "\n")
}
// Stdin creates a pipeline with stdin as input
func Stdin() *Pipe {
return NewPipe().Pipe(std.Stdin())
}
// Program shortcuts:
// AppendFile reads the input and appends it to the file path, creating it if necessary,
// and outputs the number of bytes successfully written
func (p *Pipe) AppendFile(path string) (int64, error) {
return p.Pipe(std.AppendFile(path)).Int64()
}
// CountLines returns the number of lines of input, or an error.
func (p *Pipe) CountLines() (int, error) {
return p.Pipe(std.CountLines()).Int()
}
// Get reads the input as the request body, sends the request and outputs the response
func (p *Pipe) Do(req *http.Request) *Pipe {
return p.Pipe(std.Do(req, p.httpClient))
}
// Exec executes cmdLine using sh/shell, using input as stdin and outputs the result
// func (p *Pipe) Exec(cmdLine string) *Pipe {
// return p.Pipe(shell.Exec(cmdLine))
// }
// ExecForEach renders cmdLine as a Go template for each line of input, running
// the resulting command, and outputs the combined result of these commands in sequence
// func (p *Pipe) ExecForEach(cmdLine string) *Pipe {
// return p.Pipe(shell.ExecForEach(cmdLine))
// }
// Get reads the input as the request body, sends a GET request and outputs the response
func (p *Pipe) Get(url string) *Pipe {
return p.Pipe(std.Get(url, p.httpClient))
}
// JQ reads the input (presumed to be JSON), executes the query and outputs the result
// func (p *Pipe) JQ(query string) *Pipe {
// return p.Pipe(gojq.JQ(query))
// }
// Get reads the input as the request body, sends a POST request and outputs the response
func (p *Pipe) Post(url string) *Pipe {
return p.Pipe(std.Post(url, p.httpClient))
}
// SHA256Sum reads the input and outputs the hex-encoded SHA-256 hash
func (p *Pipe) SHA256Sum() (string, error) {
return p.Pipe(std.SHA256Sum()).String()
}
// Tee reads the input and copies it to each of the supplied writers, like Unix tee(1)
func (p *Pipe) Tee(writers ...io.Writer) *Pipe {
if len(writers) == 0 {
p.Pipe(std.Tee(p.stdout)) // If no writers Tee with Pipe.stdout
}
return p.Pipe(std.Tee(writers...))
}
// WriteFile reads the input and writes it to the file path, truncating it if it exists,
// and outputs the number of bytes successfully written
func (p *Pipe) WriteFile(path string) (int64, error) {
return p.Pipe(std.WriteFile(path)).Int64()
}
// With* functions:
// WithHTTPClient sets the HTTP client c for use with subsequent requests
func (p *Pipe) WithHTTPClient(c *http.Client) *Pipe {
p.httpClient = c
return p
}
// WithStdout sets the pipe's standard output to the writer w
func (p *Pipe) WithStdout(w io.Writer) *Pipe {
p.stdout = w
p.Pipeline.WithStdout(w)
return p
}
func (p *Pipe) WithStderr(w io.Writer) *Pipe {
p.Pipeline.WithStderr(w)
p.Pipeline.SetCombinedOutput(false)
return p
}
func NewReadAutoCloser(r io.Reader) io.Reader {
return pipeline.NewReadOnlyPipe(r)
}