|
1 | 1 | package nix |
2 | 2 |
|
3 | | -import ( |
4 | | - "bytes" |
5 | | - "context" |
6 | | - "errors" |
7 | | - "fmt" |
8 | | - "io" |
9 | | - "log/slog" |
10 | | - "os" |
11 | | - "os/exec" |
12 | | - "slices" |
13 | | - "strconv" |
14 | | - "strings" |
15 | | - "syscall" |
16 | | - "time" |
17 | | -) |
18 | | - |
19 | | -type cmd struct { |
20 | | - Args cmdArgs |
21 | | - Env []string |
22 | | - |
23 | | - Stdin io.Reader |
24 | | - Stdout io.Writer |
25 | | - Stderr io.Writer |
26 | | - |
27 | | - execCmd *exec.Cmd |
28 | | - err error |
29 | | - dur time.Duration |
30 | | - logger *slog.Logger |
31 | | -} |
32 | | - |
33 | | -func command(args ...any) *cmd { |
34 | | - cmd := &cmd{ |
35 | | - Args: append(cmdArgs{ |
36 | | - "nix", |
37 | | - "--extra-experimental-features", "ca-derivations", |
38 | | - "--option", "experimental-features", "nix-command flakes fetch-closure", |
39 | | - }, args...), |
40 | | - logger: slog.Default(), |
41 | | - } |
42 | | - return cmd |
43 | | -} |
44 | | - |
45 | | -func (c *cmd) CombinedOutput(ctx context.Context) ([]byte, error) { |
46 | | - cmd := c.initExecCommand(ctx) |
47 | | - c.logger.DebugContext(ctx, "nix command starting", "cmd", c) |
48 | | - |
49 | | - start := time.Now() |
50 | | - out, err := cmd.CombinedOutput() |
51 | | - c.dur = time.Since(start) |
52 | | - |
53 | | - c.err = c.error(ctx, err) |
54 | | - c.logger.DebugContext(ctx, "nix command exited", "cmd", c) |
55 | | - return out, c.err |
56 | | -} |
57 | | - |
58 | | -func (c *cmd) Output(ctx context.Context) ([]byte, error) { |
59 | | - cmd := c.initExecCommand(ctx) |
60 | | - c.logger.DebugContext(ctx, "nix command starting", "cmd", c) |
61 | | - |
62 | | - start := time.Now() |
63 | | - out, err := cmd.Output() |
64 | | - c.dur = time.Since(start) |
65 | | - |
66 | | - c.err = c.error(ctx, err) |
67 | | - c.logger.DebugContext(ctx, "nix command exited", "cmd", c) |
68 | | - return out, c.err |
69 | | -} |
70 | | - |
71 | | -func (c *cmd) Run(ctx context.Context) error { |
72 | | - cmd := c.initExecCommand(ctx) |
73 | | - c.logger.DebugContext(ctx, "nix command starting", "cmd", c) |
74 | | - |
75 | | - start := time.Now() |
76 | | - err := cmd.Run() |
77 | | - c.dur = time.Since(start) |
78 | | - |
79 | | - c.err = c.error(ctx, err) |
80 | | - c.logger.DebugContext(ctx, "nix command exited", "cmd", c) |
81 | | - return c.err |
82 | | -} |
83 | | - |
84 | | -func (c *cmd) LogValue() slog.Value { |
85 | | - attrs := []slog.Attr{ |
86 | | - slog.Any("args", c.Args), |
87 | | - } |
88 | | - if c.execCmd == nil { |
89 | | - return slog.GroupValue(attrs...) |
90 | | - } |
91 | | - attrs = append(attrs, slog.String("path", c.execCmd.Path)) |
92 | | - |
93 | | - var exitErr *exec.ExitError |
94 | | - if errors.As(c.err, &exitErr) { |
95 | | - stderr := c.stderrExcerpt(exitErr.Stderr) |
96 | | - if len(stderr) != 0 { |
97 | | - attrs = append(attrs, slog.String("stderr", stderr)) |
98 | | - } |
99 | | - } |
100 | | - if proc := c.execCmd.Process; proc != nil { |
101 | | - attrs = append(attrs, slog.Int("pid", proc.Pid)) |
102 | | - } |
103 | | - if procState := c.execCmd.ProcessState; procState != nil { |
104 | | - if procState.Exited() { |
105 | | - attrs = append(attrs, slog.Int("code", procState.ExitCode())) |
106 | | - } |
107 | | - if status, ok := procState.Sys().(syscall.WaitStatus); ok && status.Signaled() { |
108 | | - if status.Signaled() { |
109 | | - attrs = append(attrs, slog.String("signal", status.Signal().String())) |
110 | | - } |
111 | | - } |
112 | | - } |
113 | | - if c.dur != 0 { |
114 | | - attrs = append(attrs, slog.Duration("dur", c.dur)) |
115 | | - } |
116 | | - return slog.GroupValue(attrs...) |
117 | | -} |
118 | | - |
119 | | -func (c *cmd) String() string { |
120 | | - return c.Args.String() |
121 | | -} |
122 | | - |
123 | | -func (c *cmd) initExecCommand(ctx context.Context) *exec.Cmd { |
124 | | - if c.execCmd != nil { |
125 | | - return c.execCmd |
126 | | - } |
127 | | - |
128 | | - args := c.Args.StringSlice() |
129 | | - c.execCmd = exec.CommandContext(ctx, args[0], args[1:]...) |
130 | | - c.execCmd.Env = c.Env |
131 | | - c.execCmd.Stdin = c.Stdin |
132 | | - c.execCmd.Stdout = c.Stdout |
133 | | - c.execCmd.Stderr = c.Stderr |
134 | | - |
135 | | - c.execCmd.Cancel = func() error { |
136 | | - // Try to let Nix exit gracefully by sending an interrupt |
137 | | - // instead of the default behavior of killing it. |
138 | | - c.logger.DebugContext(ctx, "sending interrupt to nix process", slog.Group("cmd", |
139 | | - "args", c.Args, |
140 | | - "path", c.execCmd.Path, |
141 | | - "pid", c.execCmd.Process.Pid, |
142 | | - )) |
143 | | - err := c.execCmd.Process.Signal(os.Interrupt) |
144 | | - if errors.Is(err, os.ErrProcessDone) { |
145 | | - // Nix already exited; execCmd.Wait will use the exit |
146 | | - // code. |
147 | | - return err |
148 | | - } |
149 | | - if err != nil { |
150 | | - // We failed to send SIGINT, so kill the process |
151 | | - // instead. |
152 | | - // |
153 | | - // - If Nix already exited, Kill will return |
154 | | - // os.ErrProcessDone and execCmd.Wait will use |
155 | | - // the exit code. |
156 | | - // - Otherwise, execCmd.Wait will always return an |
157 | | - // error. |
158 | | - c.logger.ErrorContext(ctx, "error interrupting nix process, attempting to kill", |
159 | | - "err", err, slog.Group("cmd", |
160 | | - "args", c.Args, |
161 | | - "path", c.execCmd.Path, |
162 | | - "pid", c.execCmd.Process.Pid, |
163 | | - )) |
164 | | - return c.execCmd.Process.Kill() |
165 | | - } |
166 | | - |
167 | | - // We sent the SIGINT successfully. It's still possible for Nix |
168 | | - // to exit successfully, so return os.ErrProcessDone so that |
169 | | - // execCmd.Wait uses the exit code instead of ctx.Err. |
170 | | - return os.ErrProcessDone |
171 | | - } |
172 | | - // Kill Nix if it doesn't exit within 15 seconds of Devbox sending an |
173 | | - // interrupt. |
174 | | - c.execCmd.WaitDelay = 15 * time.Second |
175 | | - return c.execCmd |
176 | | -} |
177 | | - |
178 | | -func (c *cmd) error(ctx context.Context, err error) error { |
179 | | - if err == nil { |
180 | | - return nil |
181 | | - } |
182 | | - |
183 | | - cmdErr := &cmdError{err: err} |
184 | | - if errors.Is(err, exec.ErrNotFound) { |
185 | | - cmdErr.msg = fmt.Sprintf("nix: %s not found in $PATH", c.Args[0]) |
| 3 | +func init() { |
| 4 | + Default.ExtraArgs = Args{ |
| 5 | + "--extra-experimental-features", "ca-derivations", |
| 6 | + "--option", "experimental-features", "nix-command flakes fetch-closure", |
186 | 7 | } |
187 | | - |
188 | | - switch { |
189 | | - case errors.Is(ctx.Err(), context.Canceled): |
190 | | - cmdErr.msg = "nix: command canceled" |
191 | | - case errors.Is(ctx.Err(), context.DeadlineExceeded): |
192 | | - cmdErr.msg = "nix: command timed out" |
193 | | - default: |
194 | | - cmdErr.msg = "nix: command error" |
195 | | - } |
196 | | - cmdErr.msg += ": " + c.String() |
197 | | - |
198 | | - var exitErr *exec.ExitError |
199 | | - if errors.As(err, &exitErr) { |
200 | | - if stderr := c.stderrExcerpt(exitErr.Stderr); len(stderr) != 0 { |
201 | | - cmdErr.msg += ": " + stderr |
202 | | - } |
203 | | - if exitErr.Exited() { |
204 | | - cmdErr.msg += fmt.Sprintf(": exit code %d", exitErr.ExitCode()) |
205 | | - return cmdErr |
206 | | - } |
207 | | - if stat, ok := exitErr.Sys().(syscall.WaitStatus); ok && stat.Signaled() { |
208 | | - cmdErr.msg += fmt.Sprintf(": exit due to signal %d (%[1]s)", stat.Signal()) |
209 | | - return cmdErr |
210 | | - } |
211 | | - } |
212 | | - |
213 | | - if !errors.Is(err, ctx.Err()) { |
214 | | - cmdErr.msg += ": " + err.Error() |
215 | | - } |
216 | | - return cmdErr |
217 | | -} |
218 | | - |
219 | | -func (*cmd) stderrExcerpt(stderr []byte) string { |
220 | | - stderr = bytes.TrimSpace(stderr) |
221 | | - if len(stderr) == 0 { |
222 | | - return "" |
223 | | - } |
224 | | - |
225 | | - lines := bytes.Split(stderr, []byte("\n")) |
226 | | - slices.Reverse(lines) |
227 | | - for _, line := range lines { |
228 | | - line = bytes.TrimSpace(line) |
229 | | - after, found := bytes.CutPrefix(line, []byte("error: ")) |
230 | | - if !found { |
231 | | - continue |
232 | | - } |
233 | | - after = bytes.TrimSpace(after) |
234 | | - if len(after) == 0 { |
235 | | - continue |
236 | | - } |
237 | | - stderr = after |
238 | | - break |
239 | | - |
240 | | - } |
241 | | - |
242 | | - excerpt := string(stderr) |
243 | | - if !strconv.CanBackquote(excerpt) { |
244 | | - quoted := strconv.Quote(excerpt) |
245 | | - excerpt = quoted[1 : len(quoted)-1] |
246 | | - } |
247 | | - return excerpt |
248 | 8 | } |
249 | 9 |
|
250 | | -type cmdArgs []any |
251 | | - |
252 | | -func appendArgs[E any](args cmdArgs, new []E) cmdArgs { |
| 10 | +func appendArgs[E any](args Args, new []E) Args { |
253 | 11 | for _, elem := range new { |
254 | 12 | args = append(args, elem) |
255 | 13 | } |
256 | 14 | return args |
257 | 15 | } |
258 | 16 |
|
259 | | -func (c cmdArgs) StringSlice() []string { |
260 | | - s := make([]string, len(c)) |
261 | | - for i := range c { |
262 | | - s[i] = fmt.Sprint(c[i]) |
263 | | - } |
264 | | - return s |
265 | | -} |
266 | | - |
267 | | -func (c cmdArgs) String() string { |
268 | | - if len(c) == 0 { |
269 | | - return "" |
270 | | - } |
271 | | - |
272 | | - sb := &strings.Builder{} |
273 | | - c.writeQuoted(sb, fmt.Sprint(c[0])) |
274 | | - if len(c) == 1 { |
275 | | - return sb.String() |
276 | | - } |
277 | | - |
278 | | - for _, arg := range c[1:] { |
279 | | - sb.WriteByte(' ') |
280 | | - c.writeQuoted(sb, fmt.Sprint(arg)) |
281 | | - } |
282 | | - return sb.String() |
283 | | -} |
284 | | - |
285 | | -func (cmdArgs) writeQuoted(dst *strings.Builder, str string) { |
286 | | - needsQuote := strings.ContainsAny(str, ";\"'()$|&><` \t\r\n\\#{~*?[=") |
287 | | - if !needsQuote { |
288 | | - dst.WriteString(str) |
289 | | - return |
290 | | - } |
291 | | - |
292 | | - canSingleQuote := !strings.Contains(str, "'") |
293 | | - if canSingleQuote { |
294 | | - dst.WriteByte('\'') |
295 | | - dst.WriteString(str) |
296 | | - dst.WriteByte('\'') |
297 | | - return |
298 | | - } |
299 | | - |
300 | | - dst.WriteByte('"') |
301 | | - for _, r := range str { |
302 | | - switch r { |
303 | | - // Special characters inside double quotes: |
304 | | - // https://pubs.opengroup.org/onlinepubs/009604499/utilities/xcu_chap02.html#tag_02_02_03 |
305 | | - case '$', '`', '"', '\\': |
306 | | - dst.WriteRune('\\') |
307 | | - } |
308 | | - dst.WriteRune(r) |
309 | | - } |
310 | | - dst.WriteByte('"') |
311 | | -} |
312 | | - |
313 | | -type cmdError struct { |
314 | | - msg string |
315 | | - err error |
316 | | -} |
317 | | - |
318 | | -func (c *cmdError) Redact() string { |
319 | | - return c.Error() |
320 | | -} |
321 | | - |
322 | | -func (c *cmdError) Error() string { |
323 | | - return c.msg |
324 | | -} |
325 | | - |
326 | | -func (c *cmdError) Unwrap() error { |
327 | | - return c.err |
328 | | -} |
329 | | - |
330 | 17 | func allowUnfreeEnv(curEnv []string) []string { |
331 | 18 | return append(curEnv, "NIXPKGS_ALLOW_UNFREE=1") |
332 | 19 | } |
|
0 commit comments