mirror of
https://github.com/SAP/jenkins-library.git
synced 2024-12-12 10:55:20 +02:00
109f5cd1bf
* Update abapEnvironmentAssemblePackages.go * Save 6 Seconds in Unit Tests Co-authored-by: Daniel Mieg <56156797+DanielMieg@users.noreply.github.com>
215 lines
6.7 KiB
Go
215 lines
6.7 KiB
Go
package cmd
|
|
|
|
import (
|
|
"encoding/json"
|
|
"path"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
abapbuild "github.com/SAP/jenkins-library/pkg/abap/build"
|
|
"github.com/SAP/jenkins-library/pkg/abaputils"
|
|
"github.com/SAP/jenkins-library/pkg/command"
|
|
piperhttp "github.com/SAP/jenkins-library/pkg/http"
|
|
"github.com/SAP/jenkins-library/pkg/log"
|
|
"github.com/SAP/jenkins-library/pkg/telemetry"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
type buildWithRepository struct {
|
|
build abapbuild.Build
|
|
repo abaputils.Repository
|
|
}
|
|
|
|
func abapEnvironmentAssemblePackages(config abapEnvironmentAssemblePackagesOptions, telemetryData *telemetry.CustomData, cpe *abapEnvironmentAssemblePackagesCommonPipelineEnvironment) {
|
|
// for command execution use Command
|
|
c := command.Command{}
|
|
// reroute command output to logging framework
|
|
c.Stdout(log.Writer())
|
|
c.Stderr(log.Writer())
|
|
|
|
var autils = abaputils.AbapUtils{
|
|
Exec: &c,
|
|
}
|
|
|
|
client := piperhttp.Client{}
|
|
err := runAbapEnvironmentAssemblePackages(&config, telemetryData, &autils, &client, cpe)
|
|
if err != nil {
|
|
log.Entry().WithError(err).Fatal("step execution failed")
|
|
}
|
|
}
|
|
|
|
func runAbapEnvironmentAssemblePackages(config *abapEnvironmentAssemblePackagesOptions, telemetryData *telemetry.CustomData, com abaputils.Communication, client piperhttp.Sender, cpe *abapEnvironmentAssemblePackagesCommonPipelineEnvironment) error {
|
|
conn := new(abapbuild.Connector)
|
|
var connConfig abapbuild.ConnectorConfiguration
|
|
connConfig.CfAPIEndpoint = config.CfAPIEndpoint
|
|
connConfig.CfOrg = config.CfOrg
|
|
connConfig.CfSpace = config.CfSpace
|
|
connConfig.CfServiceInstance = config.CfServiceInstance
|
|
connConfig.CfServiceKeyName = config.CfServiceKeyName
|
|
connConfig.Host = config.Host
|
|
connConfig.Username = config.Username
|
|
connConfig.Password = config.Password
|
|
connConfig.AddonDescriptor = config.AddonDescriptor
|
|
connConfig.MaxRuntimeInMinutes = config.MaxRuntimeInMinutes
|
|
|
|
err := conn.InitBuildFramework(connConfig, com, client)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
var addonDescriptor abaputils.AddonDescriptor
|
|
err = json.Unmarshal([]byte(config.AddonDescriptor), &addonDescriptor)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
delayBetweenPostsInSeconds := time.Duration(3 * time.Second)
|
|
builds, buildsAlreadyReleased, err := starting(addonDescriptor.Repositories, *conn, delayBetweenPostsInSeconds)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
maxRuntimeInMinutes := time.Duration(config.MaxRuntimeInMinutes) * time.Minute
|
|
pollIntervalsInSeconds := time.Duration(60 * time.Second)
|
|
err = polling(builds, maxRuntimeInMinutes, pollIntervalsInSeconds)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = checkIfFailedAndPrintLogs(builds)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
reposBackToCPE, err := downloadSARXML(builds)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// also write the already released packages back to cpe
|
|
for _, b := range buildsAlreadyReleased {
|
|
reposBackToCPE = append(reposBackToCPE, b.repo)
|
|
}
|
|
addonDescriptor.Repositories = reposBackToCPE
|
|
backToCPE, _ := json.Marshal(addonDescriptor)
|
|
cpe.abap.addonDescriptor = string(backToCPE)
|
|
|
|
return nil
|
|
}
|
|
|
|
func downloadSARXML(builds []buildWithRepository) ([]abaputils.Repository, error) {
|
|
var reposBackToCPE []abaputils.Repository
|
|
resultName := "SAR_XML"
|
|
envPath := filepath.Join(GeneralConfig.EnvRootPath, "commonPipelineEnvironment", "abap")
|
|
for i, b := range builds {
|
|
resultSARXML, err := b.build.GetResult(resultName)
|
|
if err != nil {
|
|
return reposBackToCPE, err
|
|
}
|
|
sarPackage := resultSARXML.AdditionalInfo
|
|
downloadPath := filepath.Join(envPath, path.Base(sarPackage))
|
|
log.Entry().Infof("Downloading SAR file %s to %s", path.Base(sarPackage), downloadPath)
|
|
err = resultSARXML.Download(downloadPath)
|
|
if err != nil {
|
|
return reposBackToCPE, err
|
|
}
|
|
builds[i].repo.SarXMLFilePath = downloadPath
|
|
reposBackToCPE = append(reposBackToCPE, builds[i].repo)
|
|
}
|
|
return reposBackToCPE, nil
|
|
}
|
|
|
|
func checkIfFailedAndPrintLogs(builds []buildWithRepository) error {
|
|
var buildFailed bool = false
|
|
for i := range builds {
|
|
if builds[i].build.RunState == abapbuild.Failed {
|
|
log.Entry().Errorf("Assembly of %s failed", builds[i].repo.PackageName)
|
|
buildFailed = true
|
|
}
|
|
builds[i].build.PrintLogs()
|
|
}
|
|
if buildFailed {
|
|
return errors.New("At least the assembly of one package failed")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func starting(repos []abaputils.Repository, conn abapbuild.Connector, delayBetweenPostsInSeconds time.Duration) ([]buildWithRepository, []buildWithRepository, error) {
|
|
var builds []buildWithRepository
|
|
var buildsAlreadyReleased []buildWithRepository
|
|
for _, repo := range repos {
|
|
assemblyBuild := abapbuild.Build{
|
|
Connector: conn,
|
|
}
|
|
buildRepo := buildWithRepository{
|
|
build: assemblyBuild,
|
|
repo: repo,
|
|
}
|
|
if repo.Status == "P" {
|
|
err := buildRepo.start()
|
|
if err != nil {
|
|
return builds, buildsAlreadyReleased, err
|
|
}
|
|
builds = append(builds, buildRepo)
|
|
} else {
|
|
log.Entry().Infof("Packages %s is in status '%s'. No need to run the assembly", repo.PackageName, repo.Status)
|
|
buildsAlreadyReleased = append(buildsAlreadyReleased, buildRepo)
|
|
}
|
|
|
|
//as batch events in the ABAP Backend need a little time
|
|
time.Sleep(delayBetweenPostsInSeconds)
|
|
}
|
|
return builds, buildsAlreadyReleased, nil
|
|
}
|
|
|
|
func polling(builds []buildWithRepository, maxRuntimeInMinutes time.Duration, pollIntervalsInSeconds time.Duration) error {
|
|
timeout := time.After(maxRuntimeInMinutes)
|
|
ticker := time.Tick(pollIntervalsInSeconds)
|
|
for {
|
|
select {
|
|
case <-timeout:
|
|
return errors.New("Timed out")
|
|
case <-ticker:
|
|
var allFinished bool = true
|
|
for i := range builds {
|
|
builds[i].build.Get()
|
|
if !builds[i].build.IsFinished() {
|
|
log.Entry().Infof("Assembly of %s is not yet finished, check again in %s", builds[i].repo.PackageName, pollIntervalsInSeconds)
|
|
allFinished = false
|
|
}
|
|
}
|
|
if allFinished {
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (b *buildWithRepository) start() error {
|
|
if b.repo.Name == "" || b.repo.Version == "" || b.repo.SpLevel == "" || b.repo.Namespace == "" || b.repo.PackageType == "" || b.repo.PackageName == "" {
|
|
return errors.New("Parameters missing. Please provide software component name, version, sp-level, namespace, packagetype and packagename")
|
|
}
|
|
valuesInput := abapbuild.Values{
|
|
Values: []abapbuild.Value{
|
|
{
|
|
ValueID: "SWC",
|
|
Value: b.repo.Name,
|
|
},
|
|
{
|
|
ValueID: "CVERS",
|
|
Value: b.repo.Name + "." + b.repo.Version + "." + b.repo.SpLevel,
|
|
},
|
|
{
|
|
ValueID: "NAMESPACE",
|
|
Value: b.repo.Namespace,
|
|
},
|
|
{
|
|
ValueID: "PACKAGE_NAME_" + b.repo.PackageType,
|
|
Value: b.repo.PackageName,
|
|
},
|
|
},
|
|
}
|
|
if b.repo.PredecessorCommitID != "" {
|
|
valuesInput.Values = append(valuesInput.Values,
|
|
abapbuild.Value{ValueID: "PREVIOUS_DELIVERY_COMMIT",
|
|
Value: b.repo.PredecessorCommitID})
|
|
}
|
|
phase := "BUILD_" + b.repo.PackageType
|
|
log.Entry().Infof("Starting assembly of package %s", b.repo.PackageName)
|
|
return b.build.Start(phase, valuesInput)
|
|
}
|