mirror of
https://github.com/rocky-linux/srpmproc.git
synced 2025-01-10 02:10:55 +00:00
432 lines
12 KiB
Go
432 lines
12 KiB
Go
// Copyright (c) 2021 The Srpmproc Authors
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
// copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
// SOFTWARE.
|
|
|
|
package srpmproc
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/go-git/go-git/v5/plumbing/transport"
|
|
"github.com/rocky-linux/srpmproc/pkg/misc"
|
|
|
|
"github.com/go-git/go-billy/v5"
|
|
"github.com/go-git/go-billy/v5/memfs"
|
|
"github.com/go-git/go-git/v5"
|
|
"github.com/go-git/go-git/v5/config"
|
|
"github.com/go-git/go-git/v5/plumbing"
|
|
"github.com/go-git/go-git/v5/storage/memory"
|
|
"github.com/rocky-linux/srpmproc/modulemd"
|
|
srpmprocpb "github.com/rocky-linux/srpmproc/pb"
|
|
"github.com/rocky-linux/srpmproc/pkg/data"
|
|
"github.com/rocky-linux/srpmproc/pkg/directives"
|
|
"google.golang.org/protobuf/encoding/prototext"
|
|
)
|
|
|
|
func cfgPatches(pd *data.ProcessData, md *data.ModeData, patchTree *git.Worktree, pushTree *git.Worktree) error {
|
|
// check CFG patches
|
|
// use PATCHES directory if it exists otherwise ROCKY/CFG
|
|
cfgdir := "PATCHES"
|
|
_, err := patchTree.Filesystem.Stat(cfgdir)
|
|
if err != nil {
|
|
cfgdir = "ROCKY/CFG"
|
|
}
|
|
_, err = patchTree.Filesystem.Stat(cfgdir)
|
|
if err == nil {
|
|
// iterate through patches
|
|
infos, err := patchTree.Filesystem.ReadDir(cfgdir)
|
|
if err != nil {
|
|
return fmt.Errorf("could not walk patches: %v", err)
|
|
}
|
|
|
|
for _, info := range infos {
|
|
// can only process .cfg files
|
|
if !strings.HasSuffix(info.Name(), ".cfg") {
|
|
continue
|
|
}
|
|
|
|
pd.Log.Printf("applying directive %s", info.Name())
|
|
filePath := filepath.Join(cfgdir, info.Name())
|
|
directive, err := patchTree.Filesystem.Open(filePath)
|
|
if err != nil {
|
|
return fmt.Errorf("could not open directive file %s: %v", info.Name(), err)
|
|
}
|
|
directiveBytes, err := io.ReadAll(directive)
|
|
if err != nil {
|
|
return fmt.Errorf("could not read directive file: %v", err)
|
|
}
|
|
|
|
var cfg srpmprocpb.Cfg
|
|
err = prototext.Unmarshal(directiveBytes, &cfg)
|
|
if err != nil {
|
|
return fmt.Errorf("could not unmarshal cfg file: %v", err)
|
|
}
|
|
|
|
errs := directives.Apply(&cfg, pd, md, patchTree, pushTree)
|
|
if errs != nil {
|
|
err := json.NewEncoder(os.Stdout).Encode(errs)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return fmt.Errorf("directives could not be applied")
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func applyPatches(pd *data.ProcessData, md *data.ModeData, patchTree *git.Worktree, pushTree *git.Worktree) error {
|
|
// check if patches exist
|
|
cfgdir := "PATCHES"
|
|
_, err := patchTree.Filesystem.Stat(cfgdir)
|
|
if err != nil {
|
|
cfgdir = "ROCKY"
|
|
}
|
|
_, err = patchTree.Filesystem.Stat(cfgdir)
|
|
if err == nil {
|
|
err := cfgPatches(pd, md, patchTree, pushTree)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func executePatchesRpm(pd *data.ProcessData, md *data.ModeData) error {
|
|
// fetch patch repository
|
|
repo, err := git.Init(memory.NewStorage(), memfs.New())
|
|
if err != nil {
|
|
return fmt.Errorf("could not create new dist Repo: %v", err)
|
|
}
|
|
w, err := repo.Worktree()
|
|
if err != nil {
|
|
return fmt.Errorf("could not get dist Worktree: %v", err)
|
|
}
|
|
|
|
remoteUrl := fmt.Sprintf("%s/patch/%s.git", pd.UpstreamPrefix, gitlabify(md.Name))
|
|
refspec := config.RefSpec(fmt.Sprintf("+refs/heads/*:refs/remotes/origin/*"))
|
|
|
|
_, err = repo.CreateRemote(&config.RemoteConfig{
|
|
Name: "origin",
|
|
URLs: []string{remoteUrl},
|
|
Fetch: []config.RefSpec{refspec},
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("could not create remote: %v", err)
|
|
}
|
|
|
|
fetchOptions := &git.FetchOptions{
|
|
Auth: pd.Authenticator,
|
|
RemoteName: "origin",
|
|
RefSpecs: []config.RefSpec{refspec},
|
|
}
|
|
if !strings.HasPrefix(pd.UpstreamPrefix, "http") {
|
|
fetchOptions.Auth = pd.Authenticator
|
|
}
|
|
err = repo.Fetch(fetchOptions)
|
|
|
|
refName := plumbing.NewBranchReferenceName(md.PushBranch)
|
|
pd.Log.Printf("set reference to ref: %s", refName)
|
|
|
|
if err != nil {
|
|
if err == transport.ErrInvalidAuthMethod || err == transport.ErrAuthenticationRequired {
|
|
fetchOptions.Auth = nil
|
|
err = repo.Fetch(fetchOptions)
|
|
if err != nil {
|
|
// no patches active
|
|
log.Println("info: patch repo not found")
|
|
return nil
|
|
}
|
|
} else {
|
|
// no patches active
|
|
log.Println("info: patch repo not found")
|
|
return nil
|
|
}
|
|
}
|
|
|
|
err = w.Checkout(&git.CheckoutOptions{
|
|
Branch: plumbing.NewRemoteReferenceName("origin", "main"),
|
|
Force: true,
|
|
})
|
|
// common patches found, apply them
|
|
if err == nil {
|
|
err := applyPatches(pd, md, w, md.Worktree)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
log.Println("info: no common patches found")
|
|
}
|
|
|
|
err = w.Checkout(&git.CheckoutOptions{
|
|
Branch: plumbing.NewRemoteReferenceName("origin", md.PushBranch),
|
|
Force: true,
|
|
})
|
|
// branch specific patches found, apply them
|
|
if err == nil {
|
|
err := applyPatches(pd, md, w, md.Worktree)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
log.Println("info: no branch specific patches found")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func getTipStream(pd *data.ProcessData, module string, pushBranch string, origPushBranch string, tries int) (string, error) {
|
|
repo, err := git.Init(memory.NewStorage(), memfs.New())
|
|
if err != nil {
|
|
return "", fmt.Errorf("could not init git Repo: %v", err)
|
|
}
|
|
|
|
remoteUrl := fmt.Sprintf("%s/rpms/%s.git", pd.UpstreamPrefix, gitlabify(module))
|
|
refspec := config.RefSpec("+refs/heads/*:refs/remotes/origin/*")
|
|
remote, err := repo.CreateRemote(&config.RemoteConfig{
|
|
Name: "origin",
|
|
URLs: []string{remoteUrl},
|
|
Fetch: []config.RefSpec{refspec},
|
|
})
|
|
if err != nil {
|
|
return "", fmt.Errorf("could not create remote: %v", err)
|
|
}
|
|
|
|
list, err := remote.List(&git.ListOptions{
|
|
Auth: pd.Authenticator,
|
|
})
|
|
if err != nil {
|
|
pd.Log.Printf("could not import module: %s", module)
|
|
if tries < 3 {
|
|
pd.Log.Printf("could not get rpm refs. will retry in 3s. %v", err)
|
|
time.Sleep(3 * time.Second)
|
|
return getTipStream(pd, module, pushBranch, origPushBranch, tries+1)
|
|
}
|
|
|
|
return "", fmt.Errorf("could not get rpm refs. import the rpm before the module: %v", err)
|
|
}
|
|
|
|
var tipHash string
|
|
|
|
for _, ref := range list {
|
|
if strings.Contains(ref.Name().String(), "-bootstrap") {
|
|
continue
|
|
}
|
|
|
|
prefix := fmt.Sprintf("refs/heads/%s", pushBranch)
|
|
|
|
branchVersion := strings.Split(pushBranch, "-")
|
|
if len(branchVersion) == 3 {
|
|
version := branchVersion[2]
|
|
// incompatible pattern: v1,v2 etc.
|
|
// example can be found in refs/tags/imports/c8-stream-1.1/subversion-1.10-8030020200519083055.9ce6d490
|
|
if strings.HasPrefix(version, "v") {
|
|
prefix = strings.Replace(prefix, version, strings.TrimPrefix(version, "v"), 1)
|
|
}
|
|
}
|
|
|
|
if strings.HasPrefix(ref.Name().String(), prefix) {
|
|
tipHash = ref.Hash().String()
|
|
}
|
|
}
|
|
|
|
if tipHash == "" {
|
|
for _, ref := range list {
|
|
if strings.Contains(ref.Name().String(), "-bootstrap") {
|
|
continue
|
|
}
|
|
|
|
prefix := fmt.Sprintf("refs/heads/%s", origPushBranch)
|
|
|
|
if strings.HasPrefix(ref.Name().String(), prefix) {
|
|
tipHash = ref.Hash().String()
|
|
}
|
|
}
|
|
}
|
|
|
|
if tipHash == "" {
|
|
for _, ref := range list {
|
|
if strings.Contains(ref.Name().String(), "-bootstrap") {
|
|
continue
|
|
}
|
|
|
|
if !strings.Contains(ref.Name().String(), "stream") {
|
|
tipHash = ref.Hash().String()
|
|
}
|
|
}
|
|
}
|
|
|
|
if tipHash == "" {
|
|
for _, ref := range list {
|
|
log.Println(pushBranch, ref.Name())
|
|
}
|
|
return "", fmt.Errorf("could not find tip hash")
|
|
}
|
|
|
|
return strings.TrimSpace(tipHash), nil
|
|
}
|
|
|
|
func patchModuleYaml(pd *data.ProcessData, md *data.ModeData) error {
|
|
// special case for platform.yaml
|
|
_, err := md.Worktree.Filesystem.Open("platform.yaml")
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
|
|
mdTxtPath := ""
|
|
var f billy.File
|
|
|
|
// tagless mode implies we're looking for CentOS Stream modules, which are generally "SOURCES/NAME.yaml" (copied to SOURCES/ from import)
|
|
// if not tagless mode, proceed as usual with SOURCES/modulemd.*.txt
|
|
if pd.TaglessMode {
|
|
mdTxtPath = fmt.Sprintf("SOURCES/%s.yaml", md.Name)
|
|
f, err = md.Worktree.Filesystem.Open(mdTxtPath)
|
|
if err != nil {
|
|
mdTxtPath = fmt.Sprintf("SOURCES/%s.yml", md.Name)
|
|
f, err = md.Worktree.Filesystem.Open(mdTxtPath)
|
|
if err != nil {
|
|
return fmt.Errorf("could not open modulemd file: %v", err)
|
|
}
|
|
}
|
|
} else {
|
|
mdTxtPath = "SOURCES/modulemd.src.txt"
|
|
f, err = md.Worktree.Filesystem.Open(mdTxtPath)
|
|
if err != nil {
|
|
mdTxtPath = "SOURCES/modulemd.txt"
|
|
f, err = md.Worktree.Filesystem.Open(mdTxtPath)
|
|
if err != nil {
|
|
return fmt.Errorf("could not open modulemd file: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
content, err := io.ReadAll(f)
|
|
if err != nil {
|
|
return fmt.Errorf("could not read modulemd file: %v", err)
|
|
}
|
|
|
|
module, err := modulemd.Parse(content)
|
|
if err != nil {
|
|
return fmt.Errorf("could not parse modulemd file: %v", err)
|
|
}
|
|
|
|
// Get stream branch from tag
|
|
// (in tagless mode we are trusting the "Stream: <VERSION>" text in the source YAML to be accurate)
|
|
if !pd.TaglessMode {
|
|
match := misc.GetTagImportRegex(pd).FindStringSubmatch(md.TagBranch)
|
|
streamBranch := strings.Split(match[2], "-")
|
|
|
|
// Force stream to be the same as stream name in branch
|
|
if module.V2 != nil {
|
|
module.V2.Data.Stream = streamBranch[len(streamBranch)-1]
|
|
}
|
|
if module.V3 != nil {
|
|
module.V3.Data.Stream = streamBranch[len(streamBranch)-1]
|
|
}
|
|
}
|
|
|
|
var components *modulemd.Components
|
|
if module.V2 != nil {
|
|
components = module.V2.Data.Components
|
|
}
|
|
if module.V3 != nil {
|
|
components = module.V3.Data.Components
|
|
}
|
|
|
|
log.Println("This module contains the following rpms:")
|
|
for name := range components.Rpms {
|
|
pd.Log.Printf("\t- %s", name)
|
|
}
|
|
|
|
defaultBranch := md.PushBranch
|
|
if pd.ModuleFallbackStream != "" {
|
|
defaultBranch = fmt.Sprintf("%s%d-stream-%s", pd.BranchPrefix, pd.Version, pd.ModuleFallbackStream)
|
|
}
|
|
|
|
for name, rpm := range components.Rpms {
|
|
var tipHash string
|
|
var pushBranch string
|
|
|
|
split := strings.Split(rpm.Ref, "-")
|
|
// TODO: maybe point to correct release tag? but refer to latest for now,
|
|
// we're bootstrapping a new distro for latest RHEL8 anyways. So earlier
|
|
// versions are not that important
|
|
if strings.HasPrefix(rpm.Ref, "stream-rhel-rhel-") {
|
|
pushBranch = defaultBranch
|
|
} else if strings.HasPrefix(rpm.Ref, "stream-rhel-") && len(split) > 4 {
|
|
repString := fmt.Sprintf("%s%ss-", pd.BranchPrefix, string(split[4][0]))
|
|
newString := fmt.Sprintf("%s%s-", pd.BranchPrefix, string(split[4][0]))
|
|
pushBranch = strings.Replace(md.PushBranch, repString, newString, 1)
|
|
} else if strings.HasPrefix(rpm.Ref, "stream-") && len(split) == 2 {
|
|
pushBranch = defaultBranch
|
|
} else if strings.HasPrefix(rpm.Ref, "stream-") && len(split) == 3 {
|
|
// example: ant
|
|
pushBranch = fmt.Sprintf("%s%d-stream-%s", pd.BranchPrefix, pd.Version, split[2])
|
|
} else if strings.HasPrefix(rpm.Ref, "stream-") {
|
|
pushBranch = fmt.Sprintf("%s%s-stream-%s", pd.BranchPrefix, string(split[3][0]), split[1])
|
|
} else if strings.HasPrefix(rpm.Ref, "rhel-") {
|
|
pushBranch = defaultBranch
|
|
} else {
|
|
return fmt.Errorf("could not recognize modulemd ref")
|
|
}
|
|
|
|
tipHash, err = getTipStream(pd, name, pushBranch, md.PushBranch, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if tipHash == "0000000000000000000000000000000000000000" {
|
|
pushBranch = defaultBranch
|
|
tipHash, err = getTipStream(pd, name, pushBranch, md.PushBranch, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// If we got this far, we're good. This means the repos, branches, and commits exist.
|
|
// If this option is on, just use the branch name.
|
|
if pd.ModuleBranchNames {
|
|
tipHash = md.PushBranch
|
|
}
|
|
|
|
rpm.Ref = tipHash
|
|
}
|
|
|
|
rootModule := fmt.Sprintf("%s.yaml", md.Name)
|
|
err = module.Marshal(md.Worktree.Filesystem, rootModule)
|
|
if err != nil {
|
|
return fmt.Errorf("could not marshal root modulemd: %v", err)
|
|
}
|
|
|
|
_, err = md.Worktree.Add(rootModule)
|
|
if err != nil {
|
|
return fmt.Errorf("could not add root modulemd: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|