-
Notifications
You must be signed in to change notification settings - Fork 0
197 lines (154 loc) · 6.34 KB
/
node-cd.yml
File metadata and controls
197 lines (154 loc) · 6.34 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
# Continuous Deployment for Node.js
# Triggered only on version tags with football terminology names
# Example tag: v1.0.0-assist
name: Node.js CD
on:
push:
tags:
- "v*.*.*-*"
env:
REGISTRY: ghcr.io
IMAGE_NAME: ${{ github.repository }}
jobs:
test:
name: Test before deployment
runs-on: ubuntu-latest
permissions:
contents: read
steps:
- name: Checkout repository
uses: actions/checkout@v6
- name: Set up Node.js
uses: actions/setup-node@v6.3.0
with:
node-version-file: ".nvmrc"
cache: "npm"
cache-dependency-path: package-lock.json
- name: Install dependencies
run: npm ci
- name: Compile application
run: npm run build
- name: Run tests with coverage
run: npm run coverage
release:
name: Build and publish Docker image
needs: test
runs-on: ubuntu-latest
permissions:
contents: write
packages: write
steps:
- name: Checkout repository
uses: actions/checkout@v6
with:
fetch-depth: 0
- name: Extract version and term name from tag
id: extract_tag
run: |
# Tag format: v1.0.0-assist
TAG_NAME=${GITHUB_REF#refs/tags/}
# Extract semantic version (e.g., v1.0.0-assist -> 1.0.0)
SEMVER=$(echo "$TAG_NAME" | sed -E 's/^v([0-9]+\.[0-9]+\.[0-9]+)-.*/\1/')
# Extract term name (e.g., v1.0.0-assist -> assist)
TERM=$(echo "$TAG_NAME" | sed -E 's/^v[0-9]+\.[0-9]+\.[0-9]+-//')
# Validate semver format (X.Y.Z)
if ! echo "$SEMVER" | grep -Eq '^[0-9]+\.[0-9]+\.[0-9]+$'; then
echo "❌ Error: Invalid semantic version '$SEMVER' extracted from tag '$TAG_NAME'"
echo "Expected format: v{MAJOR}.{MINOR}.{PATCH}-{TERM} (e.g., v1.0.0-assist)"
exit 1
fi
# Valid football terminology terms (A-Z from CHANGELOG.md)
VALID_TERMS="assist bicyclekick corner dribble equalizer foul goal header interception juggle keeper lob marking nutmeg offside penalty quickthrow redcard save tackle upset volley wing xpass yellowcard zonedefense"
# Validate term name against the list
if [ -z "$TERM" ]; then
echo "❌ Error: Term name is empty in tag '$TAG_NAME'"
echo "Expected format: v{MAJOR}.{MINOR}.{PATCH}-{TERM} (e.g., v1.0.0-assist)"
exit 1
fi
if ! echo "$VALID_TERMS" | grep -qw "$TERM"; then
echo "❌ Error: Invalid term name '$TERM' in tag '$TAG_NAME'"
echo "Valid terms (A-Z): $VALID_TERMS"
echo "See CHANGELOG.md for the complete list"
exit 1
fi
# Export validated outputs
echo "tag_name=$TAG_NAME" >> $GITHUB_OUTPUT
echo "semver=$SEMVER" >> $GITHUB_OUTPUT
echo "term=$TERM" >> $GITHUB_OUTPUT
echo "📦 Release version: $SEMVER"
echo "⚽ Term name: $TERM"
- name: Log in to GitHub Container Registry
uses: docker/login-action@v4.0.0
with:
registry: ${{ env.REGISTRY }}
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v4.0.0
- name: Build and push Docker image
uses: docker/build-push-action@v7.0.0
with:
context: .
push: true
platforms: linux/amd64
provenance: false
cache-from: type=gha
cache-to: type=gha,mode=max
tags: |
${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:latest
${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${{ steps.extract_tag.outputs.semver }}
${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${{ steps.extract_tag.outputs.term }}
- name: Generate changelog
id: changelog
run: |
# Get previous tag
PREVIOUS_TAG=$(git tag --sort=-version:refname | grep -E '^v[0-9]+\.[0-9]+\.[0-9]+-' | sed -n '2p')
if [ -z "$PREVIOUS_TAG" ]; then
echo "📝 First release - no previous tag found"
CHANGELOG="Initial release"
else
echo "📝 Generating changelog from $PREVIOUS_TAG to ${{ steps.extract_tag.outputs.tag_name }}"
CHANGELOG=$(git log --pretty=format:"- %s (%h)" ${PREVIOUS_TAG}..${{ steps.extract_tag.outputs.tag_name }})
# Guard against empty changelog (e.g., re-tagging same commit)
if [ -z "$CHANGELOG" ]; then
CHANGELOG="No new changes since $PREVIOUS_TAG"
fi
fi
# Write changelog to file
echo "$CHANGELOG" > changelog.txt
cat changelog.txt
# Set output for use in release body
{
echo "changelog<<EOF"
echo "$CHANGELOG"
echo "EOF"
} >> $GITHUB_OUTPUT
- name: Create GitHub Release
uses: softprops/action-gh-release@v2.6.1
with:
name: "v${{ steps.extract_tag.outputs.semver }} - ${{ steps.extract_tag.outputs.term }} ⚽"
tag_name: ${{ steps.extract_tag.outputs.tag_name }}
body: |
# Release ${{ steps.extract_tag.outputs.semver }} - ${{ steps.extract_tag.outputs.term }} ⚽
## Docker Images
Pull this release using any of the following tags:
```bash
# By semantic version (recommended)
docker pull ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${{ steps.extract_tag.outputs.semver }}
# By term name
docker pull ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${{ steps.extract_tag.outputs.term }}
# Latest
docker pull ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:latest
```
## Quick Start
```bash
docker run -p 9000:9000 ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${{ steps.extract_tag.outputs.semver }}
```
Visit http://localhost:9000/swagger/ for API documentation.
## Changelog
${{ steps.changelog.outputs.changelog }}
---
📦 **Package:** [${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}](https://github.com/${{ github.repository }}/pkgs/container/ts-node-samples-express-restful)
draft: false
prerelease: false
generate_release_notes: true