Skip to content

github-action: add ci workflow #180

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 1 commit into from
Aug 12, 2025
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
318 changes: 318 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
@@ -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 protected] 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 protected] 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 protected] 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 protected] || 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