diff --git a/pkg/clients/has/components.go b/pkg/clients/has/components.go index de2477d56..48715db29 100644 --- a/pkg/clients/has/components.go +++ b/pkg/clients/has/components.go @@ -125,6 +125,24 @@ func (h *HasController) GetAllPipelineRunsForApplication(applicationName, namesp return nil, fmt.Errorf("no pipelinerun found for application %s", applicationName) } +// GetAllGroupSnapshotsForApplication returns the groupSnapshots for a given application in the namespace +func (h *HasController) GetAllGroupSnapshotsForApplication(applicationName, namespace string) (*appservice.SnapshotList, error) { + snapshotLabels := map[string]string{"appstudio.openshift.io/application": applicationName, "test.appstudio.openshift.io/type": "group"} + + list := &appservice.SnapshotList{} + err := h.KubeRest().List(context.Background(), list, &rclient.ListOptions{LabelSelector: labels.SelectorFromSet(snapshotLabels), Namespace: namespace}) + + if err != nil && !k8sErrors.IsNotFound(err) { + return nil, fmt.Errorf("error listing snapshots in %s namespace: %v", namespace, err) + } + + if len(list.Items) > 0 { + return list, nil + } + + return nil, fmt.Errorf("no snapshot found for application %s", applicationName) +} + // Set of options to retrigger pipelineRuns in CI to fight against flakynes type RetryOptions struct { // Indicate how many times a pipelineRun should be retriggered in case of flakines diff --git a/pkg/clients/integration/pipelineruns.go b/pkg/clients/integration/pipelineruns.go index d184012d6..ad0c2879c 100644 --- a/pkg/clients/integration/pipelineruns.go +++ b/pkg/clients/integration/pipelineruns.go @@ -267,3 +267,30 @@ func (i *IntegrationController) WaitForBuildPipelineRunToGetAnnotated(testNamesp return true, nil }) } + +// WaitForBuildPipelineToBeFinished wait for given build pipeline to finish. +// It exposes the error message from the failed task to the end user when the pipelineRun failed. +func (i *IntegrationController) WaitForBuildPipelineToBeFinished(testNamespace, applicationName, componentName string) error { + return wait.PollUntilContextTimeout(context.Background(), constants.PipelineRunPollingInterval, 15*time.Minute, true, func(ctx context.Context) (done bool, err error) { + pipelineRun, err := i.GetBuildPipelineRun(componentName, applicationName, testNamespace, false, "") + if err != nil { + GinkgoWriter.Println("Build pipelineRun has not been created yet for app %s/%s", testNamespace, applicationName) + return false, nil + } + for _, condition := range pipelineRun.Status.Conditions { + GinkgoWriter.Printf("PipelineRun %s reason: %s\n", pipelineRun.Name, condition.Reason) + + if !pipelineRun.IsDone() { + return false, nil + } + + if pipelineRun.GetStatusCondition().GetCondition(apis.ConditionSucceeded).IsTrue() { + return true, nil + } else { + logs, _ := tekton.GetFailedPipelineRunLogs(i.KubeRest(), i.KubeInterface(), pipelineRun) + return false, fmt.Errorf("%s", logs) + } + } + return false, nil + }) +} diff --git a/tests/integration-service/const.go b/tests/integration-service/const.go index 730ae0007..13b9b39de 100644 --- a/tests/integration-service/const.go +++ b/tests/integration-service/const.go @@ -20,6 +20,10 @@ const ( componentRepoNameForGeneralIntegration = "konflux-test-integration" componentRepoNameForIntegrationWithEnv = "konflux-test-integration-with-env" componentRepoNameForStatusReporting = "konflux-test-integration-status-report" + multiComponentRepoNameForGroupSnapshot = "group-snapshot-multi-component" + multiComponentDefaultBranch = "main" + multiComponentGitRevision = "0d1835404efb8ab7bb1ab5b5b82cda1ebfda4b25" + multiRepoComponentGitRevision = "2e41cf5a68674503c86b6637de35eeedc2893794" gitlabComponentRepoName = "hacbs-test-project-integration" componentDefaultBranch = "main" componentRevision = "34da5a8f51fba6a8b7ec75a727d3c72ebb5e1274" @@ -35,6 +39,8 @@ const ( snapshotAnnotation = "appstudio.openshift.io/snapshot" scenarioAnnotation = "test.appstudio.openshift.io/scenario" + groupSnapshotAnnotation = "test.appstudio.openshift.io/pr-group" + testGroupSnapshotAnnotation = "test.appstudio.openshift.io/group-test-info" pipelinerunFinalizerByIntegrationService = "test.appstudio.openshift.io/pipelinerun" snapshotRerunLabel = "test.appstudio.openshift.io/run" @@ -45,6 +51,9 @@ var ( componentGitSourceURLForGeneralIntegration = fmt.Sprintf("https://github.com/%s/%s", utils.GetEnv(constants.GITHUB_E2E_ORGANIZATION_ENV, "redhat-appstudio-qe"), componentRepoNameForGeneralIntegration) componentGitSourceURLForIntegrationWithEnv = fmt.Sprintf("https://github.com/%s/%s", utils.GetEnv(constants.GITHUB_E2E_ORGANIZATION_ENV, "redhat-appstudio-qe"), componentRepoNameForIntegrationWithEnv) componentGitSourceURLForStatusReporting = fmt.Sprintf("https://github.com/%s/%s", utils.GetEnv(constants.GITHUB_E2E_ORGANIZATION_ENV, "redhat-appstudio-qe"), componentRepoNameForStatusReporting) + multiComponentGitSourceURLForGroupSnapshotA = fmt.Sprintf("https://github.com/%s/%s", utils.GetEnv(constants.GITHUB_E2E_ORGANIZATION_ENV, "redhat-appstudio-qe"), multiComponentRepoNameForGroupSnapshot) + multiComponentGitSourceURLForGroupSnapshotB = fmt.Sprintf("https://github.com/%s/%s", utils.GetEnv(constants.GITHUB_E2E_ORGANIZATION_ENV, "redhat-appstudio-qe"), multiComponentRepoNameForGroupSnapshot) + multiComponentContextDirs = []string{"go-component", "python-component"} gitlabOrg = utils.GetEnv(constants.GITLAB_QE_ORG_ENV, constants.DefaultGitLabQEOrg) gitlabProjectIDForStatusReporting = fmt.Sprintf("%s/%s", gitlabOrg, gitlabComponentRepoName) gitlabComponentGitSourceURLForStatusReporting = fmt.Sprintf("https://gitlab.com/%s/%s", gitlabOrg, gitlabComponentRepoName) diff --git a/tests/integration-service/group-snapshots-tests.go b/tests/integration-service/group-snapshots-tests.go new file mode 100644 index 000000000..d700b34da --- /dev/null +++ b/tests/integration-service/group-snapshots-tests.go @@ -0,0 +1,542 @@ +package integration + +import ( + "fmt" + "os" + "time" + + "github.com/devfile/library/v2/pkg/util" + "github.com/google/go-github/v44/github" + "github.com/konflux-ci/e2e-tests/pkg/clients/has" + "github.com/konflux-ci/e2e-tests/pkg/constants" + "github.com/konflux-ci/e2e-tests/pkg/framework" + "github.com/konflux-ci/e2e-tests/pkg/utils" + + appstudioApi "github.com/konflux-ci/application-api/api/v1alpha1" + integrationv1beta2 "github.com/konflux-ci/integration-service/api/v1beta2" + . "github.com/onsi/ginkgo/v2" + . "github.com/onsi/gomega" + pipeline "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1" +) + +var _ = framework.IntegrationServiceSuiteDescribe("Creation of group snapshots for monorepo and multiple repos", Label("integration-service", "group-snapshot-creation"), func() { + defer GinkgoRecover() + + var f *framework.Framework + var err error + + var prNumber int + var timeout, interval time.Duration + var prHeadSha, mergeResultSha, mergeMultiResultSha string + var pacBranchNames []string + var componentNames []string + var snapshot *appstudioApi.Snapshot + var componentA *appstudioApi.Component + var componentB *appstudioApi.Component + var componentC *appstudioApi.Component + var groupSnapshots *appstudioApi.SnapshotList + var mergeResult *github.PullRequestMergeResult + var pipelineRun, testPipelinerun *pipeline.PipelineRun + var integrationTestScenarioPass *integrationv1beta2.IntegrationTestScenario + var applicationName, testNamespace, multiComponentBaseBranchName, multiComponentPRBranchName string + + AfterEach(framework.ReportFailure(&f)) + + Describe("with status reporting of Integration tests in CheckRuns", Ordered, func() { + BeforeAll(func() { + if os.Getenv(constants.SKIP_PAC_TESTS_ENV) == "true" { + Skip("Skipping this test due to configuration issue with Spray proxy") + } + + f, err = framework.NewFramework(utils.GetGeneratedNamespace("group")) + Expect(err).NotTo(HaveOccurred()) + testNamespace = f.UserNamespace + + if utils.IsPrivateHostname(f.OpenshiftConsoleHost) { + Skip("Using private cluster (not reachable from Github), skipping...") + } + + applicationName = createApp(*f, testNamespace) + + // The base branch or a ToBranch where all multi-component definitions will live + multiComponentBaseBranchName = fmt.Sprintf("love-triangle-%s", util.GenerateRandomString(6)) + err = f.AsKubeAdmin.CommonController.Github.CreateRef(multiComponentRepoNameForGroupSnapshot, multiComponentDefaultBranch, multiComponentGitRevision, multiComponentBaseBranchName) + Expect(err).ShouldNot(HaveOccurred()) + + // The base branch or ToBranch where different repo component definition will live + err = f.AsKubeAdmin.CommonController.Github.CreateRef(componentRepoNameForGeneralIntegration, multiComponentDefaultBranch, multiRepoComponentGitRevision, multiComponentBaseBranchName) + Expect(err).ShouldNot(HaveOccurred()) + + //Branch for creating pull request + multiComponentPRBranchName = fmt.Sprintf("%s-%s", "pr-branch", util.GenerateRandomString(6)) + + integrationTestScenarioPass, err = f.AsKubeAdmin.IntegrationController.CreateIntegrationTestScenario("", applicationName, testNamespace, gitURL, revision, pathInRepoPass, []string{}) + Expect(err).ShouldNot(HaveOccurred()) + }) + + AfterAll(func() { + if !CurrentSpecReport().Failed() { + cleanup(*f, testNamespace, applicationName, componentA.Name) + } + + // Delete new branches created by PaC and a testing branch used as a component's base branch + for _, pacBranchName := range pacBranchNames { + err = f.AsKubeAdmin.CommonController.Github.DeleteRef(multiComponentRepoNameForGroupSnapshot, pacBranchName) + if err != nil { + Expect(err.Error()).To(ContainSubstring("Reference does not exist")) + } + err = f.AsKubeAdmin.CommonController.Github.DeleteRef(componentRepoNameForGeneralIntegration, pacBranchName) + if err != nil { + Expect(err.Error()).To(ContainSubstring("Reference does not exist")) + } + } + + // Delete the created base branch + err = f.AsKubeAdmin.CommonController.Github.DeleteRef(multiComponentRepoNameForGroupSnapshot, multiComponentBaseBranchName) + if err != nil { + Expect(err.Error()).To(ContainSubstring(referenceDoesntExist)) + } + // Delete the created base branch for multi-repo + err = f.AsKubeAdmin.CommonController.Github.DeleteRef(componentRepoNameForGeneralIntegration, multiComponentBaseBranchName) + if err != nil { + Expect(err.Error()).To(ContainSubstring(referenceDoesntExist)) + } + + // Delete the created pr branch + err = f.AsKubeAdmin.CommonController.Github.DeleteRef(multiComponentRepoNameForGroupSnapshot, multiComponentPRBranchName) + if err != nil { + Expect(err.Error()).To(ContainSubstring(referenceDoesntExist)) + } + + // Delete the created pr branch for multi-repo + err = f.AsKubeAdmin.CommonController.Github.DeleteRef(componentRepoNameForGeneralIntegration, multiComponentPRBranchName) + if err != nil { + Expect(err.Error()).To(ContainSubstring(referenceDoesntExist)) + } + }) + + /* /\ + / \ + / /\ \ + / ____ \ + /_/ \_\ */ + When("we start creation of a new Component A", func() { + It("creates the Component A successfully", func() { + componentA = createComponentWithCustomBranch(*f, testNamespace, applicationName, multiComponentContextDirs[0]+"-"+util.GenerateRandomString(6), multiComponentGitSourceURLForGroupSnapshotA, multiComponentBaseBranchName, multiComponentContextDirs[0]) + + // Record the PaC branch names for cleanup + pacBranchName := constants.PaCPullRequestBranchPrefix + componentA.Name + pacBranchNames = append(pacBranchNames, pacBranchName) + }) + + It(fmt.Sprintf("triggers a Build PipelineRun for componentA %s", multiComponentContextDirs[0]), func() { + timeout = time.Second * 600 + interval = time.Second * 1 + Eventually(func() error { + pipelineRun, err = f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentA.Name, applicationName, testNamespace, "") + if err != nil { + GinkgoWriter.Printf("Build PipelineRun has not been created yet for the componentA %s/%s\n", testNamespace, componentA.Name) + return err + } + if !pipelineRun.HasStarted() { + return fmt.Errorf("build pipelinerunA %s/%s hasn't started yet", pipelineRun.GetNamespace(), pipelineRun.GetName()) + } + return nil + }, timeout, constants.PipelineRunPollingInterval).Should(Succeed(), fmt.Sprintf("timed out when waiting for the build PipelineRun to start for the componentA %s/%s", testNamespace, componentA.Name)) + }) + + It("does not contain an annotation with a Snapshot Name", func() { + Expect(pipelineRun.Annotations[snapshotAnnotation]).To(Equal("")) + }) + + It("should lead to build PipelineRunA finishing successfully", func() { + Expect(f.AsKubeAdmin.HasController.WaitForComponentPipelineToBeFinished(componentA, + "", f.AsKubeAdmin.TektonController, &has.RetryOptions{Retries: 2, Always: true}, pipelineRun)).To(Succeed()) + }) + + It(fmt.Sprintf("should lead to a PaC PR creation for componentA %s", multiComponentContextDirs[0]), func() { + timeout = time.Second * 300 + interval = time.Second * 1 + + Eventually(func() bool { + prs, err := f.AsKubeAdmin.CommonController.Github.ListPullRequests(multiComponentRepoNameForGroupSnapshot) + Expect(err).ShouldNot(HaveOccurred()) + + for _, pr := range prs { + if pr.Head.GetRef() == pacBranchNames[0] { + prNumber = pr.GetNumber() + prHeadSha = pr.Head.GetSHA() + return true + } + } + return false + }, timeout, interval).Should(BeTrue(), fmt.Sprintf("timed out when waiting for init PaC PR (branch name '%s') to be created in %s repository", pacBranchNames[0], multiComponentRepoNameForGroupSnapshot)) + + // in case the first pipelineRun attempt has failed and was retried, we need to update the value of pipelineRun variable + pipelineRun, err = f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentA.Name, applicationName, testNamespace, prHeadSha) + Expect(err).ShouldNot(HaveOccurred()) + }) + + It("eventually leads to the build PipelineRun'sA status reported at Checks tab", func() { + expectedCheckRunName := fmt.Sprintf("%s-%s", componentA.Name, "on-pull-request") + Expect(f.AsKubeAdmin.CommonController.Github.GetCheckRunConclusion(expectedCheckRunName, multiComponentRepoNameForGroupSnapshot, prHeadSha, prNumber)).To(Equal(constants.CheckrunConclusionSuccess)) + }) + }) + + When("the Build PLRA is finished successfully", func() { + It("checks if the Snapshot is created", func() { + snapshot, err = f.AsKubeDeveloper.IntegrationController.WaitForSnapshotToGetCreated("", pipelineRun.Name, componentA.Name, testNamespace) + Expect(err).ShouldNot(HaveOccurred()) + }) + + It("should find the related Integration PipelineRuns", func() { + testPipelinerun, err = f.AsKubeDeveloper.IntegrationController.WaitForIntegrationPipelineToGetStarted(integrationTestScenarioPass.Name, snapshot.Name, testNamespace) + Expect(err).ToNot(HaveOccurred()) + Expect(testPipelinerun.Labels[snapshotAnnotation]).To(ContainSubstring(snapshot.Name)) + Expect(testPipelinerun.Labels[scenarioAnnotation]).To(ContainSubstring(integrationTestScenarioPass.Name)) + }) + + It("integration pipeline should end up with success", func() { + timeout = time.Second * 600 + interval = time.Second * 1 + Eventually(func() error { + integrationPipelineRun, err := f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentA.Name, applicationName, testNamespace, "") + if err != nil { + GinkgoWriter.Printf("Integraiton PipelineRun has not been created yet for the componentA %s/%s\n", testNamespace, componentA.Name) + return err + } + if !pipelineRun.HasStarted() { + return fmt.Errorf("integration pipelinerun %s/%s hasn't started yet", integrationPipelineRun.GetNamespace(), pipelineRun.GetName()) + } + return nil + }, timeout, constants.PipelineRunPollingInterval).Should(Succeed(), fmt.Sprintf("timed out when waiting for the build PipelineRun to start for the componentA %s/%s", testNamespace, componentA.Name)) + }) + }) + + When("the Snapshot testing is completed successfully", func() { + It("should merge the init PaC PR successfully", func() { + Eventually(func() error { + mergeResult, err = f.AsKubeAdmin.CommonController.Github.MergePullRequest(multiComponentRepoNameForGroupSnapshot, prNumber) + return err + }, time.Minute).Should(BeNil(), fmt.Sprintf("error when merging PaC pull request #%d in repo %s", prNumber, multiComponentRepoNameForGroupSnapshot)) + + mergeResultSha = mergeResult.GetSHA() + GinkgoWriter.Printf("merged result sha: %s for PR #%d\n", mergeResultSha, prNumber) + }) + + }) + + /*____ + | _ \ + | |_) | + | _ < + | |_) | + |____/ */ + When("we start creation of a new Component B", func() { + It("creates the Component B successfully", func() { + componentB = createComponentWithCustomBranch(*f, testNamespace, applicationName, multiComponentContextDirs[1]+"-"+util.GenerateRandomString(6), multiComponentGitSourceURLForGroupSnapshotB, multiComponentBaseBranchName, multiComponentContextDirs[1]) + + // Recording the PaC branch names so they can cleaned in the AfterAll block + pacBranchName := constants.PaCPullRequestBranchPrefix + componentB.Name + pacBranchNames = append(pacBranchNames, pacBranchName) + }) + + It(fmt.Sprintf("triggers a Build PipelineRun for component %s", multiComponentContextDirs[1]), func() { + timeout = time.Second * 600 + interval = time.Second * 1 + Eventually(func() error { + pipelineRun, err = f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentB.Name, applicationName, testNamespace, "") + if err != nil { + GinkgoWriter.Printf("Build PipelineRun has not been created yet for the componentB %s/%s\n", testNamespace, componentB.Name) + return err + } + if !pipelineRun.HasStarted() { + return fmt.Errorf("build pipelinerun %s/%s hasn't started yet", pipelineRun.GetNamespace(), pipelineRun.GetName()) + } + return nil + }, timeout, constants.PipelineRunPollingInterval).Should(Succeed(), fmt.Sprintf("timed out when waiting for the build PipelineRun to start for the componentB %s/%s", testNamespace, componentB.Name)) + }) + + It("does not contain an annotation with a Snapshot Name", func() { + Expect(pipelineRun.Annotations[snapshotAnnotation]).To(Equal("")) + }) + + It("should lead to build PipelineRun finishing successfully", func() { + Expect(f.AsKubeAdmin.HasController.WaitForComponentPipelineToBeFinished(componentB, + "", f.AsKubeAdmin.TektonController, &has.RetryOptions{Retries: 2, Always: true}, pipelineRun)).To(Succeed()) + }) + + It(fmt.Sprintf("should lead to a PaC PR creation for component %s", multiComponentContextDirs[1]), func() { + timeout = time.Second * 300 + interval = time.Second * 1 + + Eventually(func() bool { + prs, err := f.AsKubeAdmin.CommonController.Github.ListPullRequests(multiComponentRepoNameForGroupSnapshot) + Expect(err).ShouldNot(HaveOccurred()) + + for _, pr := range prs { + if pr.Head.GetRef() == pacBranchNames[1] { + prNumber = pr.GetNumber() + prHeadSha = pr.Head.GetSHA() + return true + } + } + return false + }, timeout, interval).Should(BeTrue(), fmt.Sprintf("timed out when waiting for init PaC PR (branch name '%s') to be created in %s repository", pacBranchNames[1], multiComponentRepoNameForGroupSnapshot)) + + // in case the first pipelineRun attempt has failed and was retried, we need to update the value of pipelineRun variable + pipelineRun, err = f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentB.Name, applicationName, testNamespace, prHeadSha) + Expect(err).ShouldNot(HaveOccurred()) + }) + + It("eventually leads to the build PipelineRun's status reported at Checks tab", func() { + expectedCheckRunName := fmt.Sprintf("%s-%s", componentB.Name, "on-pull-request") + Expect(f.AsKubeAdmin.CommonController.Github.GetCheckRunConclusion(expectedCheckRunName, multiComponentRepoNameForGroupSnapshot, prHeadSha, prNumber)).To(Equal(constants.CheckrunConclusionSuccess)) + }) + }) + + When("the Build PLR is finished successfully", func() { + It("checks if the Snapshot is created", func() { + snapshot, err = f.AsKubeDeveloper.IntegrationController.WaitForSnapshotToGetCreated("", pipelineRun.Name, componentB.Name, testNamespace) + Expect(err).ShouldNot(HaveOccurred()) + }) + + It("should find the related Integration PipelineRuns", func() { + testPipelinerun, err = f.AsKubeDeveloper.IntegrationController.WaitForIntegrationPipelineToGetStarted(integrationTestScenarioPass.Name, snapshot.Name, testNamespace) + Expect(err).ToNot(HaveOccurred()) + Expect(testPipelinerun.Labels[snapshotAnnotation]).To(ContainSubstring(snapshot.Name)) + Expect(testPipelinerun.Labels[scenarioAnnotation]).To(ContainSubstring(integrationTestScenarioPass.Name)) + }) + + It("integration pipeline should end up with success", func() { + timeout = time.Second * 600 + interval = time.Second * 1 + Eventually(func() error { + integrationPipelineRun, err := f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentB.Name, applicationName, testNamespace, "") + if err != nil { + GinkgoWriter.Printf("Integraiton PipelineRun has not been created yet for the componentB %s/%s\n", testNamespace, componentB.Name) + return err + } + if !pipelineRun.HasStarted() { + return fmt.Errorf("integration pipelinerun %s/%s hasn't started yet", integrationPipelineRun.GetNamespace(), pipelineRun.GetName()) + } + return nil + }, timeout, constants.PipelineRunPollingInterval).Should(Succeed(), fmt.Sprintf("timed out when waiting for the build PipelineRun to start for the componentB %s/%s", testNamespace, componentB.Name)) + }) + }) + + When("the Snapshot testing is completed successfully", func() { + It("should merge the init PaC PR successfully", func() { + Eventually(func() error { + mergeResult, err = f.AsKubeAdmin.CommonController.Github.MergePullRequest(multiComponentRepoNameForGroupSnapshot, prNumber) + return err + }, time.Minute).Should(BeNil(), fmt.Sprintf("error when merging PaC pull request #%d in repo %s", prNumber, multiComponentRepoNameForGroupSnapshot)) + + mergeResultSha = mergeResult.GetSHA() + GinkgoWriter.Printf("merged result sha: %s for PR #%d\n", mergeResultSha, prNumber) + }) + + }) + /* _____ + // / ____| + // | | + // | | + // | |____ + // \_____|*/ + + When("we start creation of a new Component C", func() { + It("creates the Component C successfully", func() { + componentC = createComponentWithCustomBranch(*f, testNamespace, applicationName, componentRepoNameForGeneralIntegration+"-"+util.GenerateRandomString(6), componentGitSourceURLForGeneralIntegration, multiComponentBaseBranchName, "") + + // Recording the PaC branch names so they can cleaned in the AfterAll block + pacBranchName := constants.PaCPullRequestBranchPrefix + componentC.Name + pacBranchNames = append(pacBranchNames, pacBranchName) + }) + + It(fmt.Sprintf("triggers a Build PipelineRun for componentC %s", componentRepoNameForGeneralIntegration), func() { + timeout = time.Second * 900 + interval = time.Second * 1 + Eventually(func() error { + pipelineRun, err = f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentC.Name, applicationName, testNamespace, "") + if err != nil { + GinkgoWriter.Printf("Build PipelineRun has not been created yet for the componentC %s/%s\n", testNamespace, componentC.Name) + return err + } + if !pipelineRun.HasStarted() { + return fmt.Errorf("build pipelinerun %s/%s hasn't started yet", pipelineRun.GetNamespace(), pipelineRun.GetName()) + } + return nil + }, timeout, constants.PipelineRunPollingInterval).Should(Succeed(), fmt.Sprintf("timed out when waiting for the build PipelineRun to start for the componentC %s/%s", testNamespace, componentC.Name)) + }) + + It("does not contain an annotation with a Snapshot Name", func() { + Expect(pipelineRun.Annotations[snapshotAnnotation]).To(Equal("")) + }) + + It("should lead to build PipelineRun finishing successfully", func() { + Expect(f.AsKubeAdmin.HasController.WaitForComponentPipelineToBeFinished(componentC, + "", f.AsKubeAdmin.TektonController, &has.RetryOptions{Retries: 2, Always: true}, pipelineRun)).To(Succeed()) + }) + + It(fmt.Sprintf("should lead to a PaC PR creation for componentC %s", componentRepoNameForGeneralIntegration), func() { + timeout = time.Second * 300 + interval = time.Second * 1 + + Eventually(func() bool { + prs, err := f.AsKubeAdmin.CommonController.Github.ListPullRequests(componentRepoNameForGeneralIntegration) + Expect(err).ShouldNot(HaveOccurred()) + + for _, pr := range prs { + if pr.Head.GetRef() == pacBranchNames[2] { + prNumber = pr.GetNumber() + prHeadSha = pr.Head.GetSHA() + return true + } + } + return false + }, timeout, interval).Should(BeTrue(), fmt.Sprintf("timed out when waiting for init PaC PR (branch name '%s') to be created in %s repository", pacBranchNames[2], componentRepoNameForGeneralIntegration)) + + // in case the first pipelineRun attempt has failed and was retried, we need to update the value of pipelineRun variable + pipelineRun, err = f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentC.Name, applicationName, testNamespace, prHeadSha) + Expect(err).ShouldNot(HaveOccurred()) + }) + + It("eventually leads to the build PipelineRun's status reported at Checks tab", func() { + expectedCheckRunName := fmt.Sprintf("%s-%s", componentC.Name, "on-pull-request") + Expect(f.AsKubeAdmin.CommonController.Github.GetCheckRunConclusion(expectedCheckRunName, componentRepoNameForGeneralIntegration, prHeadSha, prNumber)).To(Equal(constants.CheckrunConclusionSuccess)) + }) + }) + + When("the Build PLR is finished successfully", func() { + It("checks if the Snapshot is created", func() { + snapshot, err = f.AsKubeDeveloper.IntegrationController.WaitForSnapshotToGetCreated("", pipelineRun.Name, componentC.Name, testNamespace) + Expect(err).ShouldNot(HaveOccurred()) + }) + + It("should find the related Integration PipelineRuns", func() { + testPipelinerun, err = f.AsKubeDeveloper.IntegrationController.WaitForIntegrationPipelineToGetStarted(integrationTestScenarioPass.Name, snapshot.Name, testNamespace) + Expect(err).ToNot(HaveOccurred()) + Expect(testPipelinerun.Labels[snapshotAnnotation]).To(ContainSubstring(snapshot.Name)) + Expect(testPipelinerun.Labels[scenarioAnnotation]).To(ContainSubstring(integrationTestScenarioPass.Name)) + }) + + It("integration pipeline should end up with success", func() { + timeout = time.Second * 600 + interval = time.Second * 1 + Eventually(func() error { + integrationPipelineRun, err := f.AsKubeAdmin.HasController.GetComponentPipelineRun(componentC.Name, applicationName, testNamespace, "") + if err != nil { + GinkgoWriter.Printf("Integraiton PipelineRun has not been created yet for the componentC %s/%s\n", testNamespace, componentC.Name) + return err + } + if !pipelineRun.HasStarted() { + return fmt.Errorf("integration pipelinerun %s/%s hasn't started yet", integrationPipelineRun.GetNamespace(), pipelineRun.GetName()) + } + return nil + }, timeout, constants.PipelineRunPollingInterval).Should(Succeed(), fmt.Sprintf("timed out when waiting for the build PipelineRun to start for the componentC %s/%s", testNamespace, componentC.Name)) + }) + }) + + When("the Snapshot testing is completed successfully", func() { + It("should merge the init PaC PR successfully", func() { + Eventually(func() error { + mergeResult, err = f.AsKubeAdmin.CommonController.Github.MergePullRequest(componentRepoNameForGeneralIntegration, prNumber) + return err + }, time.Minute).Should(BeNil(), fmt.Sprintf("error when merging PaC pull request #%d in repo %s", prNumber, componentRepoNameForGeneralIntegration)) + + mergeMultiResultSha = mergeResult.GetSHA() + GinkgoWriter.Printf("merged result sha: %s for PR #%d\n", mergeMultiResultSha, prNumber) + }) + + }) + + // ______ _____ _ _ + // | ____|_ _| \ | | + // | |__ | | | \| | + // | __| | | | . ` | + // | | _| |_| |\ | + // |_| |_____|_| \_| + + When("both the init PaC PRs are merged", func() { + It("should make change to the root folder", func() { + + //Create the ref, add the files and create the PR - monorepo + err = f.AsKubeAdmin.CommonController.Github.CreateRef(multiComponentRepoNameForGroupSnapshot, multiComponentDefaultBranch, mergeResultSha, multiComponentPRBranchName) + Expect(err).ShouldNot(HaveOccurred()) + + fileToCreatePathForCompA := fmt.Sprintf("%s/sample-file-for-componentA.txt", multiComponentContextDirs[0]) + _, err := f.AsKubeAdmin.CommonController.Github.CreateFile(multiComponentRepoNameForGroupSnapshot, fileToCreatePathForCompA, "Sleep is for weak, and I'm weak", multiComponentPRBranchName) + Expect(err).ShouldNot(HaveOccurred(), fmt.Sprintf("error while creating file: %s", fileToCreatePathForCompA)) + + fileToCreatePathForCompB := fmt.Sprintf("%s/sample-file-for-componentB.txt", multiComponentContextDirs[1]) + createdFileSha, err := f.AsKubeAdmin.CommonController.Github.CreateFile(multiComponentRepoNameForGroupSnapshot, fileToCreatePathForCompB, "Sometimes I drink water to surprise my liver", multiComponentPRBranchName) + Expect(err).ShouldNot(HaveOccurred(), fmt.Sprintf("error while creating file: %s", fileToCreatePathForCompB)) + + pr, err := f.AsKubeAdmin.CommonController.Github.CreatePullRequest(multiComponentRepoNameForGroupSnapshot, "SingleRepo multi-component PR", "sample pr body", multiComponentPRBranchName, multiComponentBaseBranchName) + Expect(err).ShouldNot(HaveOccurred()) + GinkgoWriter.Printf("PR #%d got created with sha %s\n", pr.GetNumber(), createdFileSha.GetSHA()) + }) + It("should make change to the multiple-repo", func() { + // Delete all the pipelineruns in the namespace before sending PR + //Expect(f.AsKubeAdmin.TektonController.DeleteAllPipelineRunsInASpecificNamespace(testNamespace)).To(Succeed()) + + //Create the ref, add the files and create the PR - multirepo + err = f.AsKubeAdmin.CommonController.Github.CreateRef(componentRepoNameForGeneralIntegration, multiComponentDefaultBranch, mergeMultiResultSha, multiComponentPRBranchName) + Expect(err).ShouldNot(HaveOccurred()) + + fileToCreatePathForCompC := fmt.Sprintf("%s/sample-file-for-componentC.txt", componentC.Name) + createdFileSha, err := f.AsKubeAdmin.CommonController.Github.CreateFile(componentRepoNameForGeneralIntegration, fileToCreatePathForCompC, "People say nothing is impossible, but I do nothing every day", multiComponentPRBranchName) + Expect(err).ShouldNot(HaveOccurred(), fmt.Sprintf("error while creating file in multirepo: %s", fileToCreatePathForCompC)) + + pr, err := f.AsKubeAdmin.CommonController.Github.CreatePullRequest(componentRepoNameForGeneralIntegration, "Multirepo component PR", "sample pr body", multiComponentPRBranchName, multiComponentBaseBranchName) + Expect(err).ShouldNot(HaveOccurred()) + GinkgoWriter.Printf("PR #%d got created with sha %s\n", pr.GetNumber(), createdFileSha.GetSHA()) + }) + It("wait for the last components build to finish", func() { + componentNames = []string{componentA.Name, componentB.Name, componentC.Name} + for _, component := range componentNames { + Expect(f.AsKubeDeveloper.IntegrationController.WaitForBuildPipelineToBeFinished(testNamespace, applicationName, component)).To(Succeed()) + } + }) + + It("get all group snapshots and check if pr-group annotation contains all components", func() { + // get all group snapshots + Eventually(func() error { + groupSnapshots, err = f.AsKubeAdmin.HasController.GetAllGroupSnapshotsForApplication(applicationName, testNamespace) + + if groupSnapshots == nil { + GinkgoWriter.Println("No group snapshot exists at the moment: %v", err) + return err + } + if err != nil { + GinkgoWriter.Println("failed to get all group snapshots: %v", err) + return err + } + return nil + }, time.Minute*20, constants.PipelineRunPollingInterval).Should(Succeed(), "timeout while waiting for group snapshot") + + // check annotation test.appstudio.openshift.io/group-test-info for each group snapshot + annotation := groupSnapshots.Items[0].GetAnnotations() + if annotation, ok := annotation[testGroupSnapshotAnnotation]; ok { + // konflux-test + Expect(annotation).To(ContainSubstring(componentRepoNameForGeneralIntegration)) + // go-component + Expect(annotation).To(ContainSubstring(multiComponentContextDirs[0])) + // python-compomnent + Expect(annotation).To(ContainSubstring(multiComponentContextDirs[1])) + } + + }) + It("make sure that group snapshot contains last build pipelinerun for each component", func() { + for _, component := range componentNames { + pipelineRun, err = f.AsKubeDeveloper.IntegrationController.GetBuildPipelineRun(component, applicationName, testNamespace, false, "") + Expect(err).ShouldNot(HaveOccurred()) + annotation := groupSnapshots.Items[0].GetAnnotations() + if annotation, ok := annotation[testGroupSnapshotAnnotation]; ok { + Expect(annotation).To(ContainSubstring(pipelineRun.Name)) + } + } + }) + }) + }) +}) diff --git a/tests/integration-service/integration.go b/tests/integration-service/integration.go index 635eebf56..6240f824f 100644 --- a/tests/integration-service/integration.go +++ b/tests/integration-service/integration.go @@ -465,6 +465,34 @@ func createComponent(f framework.Framework, testNamespace, applicationName, comp return originalComponent, componentName, pacBranchName, componentBaseBranchName } +func createComponentWithCustomBranch(f framework.Framework, testNamespace, applicationName, componentName, componentRepoURL string, toBranchName string, contextDir string) *appstudioApi.Component { + // get the build pipeline bundle annotation + buildPipelineAnnotation := build.GetDockerBuildPipelineBundle() + dockerFileURL := constants.DockerFilePath + if contextDir == "" { + dockerFileURL = "Dockerfile" + } + componentObj := appstudioApi.ComponentSpec{ + ComponentName: componentName, + Application: applicationName, + Source: appstudioApi.ComponentSource{ + ComponentSourceUnion: appstudioApi.ComponentSourceUnion{ + GitSource: &appstudioApi.GitSource{ + URL: componentRepoURL, + Revision: toBranchName, + Context: contextDir, + DockerfileURL: dockerFileURL, + }, + }, + }, + } + + originalComponent, err := f.AsKubeAdmin.HasController.CreateComponent(componentObj, testNamespace, "", "", applicationName, true, utils.MergeMaps(utils.MergeMaps(constants.ComponentPaCRequestAnnotation, constants.ImageControllerAnnotationRequestPublicRepo), buildPipelineAnnotation)) + Expect(err).NotTo(HaveOccurred()) + + return originalComponent +} + func cleanup(f framework.Framework, testNamespace, applicationName, componentName string) { if !CurrentSpecReport().Failed() { Expect(f.AsKubeAdmin.HasController.DeleteApplication(applicationName, testNamespace, false)).To(Succeed())