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
|
package fetcher
import (
"context"
"fmt"
"log/slog"
"os"
"os/exec"
"path"
"searchix/internal/config"
"searchix/internal/file"
"strconv"
"strings"
"github.com/pkg/errors"
)
type ChannelFetcher struct {
DataPath string
Source *config.Source
SourceFile string
Logger *slog.Logger
}
func (i *ChannelFetcher) FetchIfNeeded(
parent context.Context,
) (f FetchedFiles, updated bool, err error) {
ctx, cancel := context.WithTimeout(parent, i.Source.FetchTimeout.Duration)
defer cancel()
dest := i.DataPath
var before string
before, err = os.Readlink(dest)
if file.NeedNotExist(err) != nil {
err = errors.WithMessagef(err, "could not call readlink on file %s", dest)
return
}
i.Logger.Debug("stat before", "name", before)
args := []string{
"--no-build-output",
"--timeout",
strconv.Itoa(int(i.Source.FetchTimeout.Seconds() - 1)),
fmt.Sprintf("<%s/%s>", i.Source.Channel, i.Source.ImportPath),
"--attr",
i.Source.Attribute,
"--out-link",
dest,
}
if i.Source.URL != "" {
args = append(args, "-I", fmt.Sprintf("%s=%s", i.Source.Channel, i.Source.URL))
}
i.Logger.Debug("nix-build command", "args", args)
cmd := exec.CommandContext(ctx, "nix-build", args...)
var out []byte
out, err = cmd.Output()
if err != nil {
err = errors.WithMessage(err, "failed to run nix-build (--dry-run)")
return
}
i.Logger.Debug("nix-build", "output", strings.TrimSpace(string(out)))
outPath := path.Join(dest, i.Source.OutputPath)
i.Logger.Debug(
"checking output path",
"outputPath",
outPath,
"dest",
dest,
"source",
i.Source.OutputPath,
)
var after string
after, err = os.Readlink(dest)
if err = file.NeedNotExist(err); err != nil {
err = errors.WithMessagef(
err,
"failed to stat output file from nix-build, filename: %s",
outPath,
)
return
}
i.Logger.Debug("stat after", "name", after)
updated = before != after
f = FetchedFiles{
Options: path.Join(dest, i.Source.OutputPath, "options.json"),
Packages: path.Join(dest, i.Source.OutputPath, "packages.json"),
}
return
}
|