mirror of
https://github.com/Azure/k8s-deploy.git
synced 2026-06-23 04:52:08 +08:00
Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 9d200da279 | |||
| df58fb461e | |||
| a999ffcd6c | |||
| 00795b0b56 | |||
| d565a17533 | |||
| 1811836de2 |
@@ -0,0 +1,18 @@
|
|||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: npm
|
||||||
|
directory: /
|
||||||
|
schedule:
|
||||||
|
interval: weekly
|
||||||
|
groups:
|
||||||
|
actions:
|
||||||
|
patterns:
|
||||||
|
- '*'
|
||||||
|
- package-ecosystem: github-actions
|
||||||
|
directory: .github/workflows
|
||||||
|
schedule:
|
||||||
|
interval: weekly
|
||||||
|
groups:
|
||||||
|
actions:
|
||||||
|
patterns:
|
||||||
|
- '*'
|
||||||
@@ -10,10 +10,13 @@ jobs:
|
|||||||
CodeQL-Build:
|
CodeQL-Build:
|
||||||
# CodeQL runs on ubuntu-latest and windows-latest
|
# CodeQL runs on ubuntu-latest and windows-latest
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
security-events: write
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 #v4.1.1
|
||||||
with:
|
with:
|
||||||
# We must fetch at least the immediate parents so that if this is
|
# We must fetch at least the immediate parents so that if this is
|
||||||
# a pull request then we can checkout the head.
|
# a pull request then we can checkout the head.
|
||||||
@@ -21,7 +24,7 @@ jobs:
|
|||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v2
|
uses: github/codeql-action/init@05963f47d870e2cb19a537396c1f668a348c7d8f #v3.24.8
|
||||||
# Override language selection by uncommenting this and choosing your languages
|
# Override language selection by uncommenting this and choosing your languages
|
||||||
# with:
|
# with:
|
||||||
# languages: go, javascript, csharp, python, cpp, java
|
# languages: go, javascript, csharp, python, cpp, java
|
||||||
@@ -29,7 +32,7 @@ jobs:
|
|||||||
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
||||||
# If this step fails, then you should remove it and run the build manually (see below)
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
- name: Autobuild
|
- name: Autobuild
|
||||||
uses: github/codeql-action/autobuild@v2
|
uses: github/codeql-action/autobuild@05963f47d870e2cb19a537396c1f668a348c7d8f #v3.24.8
|
||||||
|
|
||||||
# ℹ️ Command-line programs to run using the OS shell.
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
# 📚 https://git.io/JvXDl
|
# 📚 https://git.io/JvXDl
|
||||||
@@ -43,4 +46,4 @@ jobs:
|
|||||||
# make release
|
# make release
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v2
|
uses: github/codeql-action/analyze@05963f47d870e2cb19a537396c1f668a348c7d8f #v3.24.8
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
name: release Project
|
name: Release Project
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
@@ -10,6 +10,9 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
release:
|
||||||
uses: Azure/action-release-workflows/.github/workflows/release_js_project.yaml@e4a1a0385530d6861c9a9b7262058ad33b10c769
|
permissions:
|
||||||
|
actions: read
|
||||||
|
contents: write
|
||||||
|
uses: Azure/action-release-workflows/.github/workflows/release_js_project.yaml@v1
|
||||||
with:
|
with:
|
||||||
changelogPath: ./CHANGELOG.md
|
changelogPath: ./CHANGELOG.md
|
||||||
|
|||||||
@@ -64,9 +64,13 @@ jobs:
|
|||||||
images: nginx:1.14.2
|
images: nginx:1.14.2
|
||||||
manifests: |
|
manifests: |
|
||||||
test/integration/manifests/test.yml
|
test/integration/manifests/test.yml
|
||||||
|
test/integration/manifests/manifest_test_dir/test.yml
|
||||||
action: deploy
|
action: deploy
|
||||||
|
|
||||||
- name: Checking if deployments and services were created
|
- name: Checking if deployments and services were created
|
||||||
run: |
|
run: |
|
||||||
python test/integration/k8s-deploy-test.py namespace=${{ env.NAMESPACE }} kind=Deployment name=nginx-deployment containerName=nginx:1.14.2 labels=app:nginx,workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Minikube_Integration_Tests_-_basic selectorLabels=app:nginx
|
python test/integration/k8s-deploy-test.py namespace=${{ env.NAMESPACE }} kind=Deployment name=nginx-deployment containerName=nginx:1.14.2 labels=app:nginx,workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Minikube_Integration_Tests_-_basic selectorLabels=app:nginx
|
||||||
python test/integration/k8s-deploy-test.py namespace=${{ env.NAMESPACE }} kind=Service name=nginx-service labels=workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Minikube_Integration_Tests_-_basic selectorLabels=app:nginx
|
python test/integration/k8s-deploy-test.py namespace=${{ env.NAMESPACE }} kind=Service name=nginx-service labels=workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Minikube_Integration_Tests_-_basic selectorLabels=app:nginx
|
||||||
|
|
||||||
|
python test/integration/k8s-deploy-test.py namespace=${{ env.NAMESPACE }} kind=Deployment name=nginx-deployment3 containerName=nginx:1.14.2 labels=app:nginx3,workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Minikube_Integration_Tests_-_basic selectorLabels=app:nginx3
|
||||||
|
python test/integration/k8s-deploy-test.py namespace=${{ env.NAMESPACE }} kind=Service name=nginx-service3 labels=workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Minikube_Integration_Tests_-_basic selectorLabels=app:nginx3
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
set +x
|
set +x
|
||||||
# create cluster
|
# create cluster
|
||||||
az group create --location eastus --name ${{ env.NAMESPACE }}
|
az group create --location eastus2 --name ${{ env.NAMESPACE }}
|
||||||
az aks create --name ${{ env.NAMESPACE }} --resource-group ${{ env.NAMESPACE }} --enable-private-cluster --generate-ssh-keys
|
az aks create --name ${{ env.NAMESPACE }} --resource-group ${{ env.NAMESPACE }} --enable-private-cluster --generate-ssh-keys
|
||||||
az aks get-credentials --resource-group ${{ env.NAMESPACE }} --name ${{ env.NAMESPACE }}
|
az aks get-credentials --resource-group ${{ env.NAMESPACE }} --name ${{ env.NAMESPACE }}
|
||||||
|
|
||||||
@@ -63,6 +63,7 @@ jobs:
|
|||||||
images: nginx:1.14.2
|
images: nginx:1.14.2
|
||||||
manifests: |
|
manifests: |
|
||||||
test/integration/manifests/test.yml
|
test/integration/manifests/test.yml
|
||||||
|
test/integration/manifests/test2.yml
|
||||||
action: deploy
|
action: deploy
|
||||||
private-cluster: true
|
private-cluster: true
|
||||||
resource-group: ${{ env.NAMESPACE }}
|
resource-group: ${{ env.NAMESPACE }}
|
||||||
@@ -73,6 +74,9 @@ jobs:
|
|||||||
python test/integration/k8s-deploy-test.py private=${{ env.NAMESPACE }} namespace=${{ env.NAMESPACE }} kind=Deployment name=nginx-deployment containerName=nginx:1.14.2 labels=app:nginx,workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Cluster_Integration_Tests_-_private_cluster selectorLabels=app:nginx
|
python test/integration/k8s-deploy-test.py private=${{ env.NAMESPACE }} namespace=${{ env.NAMESPACE }} kind=Deployment name=nginx-deployment containerName=nginx:1.14.2 labels=app:nginx,workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Cluster_Integration_Tests_-_private_cluster selectorLabels=app:nginx
|
||||||
python test/integration/k8s-deploy-test.py private=${{ env.NAMESPACE }} namespace=${{ env.NAMESPACE }} kind=Service name=nginx-service labels=workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Cluster_Integration_Tests_-_private_cluster selectorLabels=app:nginx
|
python test/integration/k8s-deploy-test.py private=${{ env.NAMESPACE }} namespace=${{ env.NAMESPACE }} kind=Service name=nginx-service labels=workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Cluster_Integration_Tests_-_private_cluster selectorLabels=app:nginx
|
||||||
|
|
||||||
|
python test/integration/k8s-deploy-test.py private=${{ env.NAMESPACE }} namespace=${{ env.NAMESPACE }} kind=Deployment name=nginx-deployment2 containerName=nginx:1.14.2 labels=app:nginx2,workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Cluster_Integration_Tests_-_private_cluster selectorLabels=app:nginx2
|
||||||
|
python test/integration/k8s-deploy-test.py private=${{ env.NAMESPACE }} namespace=${{ env.NAMESPACE }} kind=Service name=nginx-service2 labels=workflow:actions.github.com-k8s-deploy,workflowFriendlyName:Cluster_Integration_Tests_-_private_cluster selectorLabels=app:nginx2
|
||||||
|
|
||||||
- name: Clean up AKS cluster
|
- name: Clean up AKS cluster
|
||||||
if: ${{ always() }}
|
if: ${{ always() }}
|
||||||
run: |
|
run: |
|
||||||
|
|||||||
+18
-1
@@ -1,6 +1,23 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
## [v4.10.0] - 2023-10-30
|
## [5.0.0] - 2024-03-12
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- #309 Updated to Node20 and upgraded release workflows to @v1 tag
|
||||||
|
- #306 update release workflow to use new prefix, remove deprecated release
|
||||||
|
- #303 fix: ensure imageNames are not empty strings
|
||||||
|
- #299 bump release workflow sha
|
||||||
|
- #298 bump minikube to fix runner deps
|
||||||
|
- #297 update release workflow
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- #304 add v prefix for version tagging
|
||||||
|
- #302 adding ncc to build
|
||||||
|
- #301 adding release workflow artifact fix
|
||||||
|
|
||||||
|
## [4.10.0] - 2023-10-30
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
|
|||||||
@@ -132,7 +132,7 @@ Following are the key capabilities of this action:
|
|||||||
### Basic deployment (without any deployment strategy)
|
### Basic deployment (without any deployment strategy)
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
namespace: 'myapp'
|
namespace: 'myapp'
|
||||||
manifests: |
|
manifests: |
|
||||||
@@ -146,7 +146,7 @@ Following are the key capabilities of this action:
|
|||||||
### Private cluster deployment
|
### Private cluster deployment
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
resource-group: yourResourceGroup
|
resource-group: yourResourceGroup
|
||||||
name: yourClusterName
|
name: yourClusterName
|
||||||
@@ -166,7 +166,7 @@ Following are the key capabilities of this action:
|
|||||||
### Canary deployment without service mesh
|
### Canary deployment without service mesh
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
namespace: 'myapp'
|
namespace: 'myapp'
|
||||||
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
||||||
@@ -185,7 +185,7 @@ Following are the key capabilities of this action:
|
|||||||
To promote/reject the canary created by the above snippet, the following YAML snippet could be used:
|
To promote/reject the canary created by the above snippet, the following YAML snippet could be used:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
namespace: 'myapp'
|
namespace: 'myapp'
|
||||||
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
||||||
@@ -203,7 +203,7 @@ To promote/reject the canary created by the above snippet, the following YAML sn
|
|||||||
### Canary deployment based on Service Mesh Interface
|
### Canary deployment based on Service Mesh Interface
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
namespace: 'myapp'
|
namespace: 'myapp'
|
||||||
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
||||||
@@ -224,7 +224,7 @@ To promote/reject the canary created by the above snippet, the following YAML sn
|
|||||||
To promote/reject the canary created by the above snippet, the following YAML snippet could be used:
|
To promote/reject the canary created by the above snippet, the following YAML snippet could be used:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
namespace: 'myapp'
|
namespace: 'myapp'
|
||||||
images: 'contoso.azurecr.io/myapp:${{ event.run_id }} '
|
images: 'contoso.azurecr.io/myapp:${{ event.run_id }} '
|
||||||
@@ -243,7 +243,7 @@ To promote/reject the canary created by the above snippet, the following YAML sn
|
|||||||
### Blue-Green deployment with different route methods
|
### Blue-Green deployment with different route methods
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
namespace: 'myapp'
|
namespace: 'myapp'
|
||||||
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
||||||
@@ -263,7 +263,7 @@ To promote/reject the canary created by the above snippet, the following YAML sn
|
|||||||
To promote/reject the green workload created by the above snippet, the following YAML snippet could be used:
|
To promote/reject the green workload created by the above snippet, the following YAML snippet could be used:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
namespace: 'myapp'
|
namespace: 'myapp'
|
||||||
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
images: 'contoso.azurecr.io/myapp:${{ event.run_id }}'
|
||||||
@@ -292,7 +292,7 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@master
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- uses: Azure/docker-login@v1
|
- uses: Azure/docker-login@v1
|
||||||
with:
|
with:
|
||||||
@@ -304,23 +304,23 @@ jobs:
|
|||||||
docker build . -t contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
docker build . -t contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
||||||
docker push contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
docker push contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
||||||
|
|
||||||
- uses: azure/setup-kubectl@v2.0
|
- uses: azure/setup-kubectl@v4
|
||||||
|
|
||||||
# Set the target AKS cluster.
|
# Set the target AKS cluster.
|
||||||
- uses: Azure/aks-set-context@v1
|
- uses: Azure/aks-set-context@v4
|
||||||
with:
|
with:
|
||||||
creds: '${{ secrets.AZURE_CREDENTIALS }}'
|
creds: '${{ secrets.AZURE_CREDENTIALS }}'
|
||||||
cluster-name: contoso
|
cluster-name: contoso
|
||||||
resource-group: contoso-rg
|
resource-group: contoso-rg
|
||||||
|
|
||||||
- uses: Azure/k8s-create-secret@v1.1
|
- uses: Azure/k8s-create-secret@v4
|
||||||
with:
|
with:
|
||||||
container-registry-url: contoso.azurecr.io
|
container-registry-url: contoso.azurecr.io
|
||||||
container-registry-username: ${{ secrets.REGISTRY_USERNAME }}
|
container-registry-username: ${{ secrets.REGISTRY_USERNAME }}
|
||||||
container-registry-password: ${{ secrets.REGISTRY_PASSWORD }}
|
container-registry-password: ${{ secrets.REGISTRY_PASSWORD }}
|
||||||
secret-name: demo-k8s-secret
|
secret-name: demo-k8s-secret
|
||||||
|
|
||||||
- uses: Azure/k8s-deploy@v4
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
action: deploy
|
action: deploy
|
||||||
manifests: |
|
manifests: |
|
||||||
@@ -341,7 +341,7 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@master
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- uses: Azure/docker-login@v1
|
- uses: Azure/docker-login@v1
|
||||||
with:
|
with:
|
||||||
@@ -353,13 +353,13 @@ jobs:
|
|||||||
docker build . -t contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
docker build . -t contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
||||||
docker push contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
docker push contoso.azurecr.io/k8sdemo:${{ github.sha }}
|
||||||
|
|
||||||
- uses: azure/setup-kubectl@v2.0
|
- uses: azure/setup-kubectl@v4
|
||||||
|
|
||||||
- uses: Azure/k8s-set-context@v2
|
- uses: Azure/k8s-set-context@v4
|
||||||
with:
|
with:
|
||||||
kubeconfig: ${{ secrets.KUBE_CONFIG }}
|
kubeconfig: ${{ secrets.KUBE_CONFIG }}
|
||||||
|
|
||||||
- uses: Azure/k8s-create-secret@v1.1
|
- uses: Azure/k8s-create-secret@v4
|
||||||
with:
|
with:
|
||||||
container-registry-url: contoso.azurecr.io
|
container-registry-url: contoso.azurecr.io
|
||||||
container-registry-username: ${{ secrets.REGISTRY_USERNAME }}
|
container-registry-username: ${{ secrets.REGISTRY_USERNAME }}
|
||||||
@@ -391,7 +391,7 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@master
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- uses: Azure/docker-login@v1
|
- uses: Azure/docker-login@v1
|
||||||
with:
|
with:
|
||||||
@@ -423,16 +423,16 @@ jobs:
|
|||||||
username: ${{ secrets.REGISTRY_USERNAME }}
|
username: ${{ secrets.REGISTRY_USERNAME }}
|
||||||
password: ${{ secrets.REGISTRY_PASSWORD }}
|
password: ${{ secrets.REGISTRY_PASSWORD }}
|
||||||
|
|
||||||
- uses: azure/setup-kubectl@v2.0
|
- uses: azure/setup-kubectl@v4
|
||||||
|
|
||||||
# Set the target AKS cluster.
|
# Set the target AKS cluster.
|
||||||
- uses: Azure/aks-set-context@v1
|
- uses: Azure/aks-set-context@v4
|
||||||
with:
|
with:
|
||||||
creds: '${{ secrets.AZURE_CREDENTIALS }}'
|
creds: '${{ secrets.AZURE_CREDENTIALS }}'
|
||||||
cluster-name: contoso
|
cluster-name: contoso
|
||||||
resource-group: contoso-rg
|
resource-group: contoso-rg
|
||||||
|
|
||||||
- uses: Azure/k8s-create-secret@v1.1
|
- uses: Azure/k8s-create-secret@v4
|
||||||
with:
|
with:
|
||||||
namespace: ${{ env.NAMESPACE }}
|
namespace: ${{ env.NAMESPACE }}
|
||||||
container-registry-url: contoso.azurecr.io
|
container-registry-url: contoso.azurecr.io
|
||||||
@@ -440,7 +440,7 @@ jobs:
|
|||||||
container-registry-password: ${{ secrets.REGISTRY_PASSWORD }}
|
container-registry-password: ${{ secrets.REGISTRY_PASSWORD }}
|
||||||
secret-name: demo-k8s-secret
|
secret-name: demo-k8s-secret
|
||||||
|
|
||||||
- uses: azure/k8s-bake@v2
|
- uses: azure/k8s-bake@v3
|
||||||
with:
|
with:
|
||||||
renderEngine: 'helm'
|
renderEngine: 'helm'
|
||||||
helmChart: './aks-helloworld/'
|
helmChart: './aks-helloworld/'
|
||||||
@@ -450,7 +450,7 @@ jobs:
|
|||||||
helm-version: 'latest'
|
helm-version: 'latest'
|
||||||
id: bake
|
id: bake
|
||||||
|
|
||||||
- uses: Azure/k8s-deploy@v1.2
|
- uses: Azure/k8s-deploy@v5
|
||||||
with:
|
with:
|
||||||
action: deploy
|
action: deploy
|
||||||
manifests: ${{ steps.bake.outputs.manifestsBundle }}
|
manifests: ${{ steps.bake.outputs.manifestsBundle }}
|
||||||
|
|||||||
Generated
+2941
-6756
File diff suppressed because it is too large
Load Diff
+9
-7
@@ -1,10 +1,11 @@
|
|||||||
{
|
{
|
||||||
"name": "k8s-deploy-action",
|
"name": "k8s-deploy-action",
|
||||||
"version": "0.0.0",
|
"version": "5.0.0",
|
||||||
"author": "Deepak Sattiraju",
|
"author": "Deepak Sattiraju",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "npm i ncc && npx ncc build src/run.ts -o lib",
|
"prebuild": "npm i @vercel/ncc",
|
||||||
|
"build": "ncc build src/run.ts -o lib",
|
||||||
"test": "jest",
|
"test": "jest",
|
||||||
"coverage": "jest --coverage=true",
|
"coverage": "jest --coverage=true",
|
||||||
"format": "prettier --write .",
|
"format": "prettier --write .",
|
||||||
@@ -18,16 +19,17 @@
|
|||||||
"@octokit/core": "^3.5.1",
|
"@octokit/core": "^3.5.1",
|
||||||
"@octokit/plugin-retry": "^3.0.9",
|
"@octokit/plugin-retry": "^3.0.9",
|
||||||
"@types/minipass": "^3.1.2",
|
"@types/minipass": "^3.1.2",
|
||||||
"js-yaml": "3.13.1"
|
"js-yaml": "3.13.1",
|
||||||
|
"minimist": "^1.2.8"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/jest": "^26.0.0",
|
"@types/jest": "^26.0.0",
|
||||||
"@types/js-yaml": "^3.12.7",
|
"@types/js-yaml": "^3.12.7",
|
||||||
"@types/node": "^12.20.41",
|
"@types/node": "^12.20.41",
|
||||||
"@vercel/ncc": "^0.36.1",
|
"@vercel/ncc": "^0.36.1",
|
||||||
"jest": "^26.0.0",
|
"jest": "^29.7.0",
|
||||||
"prettier": "^2.7.1",
|
"prettier": "^2.8.8",
|
||||||
"ts-jest": "^26.0.0",
|
"ts-jest": "^29.2.3",
|
||||||
"typescript": "3.9.5"
|
"typescript": "5.5.4"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,12 +10,7 @@ import {Kubectl, Resource} from '../types/kubectl'
|
|||||||
import {deployPodCanary} from './canary/podCanaryHelper'
|
import {deployPodCanary} from './canary/podCanaryHelper'
|
||||||
import {deploySMICanary} from './canary/smiCanaryHelper'
|
import {deploySMICanary} from './canary/smiCanaryHelper'
|
||||||
import {DeploymentConfig} from '../types/deploymentConfig'
|
import {DeploymentConfig} from '../types/deploymentConfig'
|
||||||
import {
|
import {deployBlueGreen} from './blueGreen/deploy'
|
||||||
deployBlueGreen,
|
|
||||||
deployBlueGreenIngress,
|
|
||||||
deployBlueGreenService
|
|
||||||
} from './blueGreen/deploy'
|
|
||||||
import {deployBlueGreenSMI} from './blueGreen/deploy'
|
|
||||||
import {DeploymentStrategy} from '../types/deploymentStrategy'
|
import {DeploymentStrategy} from '../types/deploymentStrategy'
|
||||||
import * as core from '@actions/core'
|
import * as core from '@actions/core'
|
||||||
import {
|
import {
|
||||||
@@ -39,7 +34,6 @@ import {
|
|||||||
normalizeWorkflowStrLabel
|
normalizeWorkflowStrLabel
|
||||||
} from '../utilities/githubUtils'
|
} from '../utilities/githubUtils'
|
||||||
import {getDeploymentConfig} from '../utilities/dockerUtils'
|
import {getDeploymentConfig} from '../utilities/dockerUtils'
|
||||||
import {deploy} from '../actions/deploy'
|
|
||||||
import {DeployResult} from '../types/deployResult'
|
import {DeployResult} from '../types/deployResult'
|
||||||
|
|
||||||
export async function deployManifests(
|
export async function deployManifests(
|
||||||
|
|||||||
@@ -39,17 +39,6 @@ const testNamespace = 'testNamespace'
|
|||||||
const defaultNamespace = 'default'
|
const defaultNamespace = 'default'
|
||||||
const otherNamespace = 'otherns'
|
const otherNamespace = 'otherns'
|
||||||
describe('Kubectl class', () => {
|
describe('Kubectl class', () => {
|
||||||
describe('default namespace behavior', () => {
|
|
||||||
const kubectl = new Kubectl(kubectlPath, defaultNamespace)
|
|
||||||
const execReturn = {exitCode: 0, stdout: 'Output', stderr: ''}
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.spyOn(exec, 'getExecOutput').mockImplementation(async () => {
|
|
||||||
return execReturn
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('with a success exec return in testNamespace', () => {
|
describe('with a success exec return in testNamespace', () => {
|
||||||
const kubectl = new Kubectl(kubectlPath, testNamespace)
|
const kubectl = new Kubectl(kubectlPath, testNamespace)
|
||||||
const execReturn = {exitCode: 0, stdout: 'Output', stderr: ''}
|
const execReturn = {exitCode: 0, stdout: 'Output', stderr: ''}
|
||||||
|
|||||||
@@ -1,8 +1,14 @@
|
|||||||
import {PrivateKubectl} from './privatekubectl'
|
import * as fileUtils from '../utilities/fileUtils'
|
||||||
|
import * as fs from 'fs'
|
||||||
|
import {
|
||||||
|
PrivateKubectl,
|
||||||
|
extractFileNames,
|
||||||
|
replaceFileNamesWithShallowNamesRelativeToTemp
|
||||||
|
} from './privatekubectl'
|
||||||
import * as exec from '@actions/exec'
|
import * as exec from '@actions/exec'
|
||||||
|
|
||||||
describe('Private kubectl', () => {
|
describe('Private kubectl', () => {
|
||||||
const testString = `kubectl annotate -f test.yml,test2.yml,test3.yml -f test4.yml --filename test5.yml actions.github.com/k8s-deploy={"run":"3498366832","repository":"jaiveerk/k8s-deploy","workflow":"Minikube Integration Tests - private cluster","workflowFileName":"run-integration-tests-private.yml","jobName":"run-integration-test","createdBy":"jaiveerk","runUri":"https://github.com/jaiveerk/k8s-deploy/actions/runs/3498366832","commit":"c63b323186ea1320a31290de6dcc094c06385e75","lastSuccessRunCommit":"NA","branch":"refs/heads/main","deployTimestamp":1668787848577,"dockerfilePaths":{"nginx:1.14.2":""},"manifestsPaths":["https://github.com/jaiveerk/k8s-deploy/blob/c63b323186ea1320a31290de6dcc094c06385e75/test/integration/manifests/test.yml"],"helmChartPaths":[],"provider":"GitHub"} --overwrite --namespace test-3498366832`
|
const testString = `kubectl annotate -f /tmp/testdir/test.yml,/tmp/test2.yml,/tmp/testdir/subdir/test3.yml -f /tmp/test4.yml --filename /tmp/test5.yml actions.github.com/k8s-deploy={"run":"3498366832","repository":"jaiveerk/k8s-deploy","workflow":"Minikube Integration Tests - private cluster","workflowFileName":"run-integration-tests-private.yml","jobName":"run-integration-test","createdBy":"jaiveerk","runUri":"https://github.com/jaiveerk/k8s-deploy/actions/runs/3498366832","commit":"c63b323186ea1320a31290de6dcc094c06385e75","lastSuccessRunCommit":"NA","branch":"refs/heads/main","deployTimestamp":1668787848577,"dockerfilePaths":{"nginx:1.14.2":""},"manifestsPaths":["https://github.com/jaiveerk/k8s-deploy/blob/c63b323186ea1320a31290de6dcc094c06385e75/test/integration/test.yml"],"helmChartPaths":[],"provider":"GitHub"} --overwrite --namespace test-3498366832`
|
||||||
const mockKube = new PrivateKubectl(
|
const mockKube = new PrivateKubectl(
|
||||||
'kubectlPath',
|
'kubectlPath',
|
||||||
'namespace',
|
'namespace',
|
||||||
@@ -11,9 +17,32 @@ describe('Private kubectl', () => {
|
|||||||
'resourceName'
|
'resourceName'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const spy = jest
|
||||||
|
.spyOn(fileUtils, 'getTempDirectory')
|
||||||
|
.mockImplementation(() => {
|
||||||
|
return '/tmp'
|
||||||
|
})
|
||||||
|
|
||||||
|
jest.spyOn(fs, 'writeFileSync').mockImplementation(() => {})
|
||||||
|
jest.spyOn(fs, 'readFileSync').mockImplementation((filename) => {
|
||||||
|
return 'test contents'
|
||||||
|
})
|
||||||
|
|
||||||
it('should extract filenames correctly', () => {
|
it('should extract filenames correctly', () => {
|
||||||
expect(mockKube.extractFilesnames(testString)).toEqual(
|
expect(extractFileNames(testString)).toEqual([
|
||||||
'test.yml test2.yml test3.yml test4.yml test5.yml'
|
'/tmp/testdir/test.yml',
|
||||||
|
'/tmp/test2.yml',
|
||||||
|
'/tmp/testdir/subdir/test3.yml',
|
||||||
|
'/tmp/test4.yml',
|
||||||
|
'/tmp/test5.yml'
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should replace filenames with shallow names for relative locations in tmp correctly', () => {
|
||||||
|
expect(
|
||||||
|
replaceFileNamesWithShallowNamesRelativeToTemp(testString)
|
||||||
|
).toEqual(
|
||||||
|
`kubectl annotate -f testdir-test.yml,test2.yml,testdir-subdir-test3.yml -f test4.yml --filename test5.yml actions.github.com/k8s-deploy={"run":"3498366832","repository":"jaiveerk/k8s-deploy","workflow":"Minikube Integration Tests - private cluster","workflowFileName":"run-integration-tests-private.yml","jobName":"run-integration-test","createdBy":"jaiveerk","runUri":"https://github.com/jaiveerk/k8s-deploy/actions/runs/3498366832","commit":"c63b323186ea1320a31290de6dcc094c06385e75","lastSuccessRunCommit":"NA","branch":"refs/heads/main","deployTimestamp":1668787848577,"dockerfilePaths":{"nginx:1.14.2":""},"manifestsPaths":["https://github.com/jaiveerk/k8s-deploy/blob/c63b323186ea1320a31290de6dcc094c06385e75/test/integration/test.yml"],"helmChartPaths":[],"provider":"GitHub"} --overwrite --namespace test-3498366832`
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
+86
-99
@@ -5,6 +5,7 @@ import * as core from '@actions/core'
|
|||||||
import * as os from 'os'
|
import * as os from 'os'
|
||||||
import * as fs from 'fs'
|
import * as fs from 'fs'
|
||||||
import * as path from 'path'
|
import * as path from 'path'
|
||||||
|
import {getTempDirectory} from '../utilities/fileUtils'
|
||||||
|
|
||||||
export class PrivateKubectl extends Kubectl {
|
export class PrivateKubectl extends Kubectl {
|
||||||
protected async execute(args: string[], silent: boolean = false) {
|
protected async execute(args: string[], silent: boolean = false) {
|
||||||
@@ -18,8 +19,7 @@ export class PrivateKubectl extends Kubectl {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (this.containsFilenames(kubectlCmd)) {
|
if (this.containsFilenames(kubectlCmd)) {
|
||||||
// For private clusters, files will referenced solely by their basename
|
kubectlCmd = replaceFileNamesWithShallowNamesRelativeToTemp(kubectlCmd)
|
||||||
kubectlCmd = this.replaceFilnamesWithBasenames(kubectlCmd)
|
|
||||||
addFileFlag = true
|
addFileFlag = true
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -43,22 +43,9 @@ export class PrivateKubectl extends Kubectl {
|
|||||||
]
|
]
|
||||||
|
|
||||||
if (addFileFlag) {
|
if (addFileFlag) {
|
||||||
const filenames = this.extractFilesnames(kubectlCmd).split(' ')
|
const tempDirectory = getTempDirectory()
|
||||||
|
eo.cwd = path.join(tempDirectory, 'manifests')
|
||||||
const tempDirectory =
|
|
||||||
process.env['runner.tempDirectory'] || os.tmpdir() + '/manifests'
|
|
||||||
eo.cwd = tempDirectory
|
|
||||||
privateClusterArgs.push(...['--file', '.'])
|
privateClusterArgs.push(...['--file', '.'])
|
||||||
|
|
||||||
let filenamesArr = filenames[0].split(',')
|
|
||||||
for (let index = 0; index < filenamesArr.length; index++) {
|
|
||||||
const file = filenamesArr[index]
|
|
||||||
|
|
||||||
if (!file) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
this.moveFileToTempManifestDir(file)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
core.debug(
|
core.debug(
|
||||||
@@ -95,89 +82,89 @@ export class PrivateKubectl extends Kubectl {
|
|||||||
} as ExecOutput
|
} as ExecOutput
|
||||||
}
|
}
|
||||||
|
|
||||||
private replaceFilnamesWithBasenames(kubectlCmd: string) {
|
|
||||||
let exFilenames = this.extractFilesnames(kubectlCmd)
|
|
||||||
let filenames = exFilenames.split(' ')
|
|
||||||
let filenamesArr = filenames[0].split(',')
|
|
||||||
|
|
||||||
for (let index = 0; index < filenamesArr.length; index++) {
|
|
||||||
filenamesArr[index] = path.basename(filenamesArr[index])
|
|
||||||
}
|
|
||||||
|
|
||||||
let baseFilenames = filenamesArr.join()
|
|
||||||
|
|
||||||
let result = kubectlCmd.replace(exFilenames, baseFilenames)
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
public extractFilesnames(strToParse: string) {
|
|
||||||
const fileNames: string[] = []
|
|
||||||
const argv = minimist(strToParse.split(' '))
|
|
||||||
const fArg = 'f'
|
|
||||||
const filenameArg = 'filename'
|
|
||||||
|
|
||||||
fileNames.push(...this.extractFilesFromMinimist(argv, fArg))
|
|
||||||
fileNames.push(...this.extractFilesFromMinimist(argv, filenameArg))
|
|
||||||
|
|
||||||
return fileNames.join(' ')
|
|
||||||
}
|
|
||||||
|
|
||||||
private extractFilesFromMinimist(argv, arg: string): string[] {
|
|
||||||
if (!argv[arg]) {
|
|
||||||
return []
|
|
||||||
}
|
|
||||||
const toReturn: string[] = []
|
|
||||||
if (typeof argv[arg] === 'string') {
|
|
||||||
toReturn.push(...argv[arg].split(','))
|
|
||||||
} else {
|
|
||||||
for (const value of argv[arg] as string[]) {
|
|
||||||
toReturn.push(...value.split(','))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return toReturn
|
|
||||||
}
|
|
||||||
|
|
||||||
private containsFilenames(str: string) {
|
private containsFilenames(str: string) {
|
||||||
return str.includes('-f ') || str.includes('filename ')
|
return str.includes('-f ') || str.includes('filename ')
|
||||||
}
|
}
|
||||||
|
}
|
||||||
private createTempManifestsDirectory() {
|
|
||||||
const manifestsDir = '/tmp/manifests'
|
function createTempManifestsDirectory(): string {
|
||||||
if (!fs.existsSync('/tmp/manifests')) {
|
const manifestsDirPath = path.join(getTempDirectory(), 'manifests')
|
||||||
fs.mkdirSync('/tmp/manifests', {recursive: true})
|
if (!fs.existsSync(manifestsDirPath)) {
|
||||||
}
|
fs.mkdirSync(manifestsDirPath, {recursive: true})
|
||||||
}
|
}
|
||||||
|
|
||||||
private moveFileToTempManifestDir(file: string) {
|
return manifestsDirPath
|
||||||
this.createTempManifestsDirectory()
|
}
|
||||||
if (!fs.existsSync('/tmp/' + file)) {
|
|
||||||
core.debug(
|
export function replaceFileNamesWithShallowNamesRelativeToTemp(
|
||||||
'/tmp/' +
|
kubectlCmd: string
|
||||||
file +
|
) {
|
||||||
' does not exist, and therefore cannot be moved to the manifest directory'
|
let filenames = extractFileNames(kubectlCmd)
|
||||||
)
|
core.debug(`filenames originally provided in kubectl command: ${filenames}`)
|
||||||
}
|
let relativeShallowNames = filenames.map((filename) => {
|
||||||
|
const relativeName = path.relative(getTempDirectory(), filename)
|
||||||
fs.copyFile('/tmp/' + file, '/tmp/manifests/' + file, function (err) {
|
|
||||||
if (err) {
|
const relativePathElements = relativeName.split(path.sep)
|
||||||
core.debug(
|
|
||||||
'Could not rename ' +
|
const shallowName = relativePathElements.join('-')
|
||||||
'/tmp/' +
|
|
||||||
file +
|
// make manifests dir in temp if it doesn't already exist
|
||||||
' to ' +
|
const manifestsTempDir = createTempManifestsDirectory()
|
||||||
'/tmp/manifests/' +
|
|
||||||
file +
|
const shallowPath = path.join(manifestsTempDir, shallowName)
|
||||||
' ERROR: ' +
|
core.debug(
|
||||||
err
|
`moving contents from ${filename} to shallow location at ${shallowPath}`
|
||||||
)
|
)
|
||||||
return
|
|
||||||
}
|
core.debug(`reading contents from ${filename}`)
|
||||||
core.debug(
|
const contents = fs.readFileSync(filename).toString()
|
||||||
"Successfully moved file '" +
|
|
||||||
file +
|
core.debug(`writing contents to new path ${shallowPath}`)
|
||||||
"' from /tmp to /tmp/manifest directory"
|
fs.writeFileSync(shallowPath, contents)
|
||||||
)
|
|
||||||
})
|
return shallowName
|
||||||
}
|
})
|
||||||
|
|
||||||
|
let result = kubectlCmd
|
||||||
|
if (filenames.length != relativeShallowNames.length) {
|
||||||
|
throw Error(
|
||||||
|
'replacing filenames with relative path from temp dir, ' +
|
||||||
|
filenames.length +
|
||||||
|
' filenames != ' +
|
||||||
|
relativeShallowNames.length +
|
||||||
|
'basenames'
|
||||||
|
)
|
||||||
|
}
|
||||||
|
for (let index = 0; index < filenames.length; index++) {
|
||||||
|
result = result.replace(filenames[index], relativeShallowNames[index])
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
export function extractFileNames(strToParse: string) {
|
||||||
|
const fileNames: string[] = []
|
||||||
|
const argv = minimist(strToParse.split(' '))
|
||||||
|
const fArg = 'f'
|
||||||
|
const filenameArg = 'filename'
|
||||||
|
|
||||||
|
fileNames.push(...extractFilesFromMinimist(argv, fArg))
|
||||||
|
fileNames.push(...extractFilesFromMinimist(argv, filenameArg))
|
||||||
|
|
||||||
|
return fileNames
|
||||||
|
}
|
||||||
|
|
||||||
|
export function extractFilesFromMinimist(argv, arg: string): string[] {
|
||||||
|
if (!argv[arg]) {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
const toReturn: string[] = []
|
||||||
|
if (typeof argv[arg] === 'string') {
|
||||||
|
toReturn.push(...argv[arg].split(','))
|
||||||
|
} else {
|
||||||
|
for (const value of argv[arg] as string[]) {
|
||||||
|
toReturn.push(...value.split(','))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return toReturn
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,14 @@
|
|||||||
import {
|
import * as fileUtils from './fileUtils'
|
||||||
getFilesFromDirectoriesAndURLs,
|
|
||||||
getTempDirectory,
|
|
||||||
urlFileKind,
|
|
||||||
writeYamlFromURLToFile
|
|
||||||
} from './fileUtils'
|
|
||||||
|
|
||||||
import * as yaml from 'js-yaml'
|
import * as yaml from 'js-yaml'
|
||||||
import * as fs from 'fs'
|
import * as fs from 'fs'
|
||||||
import * as path from 'path'
|
import * as path from 'path'
|
||||||
import {succeeded} from '../types/errorable'
|
|
||||||
|
|
||||||
const sampleYamlUrl =
|
const sampleYamlUrl =
|
||||||
'https://raw.githubusercontent.com/kubernetes/website/main/content/en/examples/controllers/nginx-deployment.yaml'
|
'https://raw.githubusercontent.com/kubernetes/website/main/content/en/examples/controllers/nginx-deployment.yaml'
|
||||||
describe('File utils', () => {
|
describe('File utils', () => {
|
||||||
test('correctly parses a yaml file from a URL', async () => {
|
test('correctly parses a yaml file from a URL', async () => {
|
||||||
const tempFile = await writeYamlFromURLToFile(sampleYamlUrl, 0)
|
const tempFile = await fileUtils.writeYamlFromURLToFile(sampleYamlUrl, 0)
|
||||||
const fileContents = fs.readFileSync(tempFile).toString()
|
const fileContents = fs.readFileSync(tempFile).toString()
|
||||||
const inputObjects = yaml.safeLoadAll(fileContents)
|
const inputObjects = yaml.safeLoadAll(fileContents)
|
||||||
expect(inputObjects).toHaveLength(1)
|
expect(inputObjects).toHaveLength(1)
|
||||||
@@ -30,34 +24,34 @@ describe('File utils', () => {
|
|||||||
|
|
||||||
const testPath = path.join('test', 'unit', 'manifests')
|
const testPath = path.join('test', 'unit', 'manifests')
|
||||||
await expect(
|
await expect(
|
||||||
getFilesFromDirectoriesAndURLs([testPath, badUrl])
|
fileUtils.getFilesFromDirectoriesAndURLs([testPath, badUrl])
|
||||||
).rejects.toThrow()
|
).rejects.toThrow()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('detects files in nested directories and ignores non-manifest files and empty dirs', async () => {
|
it('detects files in nested directories with the same name and ignores non-manifest files and empty dirs', async () => {
|
||||||
const testPath = path.join('test', 'unit', 'manifests')
|
const testPath = path.join('test', 'unit', 'manifests')
|
||||||
const testSearch: string[] = await getFilesFromDirectoriesAndURLs([
|
const testSearch: string[] =
|
||||||
testPath,
|
await fileUtils.getFilesFromDirectoriesAndURLs([
|
||||||
sampleYamlUrl
|
testPath,
|
||||||
])
|
sampleYamlUrl
|
||||||
|
])
|
||||||
|
|
||||||
const expectedManifests = [
|
const expectedManifests = [
|
||||||
'test/unit/manifests/manifest_test_dir/another_layer/deep-ingress.yaml',
|
'test/unit/manifests/manifest_test_dir/another_layer/test-ingress.yaml',
|
||||||
'test/unit/manifests/manifest_test_dir/another_layer/deep-service.yaml',
|
'test/unit/manifests/manifest_test_dir/another_layer/nested-test-service.yaml',
|
||||||
'test/unit/manifests/manifest_test_dir/nested-test-service.yaml',
|
'test/unit/manifests/manifest_test_dir/nested-test-service.yaml',
|
||||||
'test/unit/manifests/test-ingress.yml',
|
'test/unit/manifests/test-ingress.yml',
|
||||||
'test/unit/manifests/test-ingress-new.yml',
|
'test/unit/manifests/test-ingress-new.yml',
|
||||||
'test/unit/manifests/test-service.yml'
|
'test/unit/manifests/test-service.yml'
|
||||||
]
|
]
|
||||||
|
|
||||||
// is there a more efficient way to test equality w random order?
|
|
||||||
expect(testSearch).toHaveLength(8)
|
expect(testSearch).toHaveLength(8)
|
||||||
expectedManifests.forEach((fileName) => {
|
expectedManifests.forEach((fileName) => {
|
||||||
if (fileName.startsWith('test/unit')) {
|
if (fileName.startsWith('test/unit')) {
|
||||||
expect(testSearch).toContain(fileName)
|
expect(testSearch).toContain(fileName)
|
||||||
} else {
|
} else {
|
||||||
expect(fileName.includes(urlFileKind)).toBe(true)
|
expect(fileName.includes(fileUtils.urlFileKind)).toBe(true)
|
||||||
expect(fileName.startsWith(getTempDirectory()))
|
expect(fileName.startsWith(fileUtils.getTempDirectory()))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@@ -72,7 +66,7 @@ describe('File utils', () => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
expect(
|
expect(
|
||||||
getFilesFromDirectoriesAndURLs([badPath, goodPath])
|
fileUtils.getFilesFromDirectoriesAndURLs([badPath, goodPath])
|
||||||
).rejects.toThrowError()
|
).rejects.toThrowError()
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -92,7 +86,7 @@ describe('File utils', () => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
expect(
|
expect(
|
||||||
await getFilesFromDirectoriesAndURLs([
|
await fileUtils.getFilesFromDirectoriesAndURLs([
|
||||||
outerPath,
|
outerPath,
|
||||||
fileAtOuter,
|
fileAtOuter,
|
||||||
innerPath
|
innerPath
|
||||||
@@ -102,6 +96,24 @@ describe('File utils', () => {
|
|||||||
|
|
||||||
it('throws an error for an invalid URL', async () => {
|
it('throws an error for an invalid URL', async () => {
|
||||||
const badUrl = 'https://www.github.com'
|
const badUrl = 'https://www.github.com'
|
||||||
await expect(writeYamlFromURLToFile(badUrl, 0)).rejects.toBeTruthy()
|
await expect(
|
||||||
|
fileUtils.writeYamlFromURLToFile(badUrl, 0)
|
||||||
|
).rejects.toBeTruthy()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('moving files to temp', () => {
|
||||||
|
it('correctly moves the contents of a file to the temporary directory', () => {
|
||||||
|
jest.spyOn(fs, 'writeFileSync').mockImplementation(() => {})
|
||||||
|
jest.spyOn(fs, 'readFileSync').mockImplementation((filename) => {
|
||||||
|
return 'test contents'
|
||||||
|
})
|
||||||
|
const originalFilePath = path.join('path', 'in', 'repo')
|
||||||
|
|
||||||
|
const output = fileUtils.moveFileToTmpDir(originalFilePath)
|
||||||
|
|
||||||
|
expect(output).toEqual(
|
||||||
|
path.join(fileUtils.getTempDirectory(), '/path/in/repo')
|
||||||
|
)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ export function writeObjectsToFile(inputObjects: any[]): string[] {
|
|||||||
const inputObjectString = JSON.stringify(inputObject)
|
const inputObjectString = JSON.stringify(inputObject)
|
||||||
|
|
||||||
if (inputObject?.metadata?.name) {
|
if (inputObject?.metadata?.name) {
|
||||||
const fileName = getManifestFileName(
|
const fileName = getNewTempManifestFileName(
|
||||||
inputObject.kind,
|
inputObject.kind,
|
||||||
inputObject.metadata.name
|
inputObject.metadata.name
|
||||||
)
|
)
|
||||||
@@ -52,7 +52,7 @@ export function writeManifestToFile(
|
|||||||
): string {
|
): string {
|
||||||
if (inputObjectString) {
|
if (inputObjectString) {
|
||||||
try {
|
try {
|
||||||
const fileName = getManifestFileName(kind, name)
|
const fileName = getNewTempManifestFileName(kind, name)
|
||||||
fs.writeFileSync(path.join(fileName), inputObjectString)
|
fs.writeFileSync(path.join(fileName), inputObjectString)
|
||||||
return fileName
|
return fileName
|
||||||
} catch (ex) {
|
} catch (ex) {
|
||||||
@@ -63,7 +63,27 @@ export function writeManifestToFile(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function getManifestFileName(kind: string, name: string) {
|
export function moveFileToTmpDir(originalFilepath: string) {
|
||||||
|
const tempDirectory = getTempDirectory()
|
||||||
|
const newPath = path.join(tempDirectory, originalFilepath)
|
||||||
|
|
||||||
|
core.debug(`reading original contents from path: ${originalFilepath}`)
|
||||||
|
const contents = fs.readFileSync(originalFilepath).toString()
|
||||||
|
|
||||||
|
const dirName = path.dirname(newPath)
|
||||||
|
if (!fs.existsSync(dirName)) {
|
||||||
|
core.debug(`path ${dirName} doesn't exist yet, making new dir...`)
|
||||||
|
fs.mkdirSync(dirName, {recursive: true})
|
||||||
|
}
|
||||||
|
core.debug(`writing contents to new path ${newPath}`)
|
||||||
|
fs.writeFileSync(path.join(newPath), contents)
|
||||||
|
|
||||||
|
core.debug(`moved contents from ${originalFilepath} to ${newPath}`)
|
||||||
|
|
||||||
|
return newPath
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNewTempManifestFileName(kind: string, name: string) {
|
||||||
const filePath = `${kind}_${name}_${getCurrentTime().toString()}`
|
const filePath = `${kind}_${name}_${getCurrentTime().toString()}`
|
||||||
const tempDirectory = getTempDirectory()
|
const tempDirectory = getTempDirectory()
|
||||||
return path.join(tempDirectory, path.basename(filePath))
|
return path.join(tempDirectory, path.basename(filePath))
|
||||||
@@ -130,7 +150,7 @@ export async function writeYamlFromURLToFile(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
const targetPath = getManifestFileName(
|
const targetPath = getNewTempManifestFileName(
|
||||||
urlFileKind,
|
urlFileKind,
|
||||||
fileNumber.toString()
|
fileNumber.toString()
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import * as fileUtils from './fileUtils'
|
||||||
|
import * as manifestUpdateUtils from './manifestUpdateUtils'
|
||||||
|
import * as path from 'path'
|
||||||
|
import * as fs from 'fs'
|
||||||
|
|
||||||
|
describe('manifestUpdateUtils', () => {
|
||||||
|
jest.spyOn(fileUtils, 'moveFileToTmpDir').mockImplementation((filename) => {
|
||||||
|
return path.join('/tmp', filename)
|
||||||
|
})
|
||||||
|
jest.spyOn(fs, 'writeFileSync').mockImplementation(() => {})
|
||||||
|
jest.spyOn(fs, 'readFileSync').mockImplementation((filename) => {
|
||||||
|
return 'test contents'
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should place all files within the temp dir with the same path that they have in the repo', () => {
|
||||||
|
const originalFilePaths: string[] = [
|
||||||
|
'path/in/repo/test.txt',
|
||||||
|
'path/deeper/in/repo/test.txt'
|
||||||
|
]
|
||||||
|
const expected: string[] = [
|
||||||
|
'/tmp/path/in/repo/test.txt',
|
||||||
|
'/tmp/path/deeper/in/repo/test.txt'
|
||||||
|
]
|
||||||
|
const newFilePaths =
|
||||||
|
manifestUpdateUtils.moveFilesToTmpDir(originalFilePaths)
|
||||||
|
expect(newFilePaths).toEqual(expected)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -3,7 +3,7 @@ import * as fs from 'fs'
|
|||||||
import * as yaml from 'js-yaml'
|
import * as yaml from 'js-yaml'
|
||||||
import * as path from 'path'
|
import * as path from 'path'
|
||||||
import * as fileHelper from './fileUtils'
|
import * as fileHelper from './fileUtils'
|
||||||
import {getTempDirectory} from './fileUtils'
|
import {moveFileToTmpDir} from './fileUtils'
|
||||||
import {
|
import {
|
||||||
InputObjectKindNotDefinedError,
|
InputObjectKindNotDefinedError,
|
||||||
InputObjectMetadataNotDefinedError,
|
InputObjectMetadataNotDefinedError,
|
||||||
@@ -26,10 +26,14 @@ export function updateManifestFiles(manifestFilePaths: string[]) {
|
|||||||
throw new Error('Manifest files not provided')
|
throw new Error('Manifest files not provided')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// move original set of input files to tmp dir
|
||||||
|
const manifestFilesInTempDir = moveFilesToTmpDir(manifestFilePaths)
|
||||||
|
|
||||||
// update container images
|
// update container images
|
||||||
const containers: string[] = core.getInput('images').split('\n')
|
const containers: string[] = core.getInput('images').split('\n')
|
||||||
|
|
||||||
const manifestFiles = updateContainerImagesInManifestFiles(
|
const manifestFiles = updateContainerImagesInManifestFiles(
|
||||||
manifestFilePaths,
|
manifestFilesInTempDir,
|
||||||
containers
|
containers
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -41,6 +45,12 @@ export function updateManifestFiles(manifestFilePaths: string[]) {
|
|||||||
return updateImagePullSecretsInManifestFiles(manifestFiles, imagePullSecrets)
|
return updateImagePullSecretsInManifestFiles(manifestFiles, imagePullSecrets)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function moveFilesToTmpDir(filepaths: string[]): string[] {
|
||||||
|
return filepaths.map((filename) => {
|
||||||
|
return moveFileToTmpDir(filename)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
export function UnsetClusterSpecificDetails(resource: any) {
|
export function UnsetClusterSpecificDetails(resource: any) {
|
||||||
if (!resource) {
|
if (!resource) {
|
||||||
return
|
return
|
||||||
@@ -70,12 +80,9 @@ function updateContainerImagesInManifestFiles(
|
|||||||
): string[] {
|
): string[] {
|
||||||
if (filePaths?.length <= 0) return filePaths
|
if (filePaths?.length <= 0) return filePaths
|
||||||
|
|
||||||
const newFilePaths = []
|
|
||||||
|
|
||||||
// update container images
|
// update container images
|
||||||
filePaths.forEach((filePath: string) => {
|
filePaths.forEach((filePath: string) => {
|
||||||
let contents = fs.readFileSync(filePath).toString()
|
let contents = fs.readFileSync(filePath).toString()
|
||||||
|
|
||||||
containers.forEach((container: string) => {
|
containers.forEach((container: string) => {
|
||||||
let [imageName] = container.split(':')
|
let [imageName] = container.split(':')
|
||||||
if (imageName.indexOf('@') > 0) {
|
if (imageName.indexOf('@') > 0) {
|
||||||
@@ -91,13 +98,10 @@ function updateContainerImagesInManifestFiles(
|
|||||||
})
|
})
|
||||||
|
|
||||||
// write updated files
|
// write updated files
|
||||||
const tempDirectory = getTempDirectory()
|
fs.writeFileSync(path.join(filePath), contents)
|
||||||
const fileName = path.join(tempDirectory, path.basename(filePath))
|
|
||||||
fs.writeFileSync(path.join(fileName), contents)
|
|
||||||
newFilePaths.push(fileName)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
return newFilePaths
|
return filePaths
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|||||||
@@ -232,8 +232,7 @@ def main():
|
|||||||
getAllObjectsCmd = azPrefix + 'kubectl get '+kind+' -n '+namespace
|
getAllObjectsCmd = azPrefix + 'kubectl get '+kind+' -n '+namespace
|
||||||
if not azPrefix == "":
|
if not azPrefix == "":
|
||||||
getAllObjectsCmd = azPrefix + "'{getAllObjectsCmd}'" # add extra set of quotes
|
getAllObjectsCmd = azPrefix + "'{getAllObjectsCmd}'" # add extra set of quotes
|
||||||
cmd = + "'" + cmd + "'"
|
foundObjects = os.popen(getAllObjectsCmd).read()
|
||||||
foundObjects = os.popen().read()
|
|
||||||
suffix = f"resources of type {kind}: {foundObjects}"
|
suffix = f"resources of type {kind}: {foundObjects}"
|
||||||
sys.exit(msg + " " + suffix)
|
sys.exit(msg + " " + suffix)
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
apiVersion: apps/v1
|
||||||
|
kind: Deployment
|
||||||
|
metadata:
|
||||||
|
name: nginx-deployment3
|
||||||
|
labels:
|
||||||
|
app: nginx3
|
||||||
|
spec:
|
||||||
|
replicas: 1
|
||||||
|
selector:
|
||||||
|
matchLabels:
|
||||||
|
app: nginx3
|
||||||
|
template:
|
||||||
|
metadata:
|
||||||
|
labels:
|
||||||
|
app: nginx3
|
||||||
|
spec:
|
||||||
|
containers:
|
||||||
|
- name: nginx
|
||||||
|
image: nginx
|
||||||
|
ports:
|
||||||
|
- containerPort: 80
|
||||||
|
---
|
||||||
|
apiVersion: v1
|
||||||
|
kind: Service
|
||||||
|
metadata:
|
||||||
|
name: nginx-service3
|
||||||
|
spec:
|
||||||
|
selector:
|
||||||
|
app: nginx3
|
||||||
|
ports:
|
||||||
|
- protocol: TCP
|
||||||
|
port: 80
|
||||||
|
targetPort: 80
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
apiVersion: apps/v1
|
||||||
|
kind: Deployment
|
||||||
|
metadata:
|
||||||
|
name: nginx-deployment2
|
||||||
|
labels:
|
||||||
|
app: nginx2
|
||||||
|
spec:
|
||||||
|
replicas: 1
|
||||||
|
selector:
|
||||||
|
matchLabels:
|
||||||
|
app: nginx2
|
||||||
|
template:
|
||||||
|
metadata:
|
||||||
|
labels:
|
||||||
|
app: nginx2
|
||||||
|
spec:
|
||||||
|
containers:
|
||||||
|
- name: nginx
|
||||||
|
image: nginx
|
||||||
|
ports:
|
||||||
|
- containerPort: 80
|
||||||
|
---
|
||||||
|
apiVersion: v1
|
||||||
|
kind: Service
|
||||||
|
metadata:
|
||||||
|
name: nginx-service2
|
||||||
|
spec:
|
||||||
|
selector:
|
||||||
|
app: nginx2
|
||||||
|
ports:
|
||||||
|
- protocol: TCP
|
||||||
|
port: 80
|
||||||
|
targetPort: 80
|
||||||
Reference in New Issue
Block a user