This is an automated email from the ASF dual-hosted git repository.

dahn pushed a commit to branch main
in repository https://gitbox.apache.org/repos/asf/cloudstack-cloudmonkey.git


The following commit(s) were added to refs/heads/main by this push:
     new ec71a4d  github-action: add ci workflow (#180)
ec71a4d is described below

commit ec71a4dfc77ec5c65788c536b96886dd9bce1798
Author: Abhishek Kumar <abhishek.mr...@gmail.com>
AuthorDate: Tue Aug 12 17:51:20 2025 +0530

    github-action: add ci workflow (#180)
---
 .github/workflows/ci.yml | 318 +++++++++++++++++++++++++++++++++++++++++++++++
 1 file changed, 318 insertions(+)

diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml
new file mode 100644
index 0000000..4fb3e13
--- /dev/null
+++ b/.github/workflows/ci.yml
@@ -0,0 +1,318 @@
+# Licensed to the Apache Software Foundation (ASF) under one
+# or more contributor license agreements.  See the NOTICE file
+# distributed with this work for additional information
+# regarding copyright ownership.  The ASF licenses this file
+# to you under the Apache License, Version 2.0 (the
+# "License"); you may not use this file except in compliance
+# with the License.  You may obtain a copy of the License at
+#
+#   http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing,
+# software distributed under the License is distributed on an
+# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+# KIND, either express or implied.  See the License for the
+# specific language governing permissions and limitations
+# under the License.
+
+name: Simulator CI
+
+on:
+  pull_request:
+
+concurrency:
+  group: ${{ github.workflow }}-${{ github.event.pull_request.number }}
+  cancel-in-progress: true
+
+permissions:
+  contents: read
+
+jobs:
+  simulator-latest-ci:
+    runs-on: ubuntu-24.04
+    timeout-minutes: 60
+
+    env:
+      GO111MODULE: on
+      CMK_BIN: bin/cmk
+      CLOUDSTACK_SIM_API: http://127.0.0.1:8096/client/api
+      CLOUDSTACK_UI_API: http://127.0.0.1:8080/client/api
+      MAVEN_OPTS: -Xmx4096m -XX:MaxMetaspaceSize=800m 
-Djava.security.egd=file:/dev/urandom 
--add-opens=java.base/java.lang=ALL-UNNAMED 
--add-exports=java.base/sun.security.x509=ALL-UNNAMED 
--add-opens=java.base/jdk.internal.reflect=ALL-UNNAMED
+
+    steps:
+      - name: Check out CloudMonkey
+        uses: actions/checkout@v4
+
+      - name: Set up Go
+        uses: actions/setup-go@v5
+        with:
+          go-version: '1.22'
+
+      - name: Build cmk
+        run: |
+          make run
+
+      - name: Fetch latest CloudStack release tag
+        id: csrel
+        run: |
+          TAG=$(curl -s 
https://api.github.com/repos/apache/cloudstack/releases/latest | jq -r 
.tag_name)
+          echo "tag=$TAG" >> $GITHUB_OUTPUT
+          echo "Latest CloudStack tag: $TAG"
+
+      - name: Clone CloudStack at latest release
+        run: |
+          git clone --depth=1 --branch "${{ steps.csrel.outputs.tag }}" 
https://github.com/apache/cloudstack.git
+
+      - name: Set up JDK 11 + Maven cache
+        uses: actions/setup-java@v4
+        with:
+          java-version: '11'
+          distribution: 'temurin'
+          cache: maven
+          cache-dependency-path: cloudstack/pom.xml
+
+      - name: Set up Python 3.10
+        uses: actions/setup-python@v5
+        with:
+          python-version: '3.10'
+
+      - name: Install OS deps
+        run: |
+          sudo apt-get update
+          sudo apt-get install -y \
+            mysql-server uuid-runtime genisoimage netcat-openbsd ipmitool \
+            build-essential libgcrypt20 libgpg-error-dev libgpg-error0 \
+            libopenipmi0 libssl-dev jq curl
+
+      - name: Install Python deps
+        run: |
+          python3 -m pip install --upgrade pip
+          python3 -m pip install --user urllib3 lxml paramiko nose texttable 
ipmisim pyopenssl pycryptodome mock flask netaddr pylint pycodestyle six 
astroid mysql-connector-python
+
+      - name: Setup MySQL Server
+        run: |
+          # 
https://github.com/actions/runner-images/blob/main/images/linux/Ubuntu2004-Readme.md#mysql
+          sudo systemctl start mysql
+          sudo mysql -uroot -proot -e "ALTER USER 'root'@'localhost' 
IDENTIFIED WITH mysql_native_password BY ''; FLUSH PRIVILEGES;"
+          sudo systemctl restart mysql
+          sudo mysql -uroot -e "SELECT VERSION();"
+
+      - name: Build CloudStack (simulator)
+        working-directory: cloudstack
+        run: |
+          mvn -B -P developer,systemvm -Dsimulator clean install 
-DskipTests=true -T$(nproc)
+
+      - name: Deploy simulator DB + marvin
+        working-directory: cloudstack
+        run: |
+          mvn -q -Pdeveloper -pl developer -Ddeploydb
+          mvn -q -Pdeveloper -pl developer -Ddeploydb-simulator
+          python3 -m pip install --user --upgrade 
tools/marvin/dist/Marvin-*.tar.gz
+
+      - name: Start CloudStack mgmt (Jetty) in background
+        id: start_ms
+        working-directory: cloudstack
+        run: |
+          set -euo pipefail
+          LOG=/tmp/jetty-log.txt
+          mvn -Dsimulator -Dorg.eclipse.jetty.annotations.maxWait=120 -pl 
:cloud-client-ui jetty:run > "$LOG" 2>&1 &
+          JETTY_PID=$!
+          echo "$JETTY_PID" > /tmp/jetty.pid
+          echo "Waiting for simulator API @ 8096…"
+          for i in $(seq 1 60); do
+            if nc -z 127.0.0.1 8096; then
+              echo "Simulator API is up."
+              break
+            fi
+            sleep 5
+            tail -n 50 "$LOG" || true
+            if ! kill -0 "$JETTY_PID" 2>/dev/null; then
+              echo "Jetty exited early. Last 200 lines:"
+              tail -n 200 "$LOG" || true
+              exit 1
+            fi
+          done
+
+      - name: Deploy Advanced Zone via Marvin
+        working-directory: cloudstack
+        run: |
+          python3 tools/marvin/marvin/deployDataCenter.py -i 
setup/dev/advdualzone.cfg
+
+      - name: Configure cmk profile
+        run: |
+          "${CMK_BIN}" set profile simulator
+          "${CMK_BIN}" set url "${CLOUDSTACK_UI_API}"
+          "${CMK_BIN}" set output json
+
+      - name: API discovery parity (curl vs cmk)
+        run: |
+          CURL_COUNT=$(curl -s 
"${CLOUDSTACK_SIM_API}?command=listApis&response=json" | jq 
'.listapisresponse.count')
+          echo "curl count: $CURL_COUNT"
+          CMK_COUNT=$("${CMK_BIN}" listApis | jq '.count')
+          echo "cmk count: $CMK_COUNT"
+          test -n "$CURL_COUNT" -a -n "$CMK_COUNT"
+          [ "$CMK_COUNT" -ge 1 ]
+
+      - name: List API
+        run: |
+          "${CMK_BIN}" set output table
+          OUT=$("${CMK_BIN}" listZones)
+          echo "$OUT"
+          if [ -z "$OUT" ]; then
+            echo "No output from listZones. Failing."
+            exit 1
+          fi
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" listZones)
+          echo "$OUT"
+          if [ -z "$OUT" ]; then
+            echo "No output from listZones. Failing."
+            exit 1
+          fi
+          ZONE_ID=$(echo "$OUT" | jq -r '.zone[0].id')
+          echo "ZONE_ID=$ZONE_ID" >> $GITHUB_ENV
+          test -n "$ZONE_ID"
+
+      - name: List API with filter
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" listZones filter=id,name)
+          echo "$OUT" | jq -e '
+            .zone and (.zone|length>=1)
+            and (all(.zone[]; has("id") and has("name") and ((. | keys - 
["id","name"])|length==0)))
+          '
+
+      - name: List API with exclude
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" listZones exclude=id,name)
+          echo "$OUT" | jq -e '
+            .zone and (.zone|length>=1)
+            and (all(.zone[]; (has("id") or has("name"))|not))
+          '
+
+      - name: "Create API"
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" createUser account=admin 
email=testu...@example.com firstname=Test lastname=User password=password 
username=test || true)
+          echo "$OUT"
+          if [ -n "$OUT" ]; then
+            echo "$OUT" | jq -e '
+              (.user // {}) as $p
+              | ($p|type=="object")
+              and ($p|has("id") and has("email"))
+              and ((($p|keys) - ["id","email"])|length>0)
+            '
+          else
+            echo "No output. Failing strict check."
+            exit 1
+          fi
+
+      - name: Create API with filter
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" createUser account=admin 
email=testu...@example.com firstname=Test lastname=User password=password 
username=test-filter filter=id,email || true)
+          echo "$OUT"
+          if [ -n "$OUT" ]; then
+            echo "$OUT" | jq -e '
+              (.user // {}) as $p
+              | ($p|type=="object")
+              and ($p|has("id") and has("email"))
+              and ((($p|keys) - ["id","email"])|length==0)
+            '
+          else
+            echo "No output. Failing strict check."
+            exit 1
+          fi
+
+      - name: Create API with exclude
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" createUser account=admin 
email=testu...@example.com firstname=Test lastname=User password=password 
username=test-exclude exclude=id,email || true)
+          echo "$OUT"
+          if [ -n "$OUT" ]; then
+            echo "$OUT" | jq -e '
+              (.user // {}) as $p
+              | ($p|type=="object")
+              and ((($p|has("id")) or ($p|has("email")))|not)
+            '
+          else
+            echo "No output. Failing strict check."
+            exit 1
+          fi
+
+      - name: Get template and service offering IDs for Async API test
+        run: |
+          TID=$("${CMK_BIN}" listTemplates listall=true 
templatefilter=executable | jq -r '.template[0].id')
+          SOID=$("${CMK_BIN}" listServiceOfferings | jq -r 
'.serviceoffering[0].id')
+          echo "TEMPLATE_ID=$TID" >> $GITHUB_ENV
+          echo "SERVICE_OFFERING_ID=$SOID" >> $GITHUB_ENV
+          test -n "$TID" -a -n "$SOID"
+
+      - name: Async API
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" deployVirtualMachine zoneid=${ZONE_ID} 
templateid=${TEMPLATE_ID} serviceofferingid=${SERVICE_OFFERING_ID} || true)
+          if [ -n "$OUT" ]; then
+            echo "$OUT" | jq -e '
+              (.virtualmachine // {}) as $p
+              | ($p|type=="object")
+              and ($p|has("id") and has("name"))
+              and ((($p|keys) - ["id","name"])|length>0)
+            '
+          else
+            echo "No output. Failing strict check."
+            exit 1
+          fi
+
+      - name: Async API with filter
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" deployVirtualMachine zoneid=${ZONE_ID} 
templateid=${TEMPLATE_ID} serviceofferingid=${SERVICE_OFFERING_ID} 
filter=id,name || true)
+          if [ -n "$OUT" ]; then
+            echo "$OUT" | jq -e '
+              (.virtualmachine // {}) as $p
+              | ($p|type=="object")
+              and ($p|has("id") and has("name"))
+              and ((($p|keys) - ["id","name"])|length==0)
+            '
+          else
+            echo "No output. Failing strict check."
+            exit 1
+          fi
+
+      - name: Async API with exclude
+        run: |
+          "${CMK_BIN}" set output json
+          OUT=$("${CMK_BIN}" deployVirtualMachine zoneid=${ZONE_ID} 
templateid=${TEMPLATE_ID} serviceofferingid=${SERVICE_OFFERING_ID} 
exclude=id,name || true)
+          echo "$OUT"
+          if [ -n "$OUT" ]; then
+            echo "$OUT" | jq -e '
+              (.virtualmachine // {}) as $p
+              | ($p|type=="object")
+              and ((($p|has("id")) or ($p|has("name")))|not)
+            '
+          else
+            echo "No output. Failing strict check."
+            exit 1
+          fi
+
+      - name: Change profile (user) and compare API surface
+        run: |
+          ADMIN_COUNT=$("${CMK_BIN}" listApis | jq '.count')
+          "${CMK_BIN}" createAccount username=user password=p@ssw0rd 
accounttype=0 domainid=1 firstname=Test lastname=User 
email=testu...@example.com || true
+          "${CMK_BIN}" set profile user
+          "${CMK_BIN}" set url "${CLOUDSTACK_UI_API}"
+          "${CMK_BIN}" set username user
+          "${CMK_BIN}" set password p@ssw0rd
+          USER_COUNT=$("${CMK_BIN}" listApis | jq '.count // 0')
+          echo "admin=${ADMIN_COUNT} user=${USER_COUNT}"
+          test $USER_COUNT -le $ADMIN_COUNT
+
+      - name: Stop simulator MS
+        if: ${{ always() && steps.start_ms.outcome == 'success' }}
+        working-directory: cloudstack
+        run: |
+            echo -e "Stopping Simulator, integration tests run completed\n"
+            mvn -Dsimulator -pl client jetty:stop 2>&1
+

Reply via email to