srcinfo.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. package dep
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "os"
  7. "path/filepath"
  8. aurc "github.com/Jguer/aur"
  9. gosrc "github.com/Morganamilo/go-srcinfo"
  10. "github.com/leonelquinteros/gotext"
  11. "github.com/Jguer/yay/v12/pkg/dep/topo"
  12. "github.com/Jguer/yay/v12/pkg/settings/exe"
  13. )
  14. var ErrNoBuildFiles = errors.New(gotext.Get("cannot find PKGBUILD and .SRCINFO in directory"))
  15. func (g *Grapher) GraphFromSrcInfoDirs(ctx context.Context, graph *topo.Graph[string, *InstallInfo],
  16. srcInfosDirs []string,
  17. ) (*topo.Graph[string, *InstallInfo], error) {
  18. if graph == nil {
  19. graph = NewGraph()
  20. }
  21. srcInfos := map[string]*gosrc.Srcinfo{}
  22. for _, targetDir := range srcInfosDirs {
  23. if err := srcinfoExists(ctx, g.cmdBuilder, targetDir); err != nil {
  24. return nil, err
  25. }
  26. pkgbuild, err := gosrc.ParseFile(filepath.Join(targetDir, ".SRCINFO"))
  27. if err != nil {
  28. return nil, fmt.Errorf("%s: %w", gotext.Get("failed to parse .SRCINFO"), err)
  29. }
  30. srcInfos[targetDir] = pkgbuild
  31. }
  32. aurPkgsAdded := []*aurc.Pkg{}
  33. for pkgBuildDir, pkgbuild := range srcInfos {
  34. pkgBuildDir := pkgBuildDir
  35. aurPkgs, err := makeAURPKGFromSrcinfo(g.dbExecutor, pkgbuild)
  36. if err != nil {
  37. return nil, err
  38. }
  39. if len(aurPkgs) > 1 {
  40. var errPick error
  41. aurPkgs, errPick = g.pickSrcInfoPkgs(aurPkgs)
  42. if errPick != nil {
  43. return nil, errPick
  44. }
  45. }
  46. for _, pkg := range aurPkgs {
  47. pkg := pkg
  48. reason := Explicit
  49. if pkg := g.dbExecutor.LocalPackage(pkg.Name); pkg != nil {
  50. reason = Reason(pkg.Reason())
  51. }
  52. graph.AddNode(pkg.Name)
  53. g.addAurPkgProvides(pkg, graph)
  54. validateAndSetNodeInfo(graph, pkg.Name, &topo.NodeInfo[*InstallInfo]{
  55. Color: colorMap[reason],
  56. Background: bgColorMap[AUR],
  57. Value: &InstallInfo{
  58. Source: SrcInfo,
  59. Reason: reason,
  60. SrcinfoPath: &pkgBuildDir,
  61. AURBase: &pkg.PackageBase,
  62. Version: pkg.Version,
  63. },
  64. })
  65. }
  66. aurPkgsAdded = append(aurPkgsAdded, aurPkgs...)
  67. }
  68. g.AddDepsForPkgs(ctx, aurPkgsAdded, graph)
  69. return graph, nil
  70. }
  71. func srcinfoExists(ctx context.Context,
  72. cmdBuilder exe.ICmdBuilder, targetDir string,
  73. ) error {
  74. srcInfoDir := filepath.Join(targetDir, ".SRCINFO")
  75. pkgbuildDir := filepath.Join(targetDir, "PKGBUILD")
  76. if _, err := os.Stat(srcInfoDir); err == nil {
  77. if _, err := os.Stat(pkgbuildDir); err == nil {
  78. return nil
  79. }
  80. }
  81. if _, err := os.Stat(pkgbuildDir); err == nil {
  82. // run makepkg to generate .SRCINFO
  83. srcinfo, stderr, err := cmdBuilder.Capture(cmdBuilder.BuildMakepkgCmd(ctx, targetDir, "--printsrcinfo"))
  84. if err != nil {
  85. return fmt.Errorf("unable to generate .SRCINFO: %w - %s", err, stderr)
  86. }
  87. if err := os.WriteFile(srcInfoDir, []byte(srcinfo), 0o600); err != nil {
  88. return fmt.Errorf("unable to write .SRCINFO: %w", err)
  89. }
  90. return nil
  91. }
  92. return fmt.Errorf("%w: %s", ErrNoBuildFiles, targetDir)
  93. }