initial attempt at IaC and CI/CD
This commit is contained in:
parent
597c7e4e0b
commit
820b251c61
11 changed files with 355 additions and 1 deletions
70
.github/workflows/app.yml
vendored
Normal file
70
.github/workflows/app.yml
vendored
Normal file
|
@ -0,0 +1,70 @@
|
|||
name: Deploy App
|
||||
|
||||
on:
|
||||
push:
|
||||
paths-ignore:
|
||||
- infrastructure/**
|
||||
branches:
|
||||
- main
|
||||
pull_request:
|
||||
paths-ignore:
|
||||
- infrastructure/**
|
||||
types: [opened, synchronize, reopened, closed]
|
||||
branches:
|
||||
- main
|
||||
|
||||
env:
|
||||
IMAGE_NAME: calque-app
|
||||
|
||||
permissions:
|
||||
id-token: write
|
||||
contents: read
|
||||
|
||||
jobs:
|
||||
versionize:
|
||||
name: Versionize
|
||||
if: github.event_name == 'push' || (github.event_name == 'pull_request' && github.event.action != 'closed')
|
||||
runs-on: ubuntu-latest
|
||||
outputs:
|
||||
semver: ${{ steps.gitversion.outputs.SemVer }}
|
||||
steps:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0
|
||||
- name: Install GitVersion
|
||||
uses: gittools/actions/gitversion/setup@v1.1.1
|
||||
with:
|
||||
versionSpec: "5.x"
|
||||
- name: Determine Version
|
||||
id: gitversion
|
||||
uses: gittools/actions/gitversion/execute@v1.1.1
|
||||
with:
|
||||
useConfigFile: true
|
||||
|
||||
build_and_publish:
|
||||
name: Build & Publish
|
||||
if: github.event_name == 'push' || (github.event_name == 'pull_request' && github.event.action != 'closed')
|
||||
runs-on: ubuntu-latest
|
||||
needs: versionize
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Build container images
|
||||
working-directory: src
|
||||
run: |
|
||||
docker build . --file Dockerfile --tag ${{ secrets.ACR_LOGIN_SERVER }}/$IMAGE_NAME:${{needs.versionize.outputs.semver}}
|
||||
docker build . --file Dockerfile --tag ${{ secrets.ACR_LOGIN_SERVER }}/$IMAGE_NAME:latest
|
||||
|
||||
- name: Login to ACR
|
||||
uses: azure/docker-login@v2
|
||||
with:
|
||||
login-server: ${{ secrets.ACR_LOGIN_SERVER }}
|
||||
username: ${{ secrets.ACR_USERNAME }}
|
||||
password: ${{ secrets.ACR_PASSWORD }}
|
||||
|
||||
- name: Push container images
|
||||
working-directory: src
|
||||
run: |
|
||||
docker push ${{ secrets.ACR_LOGIN_SERVER }}/$IMAGE_NAME:${{needs.versionize.outputs.semver}}
|
||||
docker push ${{ secrets.ACR_LOGIN_SERVER }}/$IMAGE_NAME:latest
|
33
.github/workflows/infra.yml
vendored
Normal file
33
.github/workflows/infra.yml
vendored
Normal file
|
@ -0,0 +1,33 @@
|
|||
name: Deploy infrastructure
|
||||
|
||||
on:
|
||||
push:
|
||||
paths:
|
||||
- infrastructure/**
|
||||
branches:
|
||||
- main
|
||||
pull_request:
|
||||
paths:
|
||||
- infrastructure/**
|
||||
types: [opened, synchronize, reopened, closed]
|
||||
branches:
|
||||
- main
|
||||
|
||||
permissions:
|
||||
id-token: write
|
||||
contents: read
|
||||
|
||||
jobs:
|
||||
production:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
with:
|
||||
sparse-checkout: |
|
||||
infrastructure
|
||||
|
||||
- name: Deploy bicep
|
||||
uses: Azure/cli@v2
|
||||
with:
|
||||
inlineScript: |
|
||||
az deployment sub create --location westeurope --template-file infrastructure/main.bicep --parameters infrastructure/params/prod.bicepparam
|
33
.vscode/launch.json
vendored
33
.vscode/launch.json
vendored
|
@ -4,7 +4,7 @@
|
|||
{
|
||||
"type": "bun",
|
||||
"request": "launch",
|
||||
"name": "Debug Bun",
|
||||
"name": "Start dev",
|
||||
// The path to a JavaScript or TypeScript file to run.
|
||||
"program": "${file}",
|
||||
// The arguments to pass to the program, if any.
|
||||
|
@ -29,6 +29,37 @@
|
|||
// Unlike `args`, these are passed to the executable itself, not the program.
|
||||
"runtimeArgs": [],
|
||||
},
|
||||
{
|
||||
"type": "bun",
|
||||
"request": "launch",
|
||||
"name": "Run tests",
|
||||
// The path to a JavaScript or TypeScript file to run.
|
||||
"program": "${file}",
|
||||
// The arguments to pass to the program, if any.
|
||||
"args": [],
|
||||
// The working directory of the program.
|
||||
"cwd": "${workspaceFolder}",
|
||||
// The environment variables to pass to the program.
|
||||
"env": {},
|
||||
// If the environment variables should not be inherited from the parent process.
|
||||
"strictEnv": false,
|
||||
// If the program should be run in watch mode.
|
||||
// This is equivalent to passing `--watch` to the `bun` executable.
|
||||
// You can also set this to "hot" to enable hot reloading using `--hot`.
|
||||
"watchMode": false,
|
||||
// If the debugger should stop on the first line of the program.
|
||||
"stopOnEntry": false,
|
||||
// If the debugger should be disabled. (for example, breakpoints will not be hit)
|
||||
"noDebug": false,
|
||||
// The path to the `bun` executable, defaults to your `PATH` environment variable.
|
||||
"runtime": "bun",
|
||||
// The arguments to pass to the `bun` executable, if any.
|
||||
// Unlike `args`, these are passed to the executable itself, not the program.
|
||||
"runtimeArgs": [
|
||||
"--bun",
|
||||
"test"
|
||||
],
|
||||
},
|
||||
{
|
||||
"type": "bun",
|
||||
"request": "attach",
|
||||
|
|
32
Dockerfile
Normal file
32
Dockerfile
Normal file
|
@ -0,0 +1,32 @@
|
|||
FROM oven/bun:1 as base
|
||||
WORKDIR /usr/src/app
|
||||
|
||||
FROM base AS install
|
||||
RUN mkdir -p /temp/dev
|
||||
COPY package.json bun.lockb /temp/dev
|
||||
RUN cd /temp/dev && bun install --frozen-lockfile
|
||||
|
||||
RUN mkdir -p /temp/prod
|
||||
COPY package.json bun.lockb /temp/prod/
|
||||
RUN cd /temp/prod && bun install --frozen-lockfile --production
|
||||
|
||||
FROM base AS prerelease
|
||||
COPY --from=install /temp/dev/node_modules node_modules
|
||||
COPY . .
|
||||
|
||||
ENV NODE_ENV=production
|
||||
ENV SERVER_PRESET=bun
|
||||
RUN bun test
|
||||
RUN chmod +x node_modules/.bin/*
|
||||
RUN bun run build
|
||||
|
||||
FROM base AS release
|
||||
COPY --from=install /temp/prod/node_modules node_modules
|
||||
COPY --from=prerelease /usr/src/app/bun.lockb .
|
||||
COPY --from=prerelease /usr/src/app/package.json .
|
||||
COPY --from=prerelease /usr/src/app/.vinxi .vinxi
|
||||
COPY --from=prerelease /usr/src/app/.output .output
|
||||
|
||||
USER bun
|
||||
EXPOSE 3000/tcp
|
||||
ENTRYPOINT [ "bun", "run", "start" ]
|
90
infrastructure/app.bicep
Normal file
90
infrastructure/app.bicep
Normal file
|
@ -0,0 +1,90 @@
|
|||
import { Context } from 'types.bicep'
|
||||
|
||||
targetScope = 'resourceGroup'
|
||||
|
||||
param context Context
|
||||
param registry resource'Microsoft.ContainerRegistry/registries@2023-07-01'
|
||||
|
||||
var appName = 'app'
|
||||
var version = 'latest'
|
||||
|
||||
resource environment 'Microsoft.App/managedEnvironments@2024-03-01' = {
|
||||
name: 'acr-${context.locationAbbreviation}-${context.environment}-${context.projectName}'
|
||||
location: context.location
|
||||
properties: {
|
||||
appLogsConfiguration: {
|
||||
destination: 'azure-monitor'
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
resource app 'Microsoft.App/containerApps@2024-03-01' = {
|
||||
name: 'acr-${context.locationAbbreviation}-${context.environment}-${context.projectName}-app'
|
||||
location: context.location
|
||||
identity: {
|
||||
type: 'SystemAssigned'
|
||||
}
|
||||
properties: {
|
||||
environmentId: environment.id
|
||||
|
||||
configuration: {
|
||||
activeRevisionsMode: 'Single'
|
||||
|
||||
ingress: {
|
||||
external: true
|
||||
targetPort: 8080
|
||||
transport: 'http2'
|
||||
allowInsecure: false
|
||||
traffic: [
|
||||
{
|
||||
weight: 100
|
||||
latestRevision: true
|
||||
}
|
||||
]
|
||||
corsPolicy: {
|
||||
allowedOrigins: [
|
||||
// 'https://localhost:3000'
|
||||
'*'
|
||||
]
|
||||
allowCredentials: true
|
||||
allowedHeaders: ['*']
|
||||
allowedMethods: ['Get, POST']
|
||||
maxAge: 0
|
||||
}
|
||||
}
|
||||
registries: [
|
||||
{
|
||||
identity: 'system'
|
||||
server: registry.properties.loginServer
|
||||
}
|
||||
]
|
||||
}
|
||||
|
||||
template: {
|
||||
containers: [
|
||||
{
|
||||
image: '${registry.properties.loginServer}/${context.projectName}-${appName}:${version}'
|
||||
name: '${context.projectName}-${appName}'
|
||||
resources: {
|
||||
cpu: json('0.25')
|
||||
memory: '0.5Gi'
|
||||
}
|
||||
}
|
||||
]
|
||||
scale: {
|
||||
minReplicas: 1
|
||||
maxReplicas: 2
|
||||
rules: [
|
||||
{
|
||||
name: 'http-rule'
|
||||
http: {
|
||||
metadata: {
|
||||
concurrentRequests: '50'
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
11
infrastructure/bicepconfig.json
Normal file
11
infrastructure/bicepconfig.json
Normal file
|
@ -0,0 +1,11 @@
|
|||
{
|
||||
"experimentalFeaturesEnabled": {
|
||||
"assertions": true,
|
||||
"testFramework": true,
|
||||
"extensibility": true,
|
||||
"resourceDerivedTypes": true,
|
||||
"resourceTypedParamsAndOutputs": true,
|
||||
"sourceMapping": true,
|
||||
"symbolicNameCodegen": true
|
||||
}
|
||||
}
|
46
infrastructure/main.bicep
Normal file
46
infrastructure/main.bicep
Normal file
|
@ -0,0 +1,46 @@
|
|||
import { Context } from 'types.bicep'
|
||||
|
||||
targetScope = 'subscription'
|
||||
|
||||
param locationAbbreviation string
|
||||
param location string
|
||||
param environment string
|
||||
param projectName string
|
||||
param deployedAt string = utcNow('yyyyMMdd')
|
||||
|
||||
var context = {
|
||||
locationAbbreviation: locationAbbreviation
|
||||
location: location
|
||||
environment: environment
|
||||
projectName: projectName
|
||||
deployedAt: deployedAt
|
||||
}
|
||||
|
||||
resource calqueResourceGroup 'Microsoft.Resources/resourceGroups@2024-07-01' = {
|
||||
name: 'rg-${locationAbbreviation}-${environment}-${projectName}'
|
||||
location: location
|
||||
}
|
||||
|
||||
module monitoring 'monitoring.bicep' = {
|
||||
name: 'monitoring'
|
||||
scope: calqueResourceGroup
|
||||
params: {
|
||||
context: context
|
||||
}
|
||||
}
|
||||
|
||||
module registry 'registry.bicep' = {
|
||||
name: 'registry'
|
||||
scope: calqueResourceGroup
|
||||
params: {
|
||||
context: context
|
||||
}
|
||||
}
|
||||
|
||||
module app 'app.bicep' = {
|
||||
name: 'app'
|
||||
scope: calqueResourceGroup
|
||||
params: {
|
||||
context: context
|
||||
}
|
||||
}
|
11
infrastructure/monitoring.bicep
Normal file
11
infrastructure/monitoring.bicep
Normal file
|
@ -0,0 +1,11 @@
|
|||
import { Context } from 'types.bicep'
|
||||
|
||||
targetScope = 'resourceGroup'
|
||||
|
||||
param context Context
|
||||
|
||||
// resource monitoring 'Microsoft.___/___@___' = {
|
||||
// name: 'acr-${context.locationAbbreviation}-${context.environment}-${context.projectName}'
|
||||
// location: context.location
|
||||
// properties: {}
|
||||
// }
|
6
infrastructure/params/prod.bicepparam
Normal file
6
infrastructure/params/prod.bicepparam
Normal file
|
@ -0,0 +1,6 @@
|
|||
using '../main.bicep'
|
||||
|
||||
param locationAbbreviation = 'euw'
|
||||
param location = 'westeurope'
|
||||
param environment = 'prd'
|
||||
param projectName = 'calque'
|
16
infrastructure/registry.bicep
Normal file
16
infrastructure/registry.bicep
Normal file
|
@ -0,0 +1,16 @@
|
|||
import { Context } from 'types.bicep'
|
||||
|
||||
targetScope = 'resourceGroup'
|
||||
|
||||
param context Context
|
||||
|
||||
resource registry 'Microsoft.ContainerRegistry/registries@2023-07-01' = {
|
||||
name: 'acr-${context.locationAbbreviation}-${context.environment}-${context.projectName}'
|
||||
location: context.location
|
||||
sku: {
|
||||
name: 'Basic'
|
||||
}
|
||||
properties: {}
|
||||
}
|
||||
|
||||
output registry resource'Microsoft.ContainerRegistry/registries@2023-07-01' = registry
|
8
infrastructure/types.bicep
Normal file
8
infrastructure/types.bicep
Normal file
|
@ -0,0 +1,8 @@
|
|||
@export()
|
||||
type Context = {
|
||||
locationAbbreviation: string
|
||||
location: string
|
||||
environment: string
|
||||
projectName: string
|
||||
deployedAt: string
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue