diff --git a/.github/workflows/create-ocp-project.yaml b/.github/workflows/create-ocp-project.yaml new file mode 100644 index 00000000..7eacbf73 --- /dev/null +++ b/.github/workflows/create-ocp-project.yaml @@ -0,0 +1,108 @@ +name: Create OCP project CI + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/create-ocp-project/**' + - .github/workflows/create-ocp-project.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + TARGET_NS=sonataflow-infra + cd charts + helm install create-ocp-project create-ocp-project -n "${TARGET_NS}" + WORKFLOW_NAME=create-ocp-project + kubectl -n sonataflow-infra patch secret "${WORKFLOW_NAME}-creds" --type merge -p '{ + "data":{ + "NOTIFICATIONS_BEARER_TOKEN":"'$(kubectl get secrets -n rhdh-operator backstage-backend-auth-secret -o go-template='{{ .data.BACKEND_SECRET }}')'" + }, + "stringData":{ + "JIRA_API_TOKEN":"DUMB_TOKEN", + "OCP_API_SERVER_TOKEN":"DUMB_TOKEN" + } + }' + kubectl -n sonataflow-infra patch sonataflow create-ocp-project --type merge -p '{ + "spec": { + "podTemplate": { + "container": { + "env": [ + { + "name": "BACKSTAGE_NOTIFICATIONS_URL", + "value": "http://backstage-backstage.rhdh-operator" + }, + { + "name": "JIRA_URL", + "value": "DUMB_VALUE" + }, + { + "name": "JIRA_USERNAME", + "value": "DUMB_VALUE" + }, + { + "name": "OCP_API_SERVER_URL", + "value": "DUMB_VALUE" + }, + { + "name": "OCP_CONSOLE_URL", + "value": "DUMB_VALUE" + } + ] + } + } + } + } + ' + kubectl -n ${TARGET_NS} wait --for=condition=Ready=true pods -l app="${WORKFLOW_NAME}" --timeout=1m + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep create-ocp-project-analysis)" --type="NodePort" --port=8080 --name=create-ocp-project-svc + kubectl port-forward svc/create-ocp-project-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/create-ocp-project' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/extendable-workflow.yaml b/.github/workflows/extendable-workflow.yaml new file mode 100644 index 00000000..7cdf3ddc --- /dev/null +++ b/.github/workflows/extendable-workflow.yaml @@ -0,0 +1,68 @@ +name: Extendable Workflow CI + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/extendable-workflow/**' + - .github/workflows/extendable-workflow.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + TARGET_NS=sonataflow-infra + cd charts + helm install extendable-workflow extendable-workflow -n "${TARGET_NS}" + WORKFLOW_NAME=extendable-workflow + kubectl -n ${TARGET_NS} wait --for=condition=Ready=true pods -l app="${WORKFLOW_NAME}" --timeout=1m + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep extendable-workflow-analysis)" --type="NodePort" --port=8080 --name=extendable-workflow-svc + kubectl port-forward svc/extendable-workflow-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/extendable-workflow' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/greeting.yaml b/.github/workflows/greeting.yaml new file mode 100644 index 00000000..0f929e80 --- /dev/null +++ b/.github/workflows/greeting.yaml @@ -0,0 +1,68 @@ +name: Greeting CI + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/greeting/**' + - .github/workflows/greeting.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + TARGET_NS=sonataflow-infra + cd charts + helm install greeting greeting -n "${TARGET_NS}" + WORKFLOW_NAME=greeting + kubectl -n ${TARGET_NS} wait --for=condition=Ready=true pods -l app="${WORKFLOW_NAME}" --timeout=1m + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep greeting-analysis)" --type="NodePort" --port=8080 --name=greeting-svc + kubectl port-forward svc/greeting-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/greeting' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/modify-vm-resources.yaml b/.github/workflows/modify-vm-resources.yaml new file mode 100644 index 00000000..ee745b43 --- /dev/null +++ b/.github/workflows/modify-vm-resources.yaml @@ -0,0 +1,108 @@ +name: Modify VM Resources CI + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/modify-vm-resources/**' + - .github/workflows/modify-vm-resources.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + TARGET_NS=sonataflow-infra + cd charts + helm install modify-vm-resources modify-vm-resources -n "${TARGET_NS}" + WORKFLOW_NAME=modify-vm-resources + kubectl -n sonataflow-infra patch secret "${WORKFLOW_NAME}-creds" --type merge -p '{ + "data":{ + "NOTIFICATIONS_BEARER_TOKEN":"'$(kubectl get secrets -n rhdh-operator backstage-backend-auth-secret -o go-template='{{ .data.BACKEND_SECRET }}')'" + }, + "stringData":{ + "JIRA_API_TOKEN":"DUMB_TOKEN", + "OCP_API_SERVER_TOKEN":"DUMB_TOKEN" + } + }' + kubectl -n sonataflow-infra patch sonataflow modify-vm-resources --type merge -p '{ + "spec": { + "podTemplate": { + "container": { + "env": [ + { + "name": "BACKSTAGE_NOTIFICATIONS_URL", + "value": "http://backstage-backstage.rhdh-operator" + }, + { + "name": "JIRA_URL", + "value": "DUMB_VALUE" + }, + { + "name": "JIRA_USERNAME", + "value": "DUMB_VALUE" + }, + { + "name": "OCP_API_SERVER_URL", + "value": "DUMB_VALUE" + }, + { + "name": "OCP_CONSOLE_URL", + "value": "DUMB_VALUE" + } + ] + } + } + } + } + ' + kubectl -n ${TARGET_NS} wait --for=condition=Ready=true pods -l app="${WORKFLOW_NAME}" --timeout=1m + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep modify-vm-resources-analysis)" --type="NodePort" --port=8080 --name=modify-vm-resources-svc + kubectl port-forward svc/modify-vm-resources-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/modify-vm-resources' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/move2kube-e2e.yaml b/.github/workflows/move2kube-e2e.yaml new file mode 100644 index 00000000..34d4afaa --- /dev/null +++ b/.github/workflows/move2kube-e2e.yaml @@ -0,0 +1,66 @@ +name: Move2kube Workflow end to end tests + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/move2kube/**' + - .github/workflows/move2kube-e2e.yaml + - e2e/move2kube.sh + +jobs: + run-m2k-e2e: + runs-on: ubuntu-24.04 + steps: + - uses: actions/checkout@v4 + + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + - name: Create sonataflow-infra namespace + run: | + # Needs to knative events resources in this namespace as the broker url is hard coded at the moment + kubectl create ns sonataflow-infra + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + kubectl set env deployment/orchestrator-backstage LOG_LEVEL=DEBUG + + - name: Deploy Move2kube serverless workflow and its components + env: + SSH_PUB_KEY: ${{secrets.SSH_PUB_KEY}} + SSH_PRIV_KEY: ${{secrets.SSH_PRIV_KEY}} + run: | + echo "${SSH_PUB_KEY}" >> id_rsa.pub + echo "${SSH_PRIV_KEY}" >> id_rsa + PRIV_ID_RSA_PATH=id_rsa PUB_ID_RSA_PATH=id_rsa.pub M2K_HELM_REPO=charts/move2kube/ sh docs/main/move2kube/install_m2k.sh + + - name: Run e2e script + run: | + export BACKEND_SECRET=$(kubectl get secret orchestrator-auth -o jsonpath={.data.backend-secret} | base64 -d) + e2e/move2kube.sh + + - name: Export kind Logs + if: always() + run: | + kubectl get pods + kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of th previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/mta-v7.x-e2e.yaml b/.github/workflows/mta-v7.x-e2e.yaml new file mode 100644 index 00000000..196369c6 --- /dev/null +++ b/.github/workflows/mta-v7.x-e2e.yaml @@ -0,0 +1,64 @@ +name: MTA v7.x Workflow end to end tests + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/mta-v7.x/**' + - .github/workflows/mta-v7.x-e2e.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + MTA_HELM_REPO=charts/mta-v7.x/ sh docs/main/mta-v7.x/install-mta-v7.sh + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep mta-analysis)" --type="NodePort" --port=8080 --name=mta-svc + kubectl port-forward svc/mta-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/mta-analysis-v7' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/mtv-migration.yaml b/.github/workflows/mtv-migration.yaml new file mode 100644 index 00000000..f8cb3d46 --- /dev/null +++ b/.github/workflows/mtv-migration.yaml @@ -0,0 +1,95 @@ +name: MTV Migration CI + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/mtv-migration/**' + - .github/workflows/mtv-migration.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + TARGET_NS=sonataflow-infra + cd charts + helm install mtv-migration mtv-migration -n "${TARGET_NS}" + WORKFLOW_NAME=mtv-migration + kubectl -n sonataflow-infra patch secret "${WORKFLOW_NAME}-creds" --type merge -p '{ + "data":{ + "NOTIFICATIONS_BEARER_TOKEN":"'$(kubectl get secrets -n rhdh-operator backstage-backend-auth-secret -o go-template='{{ .data.BACKEND_SECRET }}')'" + }, + "stringData":{ + "OCP_API_SERVER_TOKEN":"DUMB_TOKEN" + } + }' + kubectl -n sonataflow-infra patch sonataflow mtv-migration --type merge -p '{ + "spec": { + "podTemplate": { + "container": { + "env": [ + { + "name": "BACKSTAGE_NOTIFICATIONS_URL", + "value": "http://backstage-backstage.rhdh-operator" + }, + { + "name": "OCP_API_SERVER_URL", + "value": "DUMB_VALUE" + } + ] + } + } + } + } + ' + kubectl -n ${TARGET_NS} wait --for=condition=Ready=true pods -l app="${WORKFLOW_NAME}" --timeout=1m + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep mtv-migration-analysis)" --type="NodePort" --port=8080 --name=mtv-migration-svc + kubectl port-forward svc/mtv-migration-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/mtv-migration' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/mtv-plan.yaml b/.github/workflows/mtv-plan.yaml new file mode 100644 index 00000000..1a900d13 --- /dev/null +++ b/.github/workflows/mtv-plan.yaml @@ -0,0 +1,95 @@ +name: MTV Plan CI + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/mtv-plan/**' + - .github/workflows/mtv-plan.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + TARGET_NS=sonataflow-infra + cd charts + helm install mtv-plan mtv-plan -n "${TARGET_NS}" + WORKFLOW_NAME=mtv-plan + kubectl -n sonataflow-infra patch secret "${WORKFLOW_NAME}-creds" --type merge -p '{ + "data":{ + "NOTIFICATIONS_BEARER_TOKEN":"'$(kubectl get secrets -n rhdh-operator backstage-backend-auth-secret -o go-template='{{ .data.BACKEND_SECRET }}')'" + }, + "stringData":{ + "OCP_API_SERVER_TOKEN":"DUMB_TOKEN" + } + }' + kubectl -n sonataflow-infra patch sonataflow mtv-plan --type merge -p '{ + "spec": { + "podTemplate": { + "container": { + "env": [ + { + "name": "BACKSTAGE_NOTIFICATIONS_URL", + "value": "http://backstage-backstage.rhdh-operator" + }, + { + "name": "OCP_API_SERVER_URL", + "value": "DUMB_VALUE" + } + ] + } + } + } + } + ' + kubectl -n ${TARGET_NS} wait --for=condition=Ready=true pods -l app="${WORKFLOW_NAME}" --timeout=1m + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep mtv-plan-analysis)" --type="NodePort" --port=8080 --name=mtv-plan-svc + kubectl port-forward svc/mtv-plan-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/mtv-plan' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/.github/workflows/request-vm-cnv.yaml b/.github/workflows/request-vm-cnv.yaml new file mode 100644 index 00000000..d5fce373 --- /dev/null +++ b/.github/workflows/request-vm-cnv.yaml @@ -0,0 +1,108 @@ +name: Request VM CNV CI + +on: + workflow_dispatch: + pull_request: + branches: + - main + paths: + - 'charts/request-vm-cnv/**' + - .github/workflows/request-vm-cnv.yaml + +jobs: + run-e2e: + runs-on: ubuntu-24.04 + steps: + - name: Create k8s Kind Cluster + uses: helm/kind-action@v1.10.0 + with: + cluster_name: kind + + - name: Install Operators Support + run: | + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/crds.yaml + # give the apiserver time + sleep 5s + kubectl apply -f https://raw.githubusercontent.com/operator-framework/operator-lifecycle-manager/master/deploy/upstream/quickstart/olm.yaml + + - name: Deploy sonataflow-operator + run: | + helm repo add orchestrator https://parodos-dev.github.io/orchestrator-helm-chart + helm install orchestrator orchestrator/orchestrator-k8s + + kubectl get sfp -A + kubectl wait --for=condition=Ready=true pods -l "app.kubernetes.io/name=backstage" --timeout=10m + kubectl get pods -o wide + kubectl wait --for=condition=Ready=true pods -l "app=sonataflow-platform" --timeout=180s + + - name: Deploy MTA serverless workflow and its components + run: | + TARGET_NS=sonataflow-infra + cd charts + helm install request-vm-cnv request-vm-cnv -n "${TARGET_NS}" + WORKFLOW_NAME=request-vm-cnv + kubectl -n sonataflow-infra patch secret "${WORKFLOW_NAME}-creds" --type merge -p '{ + "data":{ + "NOTIFICATIONS_BEARER_TOKEN":"'$(kubectl get secrets -n rhdh-operator backstage-backend-auth-secret -o go-template='{{ .data.BACKEND_SECRET }}')'" + }, + "stringData":{ + "JIRA_API_TOKEN":"DUMB_TOKEN", + "OCP_API_SERVER_TOKEN":"DUMB_TOKEN" + } + }' + kubectl -n sonataflow-infra patch sonataflow request-vm-cnv --type merge -p '{ + "spec": { + "podTemplate": { + "container": { + "env": [ + { + "name": "BACKSTAGE_NOTIFICATIONS_URL", + "value": "http://backstage-backstage.rhdh-operator" + }, + { + "name": "JIRA_URL", + "value": "DUMB_VALUE" + }, + { + "name": "JIRA_USERNAME", + "value": "DUMB_VALUE" + }, + { + "name": "OCP_API_SERVER_URL", + "value": "DUMB_VALUE" + }, + { + "name": "OCP_CONSOLE_URL", + "value": "DUMB_VALUE" + } + ] + } + } + } + } + ' + kubectl -n ${TARGET_NS} wait --for=condition=Ready=true pods -l app="${WORKFLOW_NAME}" --timeout=1m + + - uses: actions/checkout@v4 + - name: Test workflow is responsive + run: | + kubectl expose "$(kubectl get pod -o name | grep request-vm-cnv-analysis)" --type="NodePort" --port=8080 --name=request-vm-cnv-svc + kubectl port-forward svc/request-vm-cnv-svc 8080:8080 & + status_code=$(curl s -o /dev/null -w '%{http_code}' -XGET --location 'http://localhost:8080/request-vm-cnv' --header 'Accept: application/json, text/plain, */*' --header 'Content-Type: application/json') + if [ "$status_code" -ne 200 ]; then; + echo "$status_code" + exit 1 + fi + + - name: Export kind Logs + if: always() + run: kind export logs ./kind_logs + + - name: Upload Kind Logs + uses: actions/upload-artifact@v4 + # Always run this, even if one of the previous steps failed. + if: always() + with: + name: kind-logs + path: ./kind_logs/ + diff --git a/e2e/move2kube.sh b/e2e/move2kube.sh new file mode 100755 index 00000000..caec1279 --- /dev/null +++ b/e2e/move2kube.sh @@ -0,0 +1,237 @@ +#!/bin/bash + +set -x +set -e + +# holds the pid of the port forward process for cleanups +export port_forward_pid="" + +function cleanup() { + echo "cleanup $?" + kill "$port_forward_pid" || true + kill "$move2kube_port_forward_pid" || true +} + +function getAllNotifications() { + GUEST_TOKEN=$(curl $BACKSTAGE_URL/api/auth/guest/refresh | jq -r .backstageIdentity.token) + curl -s -H "Authorization: Bearer ${GUEST_TOKEN}" "${BACKSTAGE_NOTIFICATION_URL}" | jq ".notifications" +} + +trap 'cleanup' EXIT SIGTERM + +echo "Proxy Janus-idp port ⏳" +kubectl port-forward "$(kubectl get svc -l app.kubernetes.io/component=backstage -o name)" 9080:7007 & +port_forward_pid="$!" +sleep 3 +echo "Proxy Janus-idp port ✅" + +echo "Proxy move2kube instance port ⏳" +kubectl port-forward svc/move2kube-instance-svc 8080:8080 & +move2kube_port_forward_pid="$!" +sleep 3 +echo "Proxy move2kube instance port ✅" + + +echo "End to end tests start ⏳" +MOVE2KUBE_URL="http://localhost:8080" +BACKSTAGE_URL="http://localhost:9080" +BACKSTAGE_NOTIFICATION_URL="${BACKSTAGE_URL}/api/notifications/" +GIT_ORG="gfarache31/m2k-test" +GIT_REPO="bitbucket.org/${GIT_ORG}" +GIT_SOURCE_BRANCH="master" +GIT_TARGET_BRANCH="e2e-test-$(date +%s)" +echo "Creating workspace and project in move2kube instance" +WORKSPACE_ID=$(curl -X POST "${MOVE2KUBE_URL}/api/v1/workspaces" -H 'Content-Type: application/json' --data '{"name": "e2e Workspace", "description": "e2e tests"}' | jq -r .id) +PROJECT_ID=$(curl -X POST "${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects" -H 'Content-Type: application/json' --data '{"name": "e2e Project", "description": "e2e tests"}' | jq -r .id) + +echo "Wait until M2K workflow is available in backstage..." +M2K_STATUS=$(curl -XGET -s -o /dev/null -w "%{http_code}" -H "Authorization: Bearer ${BACKEND_SECRET}" ${BACKSTAGE_URL}/api/orchestrator/v2/workflows/m2k/overview) +until [ "$M2K_STATUS" -eq 200 ] +do +sleep 5 +M2K_STATUS=$(curl -XGET -s -o /dev/null -w "%{http_code}" -H "Authorization: Bearer ${BACKEND_SECRET}" ${BACKSTAGE_URL}/api/orchestrator/v2/workflows/m2k/overview) +done + +echo "M2K is available in backstage, sending execution request" +out=$(curl -XPOST -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" \ + ${BACKSTAGE_URL}/api/orchestrator/v2/workflows/m2k/execute \ + -d "{\"inputData\": {\"repositoryURL\": \"ssh://${GIT_REPO}\", \"recipients\": [\"user:default/guest\"], \"sourceBranch\": \"${GIT_SOURCE_BRANCH}\", \"targetBranch\": \"${GIT_TARGET_BRANCH}\", \"workspaceId\": \"${WORKSPACE_ID}\", \"projectId\": \"${PROJECT_ID}\"}}") +ID=$(echo "$out" | jq -r -e .id) + +echo "Workflow ID: ${ID}" + +if [ -z "$ID" ] || [ "$ID" == "null" ]; then + echo "workflow instance id is null... exiting " + exit 1 +fi + + +echo "Wait until plan exists" +retries=20 +http_status=$(curl -X GET -s -o /dev/null -w "%{http_code}" "${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects/${PROJECT_ID}/plan") +while [ ${retries} -ne 0 ] && [ "${http_status}" -eq 404 ]; do +echo "Wait until plan exists" + sleep 5 + retries=$((retries-1)) + http_status=$(curl -X GET -s -o /dev/null -w "%{http_code}" "${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects/${PROJECT_ID}/plan") +done + +if [ "${http_status}" -eq 204 ] +then + echo "Plan not created, error when creating it, checks move2kbe logs, http status=${http_status}...exiting " + exit 1 +fi + +if [ "${http_status}" -eq 404 ] +then + echo "Plan not created, http status=${http_status}...exiting " + exit 1 +fi + + +GUEST_TOKEN=$(curl $BACKSTAGE_URL/api/auth/guest/refresh | jq -r .backstageIdentity.token) + +echo "Checking if Q&A waiting notification with move2kube URL received" +retries=20 +while test ${retries} -ne 0 && getAllNotifications | jq -e '.|length == 0' ; do +echo "Wait until a message arrives" + sleep 5 + retries=$((retries-1)) +done + +ALL_NOTIFICATION=$(getAllNotifications) +printf "All notifications\n%s\n" "$ALL_NOTIFICATION" +if printf "%s" "$ALL_NOTIFICATION" | jq -e '.|length == 0' +then + printf "No notification found. The full reply is %s\n\nexiting " "${NOTIFICATION}" + exit 1 +fi + +NOTIFICATION=$(printf "%s" "$ALL_NOTIFICATION" | jq '.[0]') +if printf "%s" "${NOTIFICATION}" | jq ".payload.link | select(contains(\"${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects/${PROJECT_ID}/outputs\"))" +then + printf "Notification has payload link with matching URL: %s\n\n" "${NOTIFICATION}" +else + printf "Notification has no payload link with matching URL: %s\n\nexiting " "${NOTIFICATION}" + exit 1 +fi + +echo "Checking if Knative function running" +nb_pods=$(kubectl get pods -l app=m2k-save-transformation-func-v1 -no-headers | wc -l) +retries=20 +while [[ ${retries} -ne 0 && ${nb_pods} -eq 0 ]]; do +echo "Wait until Knative function running" + sleep 5 + retries=$((retries-1)) + nb_pods=$(kubectl get pods -l app=m2k-save-transformation-func-v1 --no-headers | wc -l) +done + +if [[ $nb_pods -ne 1 ]] +then + echo "Knative function not running...exiting " + exit 1 +fi + +echo "Answering Q&A to continue workflow" +TRANSFORMATION_ID=$(curl "${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects/${PROJECT_ID}" | jq -r '.outputs | keys'[0]) +current_question=$(curl -X GET "${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects/${PROJECT_ID}/outputs/${TRANSFORMATION_ID}/problems/current") +question_id=$(echo "${current_question}" | jq -r '.question | fromjson | .id' | sed -r -e 's/"/\\\\\\\"/g') +default_answer=$(echo "${current_question}" | jq '.question | fromjson | .default' | sed -r -e 's/"/\\"/g' | tr '\n' ' ') +while [ "${question_id}" != "" ]; do + curl -iX POST "${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects/${PROJECT_ID}/outputs/${TRANSFORMATION_ID}/problems/current/solution" \ + -H 'Content-Type: application/json' \ + -d "{\"solution\": \"{\\\"id\\\":\\\"${question_id}\\\",\\\"answer\\\":${default_answer}}\"}" + current_question=$(curl -X GET "${MOVE2KUBE_URL}/api/v1/workspaces/${WORKSPACE_ID}/projects/${PROJECT_ID}/outputs/${TRANSFORMATION_ID}/problems/current") + question_id=$(echo "${current_question}" | jq -r '.question | fromjson | .id' | sed -r -e 's/"/\\\\\\\"/g') + default_answer=$(echo "${current_question}" | jq '.question | fromjson | .default' | sed -r -e 's/"/\\"/g' | tr '\n' ' ') +done + +echo "Checking if workflow completed successfully" + +curl -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" "${BACKSTAGE_URL}/api/orchestrator/v2/workflows/instances/${ID}" + +state=$(curl -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" "${BACKSTAGE_URL}/api/orchestrator/v2/workflows/instances/${ID}" | jq -r .instance.state) +retries=20 +while [[ ${retries} -ne 0 && "$state" != "COMPLETED" ]]; do + sleep 5 + retries=$((retries-1)) + curl -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" "${BACKSTAGE_URL}/api/orchestrator/v2/workflows/instances/${ID}" + state=$(curl -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" "${BACKSTAGE_URL}/api/orchestrator/v2/workflows/instances/${ID}" | jq -r .instance.state) +done + +if [ "$state" != "COMPLETED" ]; then + echo "workflow instance state is '${state}', should be 'COMPLETED'... exiting " + exit 1 +fi + +echo "Checking if branch ${GIT_TARGET_BRANCH} created on git repo ${GIT_REPO}" + +http_status=$(curl -X GET -L -s -o /dev/null -w "%{http_code}" "https://api.bitbucket.org/2.0/repositories/${GIT_ORG}/refs/branches/${GIT_TARGET_BRANCH}") +retries=20 +while [[ ${retries} -ne 0 && ${http_status} -eq 404 ]]; do + sleep 5 + retries=$((retries-1)) +http_status=$(curl -X GET -L -s -o /dev/null -w "%{http_code}" "https://api.bitbucket.org/2.0/repositories/${GIT_ORG}/refs/branches/${GIT_TARGET_BRANCH}") +done +if [ "${http_status}" -eq 404 ] +then + echo "Branch ${GIT_TARGET_BRANCH} not created on repo ${GIT_REPO}...exiting " + exit 1 +else + echo "Branch ${GIT_TARGET_BRANCH} successfully created on repo ${GIT_REPO}! " +fi + +echo "Checking if completion notification received" +retries=20 +while test ${retries} -ne 0 && getAllNotifications | jq -e '.|length == 1' ; do +echo "Wait until a message arrives, expecting 2 messages overall" + sleep 5 + retries=$((retries-1)) +done + +ALL_NOTIFICATION=$(getAllNotifications) +printf "All notifications\n%s\n" "$ALL_NOTIFICATION" + +if printf "%s" "$ALL_NOTIFICATION" | jq -e '.|length == 1' +then + printf "No notification with result found - expecting success or failure notification. The full reply is %s\n\nexiting " "${ALL_NOTIFICATION}" + exit 1 +fi + +NOTIFICATION=$(printf "%s" "$ALL_NOTIFICATION" | jq '.[0]') +if printf "%s" "$NOTIFICATION" | jq -e '.payload| (.severity != "high" and .severity != "critical" )' +then + printf "Notification has NO result with high or critical severuty in it: %s\n\n" "${NOTIFICATION}" +else + printf "Notification has result high or critical severity in it: %s\n\nexiting " "${NOTIFICATION}" + exit 1 +fi + + +echo "Checking that when wrong inputs parameters, the workflows ends in error" +out=$(curl -XPOST -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" \ + ${BACKSTAGE_URL}/api/orchestrator/v2/workflows/m2k/execute \ + -d "{\"inputData\": {\"repositoryURL\": \"ssh://${GIT_REPO}_WRONG\", \"recipients\": [\"user:default/guest\"], \"sourceBranch\": \"${GIT_SOURCE_BRANCH}\", \"targetBranch\": \"${GIT_TARGET_BRANCH}\", \"workspaceId\": \"${WORKSPACE_ID}\", \"projectId\": \"${PROJECT_ID}\"}}") +ID=$(echo "$out" | jq -r -e .id) + +echo "Workflow ID: ${ID}" + +if [ -z "$ID" ] || [ "$ID" == "null" ]; then + echo "workflow instance id is null... exiting " + exit 1 +fi + +state=$(curl -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" "${BACKSTAGE_URL}/api/orchestrator/v2/workflows/instances/${ID}" | jq -r .instance.state) +retries=20 +while [[ ${retries} -ne 0 && "$state" != "ERROR" ]]; do + sleep 5 + retries=$((retries-1)) + state=$(curl -H "Content-Type: application/json" -H "Authorization: Bearer ${BACKEND_SECRET}" "${BACKSTAGE_URL}/api/orchestrator/v2/workflows/instances/${ID}" | jq -r .instance.state) +done + +if [ "$state" != "ERROR" ]; then + echo "workflow instance state is '${state}', should be 'ERROR'... exiting " + exit 1 +fi + +echo "End to end tests passed ✅"