internal/fetcher/channel.go (view raw)
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 | 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 NewChannelFetcher( source *config.Source, dataPath string, logger *slog.Logger, ) (*ChannelFetcher, error) { switch source.Importer { case config.Options: return &ChannelFetcher{ DataPath: dataPath, Source: source, Logger: logger, }, nil default: return nil, fmt.Errorf("unsupported importer type %s", source.Importer) } } 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"), } return } |