2020-08-21 14:49:48 +02:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"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"
|
|
|
|
)
|
|
|
|
|
2021-08-04 17:31:16 +02:00
|
|
|
func abapEnvironmentCloneGitRepo(config abapEnvironmentCloneGitRepoOptions, _ *telemetry.CustomData) {
|
2020-08-21 14:49:48 +02:00
|
|
|
|
|
|
|
c := command.Command{}
|
|
|
|
|
|
|
|
c.Stdout(log.Writer())
|
|
|
|
c.Stderr(log.Writer())
|
|
|
|
|
|
|
|
var autils = abaputils.AbapUtils{
|
|
|
|
Exec: &c,
|
|
|
|
}
|
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
apiManager := abaputils.SoftwareComponentApiManager{
|
|
|
|
Client: &piperhttp.Client{},
|
|
|
|
PollIntervall: 5 * time.Second,
|
|
|
|
}
|
2020-08-21 14:49:48 +02:00
|
|
|
// error situations should stop execution through log.Entry().Fatal() call which leads to an os.Exit(1) in the end
|
2023-11-28 14:26:31 +02:00
|
|
|
err := runAbapEnvironmentCloneGitRepo(&config, &autils, &apiManager)
|
2020-08-21 14:49:48 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Entry().WithError(err).Fatal("step execution failed")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
func runAbapEnvironmentCloneGitRepo(config *abapEnvironmentCloneGitRepoOptions, com abaputils.Communication, apiManager abaputils.SoftwareComponentApiManagerInterface) error {
|
2020-08-21 14:49:48 +02:00
|
|
|
// Mapping for options
|
2020-11-02 15:17:13 +02:00
|
|
|
subOptions := convertCloneConfig(config)
|
2020-08-21 14:49:48 +02:00
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
errConfig := checkConfiguration(config)
|
|
|
|
if errConfig != nil {
|
|
|
|
return errors.Wrap(errConfig, "The provided configuration is not allowed")
|
|
|
|
}
|
|
|
|
|
|
|
|
repositories, errGetRepos := abaputils.GetRepositories(&abaputils.RepositoriesConfig{BranchName: config.BranchName, RepositoryName: config.RepositoryName, Repositories: config.Repositories}, false)
|
|
|
|
if errGetRepos != nil {
|
|
|
|
return errors.Wrap(errGetRepos, "Could not read repositories")
|
|
|
|
}
|
|
|
|
|
2020-08-21 14:49:48 +02:00
|
|
|
// Determine the host, user and password, either via the input parameters or via a cloud foundry service key
|
2020-12-08 10:31:08 +02:00
|
|
|
connectionDetails, errorGetInfo := com.GetAbapCommunicationArrangementInfo(subOptions, "")
|
2020-08-21 14:49:48 +02:00
|
|
|
if errorGetInfo != nil {
|
|
|
|
return errors.Wrap(errorGetInfo, "Parameters for the ABAP Connection not available")
|
|
|
|
}
|
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
log.Entry().Infof("Start cloning %v repositories", len(repositories))
|
|
|
|
for _, repo := range repositories {
|
2020-08-21 14:49:48 +02:00
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
cloneError := cloneSingleRepo(apiManager, connectionDetails, repo, config, com)
|
|
|
|
if cloneError != nil {
|
|
|
|
return cloneError
|
|
|
|
}
|
2022-06-30 10:43:33 +02:00
|
|
|
}
|
2023-11-28 14:26:31 +02:00
|
|
|
abaputils.AddDefaultDashedLine(1)
|
|
|
|
log.Entry().Info("All repositories were cloned successfully")
|
|
|
|
return nil
|
|
|
|
}
|
2022-06-30 10:43:33 +02:00
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
func cloneSingleRepo(apiManager abaputils.SoftwareComponentApiManagerInterface, connectionDetails abaputils.ConnectionDetailsHTTP, repo abaputils.Repository, config *abapEnvironmentCloneGitRepoOptions, com abaputils.Communication) error {
|
|
|
|
|
|
|
|
// New API instance for each request
|
|
|
|
// Triggering the Clone of the repository into the ABAP Environment system
|
|
|
|
// Polling the status of the repository import on the ABAP Environment system
|
|
|
|
// If the repository had been cloned already, as checkout/pull has been done - polling the status is not necessary anymore
|
|
|
|
api, errGetAPI := apiManager.GetAPI(connectionDetails, repo)
|
|
|
|
if errGetAPI != nil {
|
|
|
|
return errors.Wrap(errGetAPI, "Could not initialize the connection to the system")
|
2020-08-21 14:49:48 +02:00
|
|
|
}
|
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
logString := repo.GetCloneLogString()
|
|
|
|
errorString := "Clone of " + logString + " failed on the ABAP system"
|
2020-11-02 15:17:13 +02:00
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
abaputils.AddDefaultDashedLine(1)
|
|
|
|
log.Entry().Info("Start cloning " + logString)
|
|
|
|
abaputils.AddDefaultDashedLine(1)
|
2020-11-02 15:17:13 +02:00
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
alreadyCloned, activeBranch, errCheckCloned := api.GetRepository()
|
|
|
|
if errCheckCloned != nil {
|
|
|
|
return errors.Wrapf(errCheckCloned, errorString)
|
|
|
|
}
|
2020-08-21 14:49:48 +02:00
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
if !alreadyCloned {
|
|
|
|
errClone := api.Clone()
|
|
|
|
if errClone != nil {
|
|
|
|
return errors.Wrapf(errClone, errorString)
|
2020-08-21 14:49:48 +02:00
|
|
|
}
|
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
status, errorPollEntity := abaputils.PollEntity(api, apiManager.GetPollIntervall())
|
|
|
|
if errorPollEntity != nil {
|
|
|
|
return errors.Wrapf(errorPollEntity, errorString)
|
|
|
|
}
|
|
|
|
if status == "E" {
|
|
|
|
return errors.New("Clone of " + logString + " failed on the ABAP System")
|
|
|
|
}
|
|
|
|
log.Entry().Info("The " + logString + " was cloned successfully")
|
|
|
|
} else {
|
|
|
|
abaputils.AddDefaultDashedLine(2)
|
|
|
|
log.Entry().Infof("%s", "The repository / software component has already been cloned on the ABAP Environment system ")
|
|
|
|
log.Entry().Infof("%s", "If required, a `checkout branch`, and a `pull` will be performed instead")
|
|
|
|
abaputils.AddDefaultDashedLine(2)
|
|
|
|
var returnedError error
|
|
|
|
if repo.Branch != "" && !(activeBranch == repo.Branch) {
|
|
|
|
returnedError = runAbapEnvironmentCheckoutBranch(getCheckoutOptions(config, repo), com, apiManager)
|
|
|
|
abaputils.AddDefaultDashedLine(2)
|
|
|
|
if returnedError != nil {
|
|
|
|
return returnedError
|
2022-06-30 10:43:33 +02:00
|
|
|
}
|
2020-08-21 14:49:48 +02:00
|
|
|
}
|
2023-11-28 14:26:31 +02:00
|
|
|
returnedError = runAbapEnvironmentPullGitRepo(getPullOptions(config, repo), com, apiManager)
|
|
|
|
return returnedError
|
2020-08-21 14:49:48 +02:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
func getCheckoutOptions(config *abapEnvironmentCloneGitRepoOptions, repo abaputils.Repository) *abapEnvironmentCheckoutBranchOptions {
|
|
|
|
checkoutOptions := abapEnvironmentCheckoutBranchOptions{
|
|
|
|
Username: config.Username,
|
|
|
|
Password: config.Password,
|
|
|
|
Host: config.Host,
|
|
|
|
RepositoryName: repo.Name,
|
|
|
|
BranchName: repo.Branch,
|
|
|
|
CfAPIEndpoint: config.CfAPIEndpoint,
|
|
|
|
CfOrg: config.CfOrg,
|
|
|
|
CfServiceInstance: config.CfServiceInstance,
|
|
|
|
CfServiceKeyName: config.CfServiceKeyName,
|
|
|
|
CfSpace: config.CfSpace,
|
|
|
|
}
|
|
|
|
return &checkoutOptions
|
|
|
|
}
|
|
|
|
|
|
|
|
func getPullOptions(config *abapEnvironmentCloneGitRepoOptions, repo abaputils.Repository) *abapEnvironmentPullGitRepoOptions {
|
|
|
|
pullOptions := abapEnvironmentPullGitRepoOptions{
|
|
|
|
Username: config.Username,
|
|
|
|
Password: config.Password,
|
|
|
|
Host: config.Host,
|
|
|
|
RepositoryName: repo.Name,
|
|
|
|
CommitID: repo.CommitID,
|
|
|
|
CfAPIEndpoint: config.CfAPIEndpoint,
|
|
|
|
CfOrg: config.CfOrg,
|
|
|
|
CfServiceInstance: config.CfServiceInstance,
|
|
|
|
CfServiceKeyName: config.CfServiceKeyName,
|
|
|
|
CfSpace: config.CfSpace,
|
|
|
|
}
|
|
|
|
return &pullOptions
|
|
|
|
}
|
|
|
|
|
2022-06-30 10:43:33 +02:00
|
|
|
func checkConfiguration(config *abapEnvironmentCloneGitRepoOptions) error {
|
|
|
|
if config.Repositories != "" && config.RepositoryName != "" {
|
|
|
|
return errors.New("It is not allowed to configure the parameters `repositories`and `repositoryName` at the same time")
|
|
|
|
}
|
|
|
|
if config.Repositories == "" && config.RepositoryName == "" {
|
|
|
|
return errors.New("Please provide one of the following parameters: `repositories` or `repositoryName`")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
func triggerClone(repo abaputils.Repository, api abaputils.SoftwareComponentApiInterface) (error, bool) {
|
2020-08-21 14:49:48 +02:00
|
|
|
|
2023-11-28 14:26:31 +02:00
|
|
|
//cloneConnectionDetails.URL = cloneConnectionDetails.URL + "/sap/opu/odata/sap/MANAGE_GIT_REPOSITORY/Clones"
|
2020-08-21 14:49:48 +02:00
|
|
|
|
|
|
|
// The entity "Clones" does not allow for polling. To poll the progress, the related entity "Pull" has to be called
|
|
|
|
// While "Clones" has the key fields UUID, SC_NAME and BRANCH_NAME, "Pull" only has the key field UUID
|
2023-11-28 14:26:31 +02:00
|
|
|
//uriConnectionDetails.URL = uriConnectionDetails.URL + "/sap/opu/odata/sap/MANAGE_GIT_REPOSITORY/Pull(uuid=guid'" + body.UUID + "')"
|
|
|
|
return nil, false
|
2020-08-21 14:49:48 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 15:17:13 +02:00
|
|
|
func convertCloneConfig(config *abapEnvironmentCloneGitRepoOptions) abaputils.AbapEnvironmentOptions {
|
2020-08-21 14:49:48 +02:00
|
|
|
subOptions := abaputils.AbapEnvironmentOptions{}
|
|
|
|
|
|
|
|
subOptions.CfAPIEndpoint = config.CfAPIEndpoint
|
|
|
|
subOptions.CfServiceInstance = config.CfServiceInstance
|
|
|
|
subOptions.CfServiceKeyName = config.CfServiceKeyName
|
|
|
|
subOptions.CfOrg = config.CfOrg
|
|
|
|
subOptions.CfSpace = config.CfSpace
|
|
|
|
subOptions.Host = config.Host
|
|
|
|
subOptions.Password = config.Password
|
|
|
|
subOptions.Username = config.Username
|
|
|
|
return subOptions
|
|
|
|
}
|