mirror of
https://github.com/wekan/wekan.git
synced 2025-12-28 13:18:49 +01:00
379 lines
13 KiB
YAML
379 lines
13 KiB
YAML
name: Deploy testing environment to EC2
|
|
|
|
on:
|
|
pull_request:
|
|
branches:
|
|
- main
|
|
workflow_dispatch:
|
|
|
|
jobs:
|
|
deploy:
|
|
runs-on: ubuntu-latest
|
|
outputs:
|
|
wekan_image_tag: ${{ steps.docker_image_build.outputs.tag }}
|
|
|
|
steps:
|
|
- name: Checkout repository(omriza5/wekan)
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Build and push docker image
|
|
id: docker_image_build
|
|
run: |
|
|
# Login to DockerHub
|
|
echo ${{ secrets.DOCKERHUB_PASSWORD }} | docker login -u ${{ secrets.DOCKERHUB_USERNAME }} --password-stdin
|
|
|
|
# Use short commit SHA (first 7 characters)
|
|
SHORT_SHA=$(echo "${{ github.sha }}" | cut -c1-7)
|
|
TAG="${SHORT_SHA}-$(date +%Y%m%d-%H%M%S)"
|
|
echo "tag=$TAG" >> $GITHUB_OUTPUT
|
|
|
|
docker build -t ${{ secrets.DOCKERHUB_USERNAME }}/wekan:$TAG .
|
|
|
|
docker push ${{ secrets.DOCKERHUB_USERNAME }}/wekan:$TAG
|
|
|
|
# Save the tag for later steps
|
|
echo "WEKAN_IMAGE_TAG=$TAG" >> $GITHUB_ENV
|
|
|
|
- name: Install Trivy
|
|
run: |
|
|
sudo apt-get update
|
|
sudo apt-get install -y wget
|
|
wget https://github.com/aquasecurity/trivy/releases/latest/download/trivy_0.50.0_Linux-64bit.deb
|
|
sudo dpkg -i trivy_0.50.0_Linux-64bit.deb
|
|
|
|
- name: Scan Wekan Docker image for vulnerabilities
|
|
env:
|
|
WEKAN_IMAGE_TAG: ${{ env.WEKAN_IMAGE_TAG }}
|
|
DOCKERHUB_USERNAME: ${{ secrets.DOCKERHUB_USERNAME }}
|
|
run: |
|
|
docker pull $DOCKERHUB_USERNAME/wekan:$WEKAN_IMAGE_TAG
|
|
trivy image $DOCKERHUB_USERNAME/wekan:$WEKAN_IMAGE_TAG || true
|
|
|
|
- name: Create .env file
|
|
run: |
|
|
echo "WEKAN_IMAGE=omriza5/wekan:${WEKAN_IMAGE_TAG}" >> .env
|
|
|
|
- name: Copy .env file to EC2
|
|
uses: appleboy/scp-action@v0.1.7
|
|
with:
|
|
host: ${{ secrets.WEKAN_EC2_HOST_IP }}
|
|
username: ubuntu
|
|
key: ${{ secrets.EC2_SSH_KEY }}
|
|
source: ".env"
|
|
target: "/home/ubuntu/"
|
|
|
|
- name: Copy docker-compose file to EC2
|
|
uses: appleboy/scp-action@v0.1.7
|
|
with:
|
|
host: ${{ secrets.WEKAN_EC2_HOST_IP }}
|
|
username: ubuntu
|
|
key: ${{ secrets.EC2_SSH_KEY }}
|
|
source: "docker-compose.yml"
|
|
target: "/home/ubuntu/"
|
|
|
|
- name: Deploy to EC2
|
|
uses: appleboy/ssh-action@v1.0.3
|
|
with:
|
|
host: ${{ secrets.WEKAN_EC2_HOST_IP }}
|
|
username: ubuntu
|
|
key: ${{ secrets.EC2_SSH_KEY }}
|
|
script: |
|
|
# Stop and remove containers with volumes
|
|
sudo docker compose down -v || true
|
|
|
|
# Clean up everything including named volumes
|
|
sudo docker volume rm $(sudo docker volume ls -q) 2>/dev/null || true
|
|
|
|
sudo docker stop $(sudo docker ps -aq) 2>/dev/null || true
|
|
sudo docker rm $(sudo docker ps -aq) 2>/dev/null || true
|
|
|
|
# Remove all images to free space
|
|
sudo docker rmi $(sudo docker images -q) 2>/dev/null || true
|
|
|
|
# Clean up networks (volumes already removed above)
|
|
sudo docker network prune -f || true
|
|
|
|
echo "${{ secrets.DOCKERHUB_PASSWORD }}" | sudo docker login -u "${{ secrets.DOCKERHUB_USERNAME }}" --password-stdin
|
|
|
|
sudo docker compose pull
|
|
sudo docker compose up -d
|
|
|
|
security-scan:
|
|
needs: deploy
|
|
runs-on: ubuntu-latest
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Install Trivy
|
|
run: |
|
|
sudo apt-get update
|
|
sudo apt-get install -y wget
|
|
wget https://github.com/aquasecurity/trivy/releases/latest/download/trivy_0.50.0_Linux-64bit.deb
|
|
sudo dpkg -i trivy_0.50.0_Linux-64bit.deb
|
|
|
|
- name: Scan Wekan Docker image for vulnerabilities
|
|
env:
|
|
WEKAN_IMAGE_TAG: ${{ needs.deploy.outputs.wekan_image_tag }}
|
|
DOCKERHUB_USERNAME: ${{ secrets.DOCKERHUB_USERNAME }}
|
|
run: |
|
|
docker pull $DOCKERHUB_USERNAME/wekan:$WEKAN_IMAGE_TAG
|
|
trivy image $DOCKERHUB_USERNAME/wekan:$WEKAN_IMAGE_TAG || true
|
|
|
|
API-tests:
|
|
needs: deploy
|
|
runs-on: ubuntu-latest
|
|
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Set up Python 3.12
|
|
uses: actions/setup-python@v4
|
|
with:
|
|
python-version: "3.12"
|
|
|
|
- name: Install dependencies
|
|
run: |
|
|
python -m pip install --upgrade pip
|
|
pip install -r requirements.txt
|
|
|
|
- name: Create test user via Database
|
|
uses: appleboy/ssh-action@v1.0.3
|
|
with:
|
|
host: ${{ secrets.WEKAN_EC2_HOST_IP }}
|
|
username: ubuntu
|
|
key: ${{ secrets.EC2_SSH_KEY }}
|
|
script: |
|
|
# Wait for Wekan to be fully ready
|
|
echo "Waiting for Wekan to start..."
|
|
for i in {1..24}; do
|
|
if curl -s http://localhost > /dev/null 2>&1; then
|
|
echo "Wekan is responding!"
|
|
break
|
|
fi
|
|
echo "Waiting... (attempt $i/24)"
|
|
sleep 5
|
|
done
|
|
|
|
# Create user directly in database with the exact structure from browser
|
|
echo "Creating test user directly in database..."
|
|
sudo docker exec wekan-db mongosh wekan --eval '
|
|
// Remove existing user first
|
|
db.users.deleteMany({username: "omriza5"});
|
|
|
|
// Create user with exact structure from browser
|
|
const result = db.users.insertOne({
|
|
_id: "omriza5_" + new Date().getTime(),
|
|
createdAt: new Date(),
|
|
services: {
|
|
password: {
|
|
bcrypt: "$2b$10$v9266B4sMuTCOgPsnIPibuxKoUwELIqPvTn7GQqGvvVibAEsmphsm"
|
|
},
|
|
email: {
|
|
verificationTokens: [
|
|
{
|
|
token: "token_" + Math.random().toString(36).substring(2),
|
|
address: "omriza5@gmail.com",
|
|
when: new Date()
|
|
}
|
|
]
|
|
}
|
|
},
|
|
username: "omriza5",
|
|
emails: [{ address: "omriza5@gmail.com", verified: false }],
|
|
isAdmin: true,
|
|
modifiedAt: new Date(),
|
|
profile: {
|
|
boardView: "board-view-swimlanes",
|
|
listSortBy: "-modifiedAt",
|
|
templatesBoardId: "",
|
|
cardTemplatesSwimlaneId: "",
|
|
listTemplatesSwimlaneId: "",
|
|
boardTemplatesSwimlaneId: "",
|
|
listWidths: {},
|
|
listConstraints: {},
|
|
autoWidthBoards: {},
|
|
swimlaneHeights: {},
|
|
keyboardShortcuts: false,
|
|
verticalScrollbars: true,
|
|
showWeekOfYear: true
|
|
},
|
|
authenticationMethod: "password",
|
|
sessionData: {}
|
|
});
|
|
|
|
if (result.acknowledged) {
|
|
print("User omriza5 created successfully");
|
|
} else {
|
|
print("Failed to create user");
|
|
}
|
|
' || echo "Failed to execute MongoDB command"
|
|
|
|
# Verify user was created
|
|
echo "Verifying user creation..."
|
|
sudo docker exec wekan-db mongosh wekan --eval 'db.users.findOne({username: "omriza5"}, {username: 1, emails: 1, isAdmin: 1})' || echo "User verification failed"
|
|
|
|
# Verify login works
|
|
echo "Testing login..."
|
|
LOGIN_RESPONSE=$(curl -s -w "HTTPSTATUS:%{http_code}" \
|
|
-H "Content-type:application/json" \
|
|
-X POST http://localhost/users/login \
|
|
-d '{"username":"omriza5","password":"123456"}')
|
|
|
|
LOGIN_CODE=$(echo $LOGIN_RESPONSE | tr -d '\n' | sed -e 's/.*HTTPSTATUS://')
|
|
if [[ "$LOGIN_CODE" == "200" ]]; then
|
|
echo "Login test successful"
|
|
else
|
|
echo "Login test failed (Code: $LOGIN_CODE)"
|
|
echo "Response: $(echo $LOGIN_RESPONSE | sed -e 's/HTTPSTATUS:.*//g')"
|
|
fi
|
|
|
|
- name: Run API tests
|
|
env:
|
|
BASE_URL: ${{ secrets.WEKAN_URL }}
|
|
run: |
|
|
pytest --maxfail=5 --disable-warnings -v
|
|
|
|
Selenium-ui-tests:
|
|
needs: deploy
|
|
runs-on: ubuntu-latest
|
|
|
|
strategy:
|
|
matrix:
|
|
browser: [chrome, firefox]
|
|
resolution:
|
|
- { name: "desktop", width: 1200, height: 800 }
|
|
- { name: "mobile", width: 800, height: 667 }
|
|
fail-fast: false # Continue other jobs even if one fails
|
|
|
|
name: Test ${{ matrix.browser }} - ${{ matrix.resolution.name }}
|
|
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
with:
|
|
fetch-depth: 1
|
|
repository: omriza5/wekan-selenium
|
|
token: ${{ secrets.UI_TESTING_GITHUB_TOKEN }}
|
|
|
|
- name: Set up Python 3.11
|
|
uses: actions/setup-python@v4
|
|
with:
|
|
python-version: "3.11"
|
|
|
|
- name: Install dependencies
|
|
run: |
|
|
python -m pip install --upgrade pip
|
|
pip install -r requirements.txt
|
|
|
|
- name: Install Chrome and ChromeDriver
|
|
if: matrix.browser == 'chrome'
|
|
run: |
|
|
sudo apt-get update
|
|
sudo apt-get install -y google-chrome-stable
|
|
|
|
- name: Install Firefox and GeckoDriver
|
|
if: matrix.browser == 'firefox'
|
|
run: |
|
|
sudo apt-get update
|
|
sudo apt-get install -y wget tar
|
|
# Remove Snap Firefox if installed
|
|
sudo snap remove firefox || true
|
|
# Download official Firefox
|
|
wget -O firefox.tar.xz "https://download.mozilla.org/?product=firefox-latest&os=linux64&lang=en-US"
|
|
tar xJf firefox.tar.xz
|
|
sudo mv firefox /opt/firefox
|
|
sudo ln -s /opt/firefox/firefox /usr/local/bin/firefox
|
|
# Install GeckoDriver
|
|
wget https://github.com/mozilla/geckodriver/releases/latest/download/geckodriver-v0.36.0-linux64.tar.gz
|
|
tar -xvzf geckodriver-v0.36.0-linux64.tar.gz
|
|
sudo mv geckodriver /usr/local/bin/
|
|
sudo chmod +x /usr/local/bin/geckodriver
|
|
|
|
- name: Wait for Wekan to be ready
|
|
env:
|
|
WEKAN_URL: ${{ secrets.WEKAN_URL }}
|
|
run: |
|
|
echo "Waiting for Wekan to be fully ready at $WEKAN_URL..."
|
|
for i in {1..30}; do
|
|
if curl -s -f "$WEKAN_URL" > /dev/null 2>&1; then
|
|
echo "Wekan is responding! (attempt $i)"
|
|
# Additional check to ensure it's fully loaded
|
|
sleep 5
|
|
if curl -s "$WEKAN_URL" | grep -q "Wekan"; then
|
|
echo "Wekan is fully ready!"
|
|
break
|
|
fi
|
|
fi
|
|
echo "Waiting for Wekan... (attempt $i/30)"
|
|
sleep 10
|
|
done
|
|
|
|
# Final verification
|
|
curl -I "$WEKAN_URL" || exit 1
|
|
|
|
- name: Run Selenium tests
|
|
env:
|
|
HEADLESS: true
|
|
BROWSER: ${{ matrix.browser }}
|
|
SCREEN_WIDTH: ${{ matrix.resolution.width }}
|
|
SCREEN_HEIGHT: ${{ matrix.resolution.height }}
|
|
WEKAN_URL: ${{ secrets.WEKAN_URL }}
|
|
ANTHROPIC_API_KEY: ${{ secrets.ANTHROPIC_API_KEY }}
|
|
ALLURE_SUITE: "${{ matrix.browser }} - ${{ matrix.resolution.name }}"
|
|
run: |
|
|
if [ -d "allure-results" ]; then rm -rf allure-results; fi
|
|
mkdir -p allure-results
|
|
cat > allure-results/environment.properties << EOF
|
|
os_platform=$(uname -s)
|
|
os_release=$(lsb_release -r | awk '{print $2}')
|
|
os_version=$(lsb_release -c | awk '{print $2}')
|
|
python_version=$(python --version | awk '{print $2}')
|
|
Wekan.URL=${{ secrets.WEKAN_URL }}
|
|
Headless.Mode=true
|
|
Wekan.Image=omriza5/wekan:${{ needs.deploy.outputs.wekan_image_tag }}
|
|
Browser=${{ matrix.browser }}
|
|
Resolution=${{ matrix.resolution.name }}
|
|
EOF
|
|
pytest --alluredir=allure-results tests/
|
|
|
|
- name: Upload Allure results as artifact
|
|
uses: actions/upload-artifact@v4
|
|
if: always() # Upload even if tests fail
|
|
with:
|
|
name: allure-results-${{ matrix.browser }}-${{ matrix.resolution.name }}
|
|
path: allure-results/
|
|
retention-days: 30
|
|
|
|
allure-report:
|
|
runs-on: ubuntu-latest
|
|
needs: Selenium-ui-tests
|
|
if: always()
|
|
|
|
steps:
|
|
- name: Checkout code
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Download all Allure results
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
pattern: allure-results-*
|
|
path: allure-results
|
|
merge-multiple: true
|
|
|
|
- name: Generate and Deploy Allure Report
|
|
uses: simple-elf/allure-report-action@master
|
|
with:
|
|
allure_results: allure-results
|
|
allure_report: allure-report
|
|
gh_pages: gh-pages
|
|
allure_history: allure-history
|
|
keep_reports: 20
|
|
|
|
- name: Deploy to GitHub Pages
|
|
uses: peaceiris/actions-gh-pages@v3
|
|
with:
|
|
github_token: ${{ secrets.UI_TESTING_GITHUB_TOKEN }}
|
|
publish_dir: allure-history
|
|
|