Skip to content
Open
Show file tree
Hide file tree
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
39 changes: 39 additions & 0 deletions .github/workflows/ci-merge.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
on: [pull_request, workflow_dispatch]

jobs:
build_frontend:
runs-on: ubuntu-latest
environment: production

steps:
- name: Checkout code
uses: actions/checkout@v4

- name: Install dependencies
run: npm install
working-directory: ./frontend

- name: Build React app
run: npm run build
working-directory: ./frontend

- name: Run lint
run: npm run lint
working-directory: ./frontend

build_backend:
runs-on: ubuntu-latest
environment: production

steps:
- name: Checkout repository
uses: actions/checkout@v4

- name: Setup .NET SDK 9.0.x
uses: actions/setup-dotnet@v4
with:
dotnet-version: '9.0.x'

- name: Build .NET app
run: dotnet build --configuration Release
working-directory: ./backend
117 changes: 117 additions & 0 deletions .github/workflows/deploy-infra-and-apps.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,117 @@
on:
push:
tags:
- 'v[0-9]+.[0-9]+.[0-9]+'
workflow_dispatch:

permissions:
# Require write permission to Fetch an OIDC token (required for federated credential) and write it
# It will be automatically used on actions / cli that needs it
id-token: write

env:
AZURE_RG_NAME: rg-${{ vars.PROJECT_NAME }}-${{ vars.AZURE_RESOURCE_IDENTIFIER }}
APP_VERSION: ${{ github.ref_name }}

jobs:
deploy_infrastructure:
runs-on: ubuntu-latest
environment: production # bind the job to the production environment

outputs:
appServiceName: ${{ steps.bicep_deploy.outputs.appServiceName }}
staticWebAppName: ${{ steps.bicep_deploy.outputs.staticWebAppName }}

steps:
- name: Checkout repository
uses: actions/checkout@v4

- name: Login to Azure
uses: azure/login@v2
with:
client-id: ${{ secrets.AZURE_CLIENT_ID }}
tenant-id: ${{ secrets.AZURE_TENANT_ID }}
subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
enable-AzPSSession: true

- name: Create resource group if not exists
run: |
az group show --name ${{ env.AZURE_RG_NAME }} ||
az group create --name ${{ env.AZURE_RG_NAME }} --location ${{ secrets.AZURE_REGION }}

- name: Deploy bicep
id: bicep_deploy
uses: azure/arm-deploy@v2
with:
subscriptionId: ${{ secrets.AZURE_SUBSCRIPTION }}
region: ${{ secrets.AZURE_REGION }}
template: ./infrastructure/main.bicep
parameters: project=${{ vars.PROJECT_NAME }} location=${{ secrets.AZURE_REGION }} swaLocation=${{ secrets.AZURE_SWA_REGION }} identifier=${{ vars.AZURE_RESOURCE_IDENTIFIER }}
resourceGroupName: ${{ env.AZURE_RG_NAME }}
- name: Print version
run: echo "App version est $APP_VERSION"

deploy_backend:
runs-on: ubuntu-latest
needs: deploy_infrastructure
environment: production

steps:
- name: Checkout repository
uses: actions/checkout@v4

- name: Setup .NET SDK 9.0.x
uses: actions/setup-dotnet@v4
with:
dotnet-version: '9.0.x'

- name: Publish the app
run: dotnet publish -c Release --property:PublishDir=publish # Publish the app to the API project publish folder
working-directory: ./backend # Specify where to find the solution file in repository

- name: Login to Azure
uses: azure/login@v2
with:
client-id: ${{ secrets.AZURE_CLIENT_ID }}
tenant-id: ${{ secrets.AZURE_TENANT_ID }}
subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}

- name: Deploy backend to App Service
uses: azure/webapps-deploy@v2
with:
app-name: ${{ needs.deploy_infrastructure.outputs.appServiceName }} # Access to the previous job output to get the appServiceName deployed with bicep
package: ./backend/ParkNDeploy.Api/publish # Path to the previously published app

deploy_frontend:
runs-on: ubuntu-latest
needs: deploy_infrastructure
environment: production

steps:
- name: Checkout repository
uses: actions/checkout@v4

- name: Build the app
run: npm install && npm run build
working-directory: ./frontend

- name: Login to Azure
uses: azure/login@v2
with:
client-id: ${{ secrets.AZURE_CLIENT_ID }}
tenant-id: ${{ secrets.AZURE_TENANT_ID }}
subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}

- name: Get Static Web App deployment token
run: |
SWA_DEPLOYMENT_TOKEN=$(az staticwebapp secrets list -n ${{ needs.deploy_infrastructure.outputs.staticWebAppName }} -o tsv --query properties.apiKey)
echo SWA_DEPLOYMENT_TOKEN=$SWA_DEPLOYMENT_TOKEN >> $GITHUB_ENV

- name: Deploy frontend to Static Web App
uses: azure/static-web-apps-deploy@v1
with:
azure_static_web_apps_api_token: ${{ env.SWA_DEPLOYMENT_TOKEN }}
app_location: frontend/dist
action: upload
skip_app_build: true
skip_api_build: true
56 changes: 56 additions & 0 deletions infrastructure/main.bicep
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
targetScope = 'resourceGroup' // We'll deploy the resources in the provided resource group

// Parameters to easily construct resource names
param location string
param project string

// Here we'll add an identifier to create a unique name for the App Service Plan, for example your trigram, so that everyone could deploy his own parkndeploy instance
param identifier string

// Create the AppServicePlan through the AppServicePlan module
module appServicePlan 'modules/appServicePlan.bicep' = {
name: 'appServicePlan'
params: {
location: location
project: project
identifier: identifier
}
}

// Create the AppService through the AppService module
module appService 'modules/appService.bicep' = {
name: 'appService'
params: {
location: location
project: project
identifier: identifier
planId: appServicePlan.outputs.planId // Use the appServicePlan output to get its id back => an App Service needs to reference its App Service Plan
}
}

param swaLocation string // Static Web App locations are limited, we need to add another variable

// App Service & App Service Plan creation
// ...

// Create the Static Web App through the StaticWebApp module
module staticWebApp 'modules/staticWebApp.bicep' = {
name: 'staticWebApp'
params: {
location: swaLocation
project: project
identifier: identifier
}
}

module staticWebAppBackend 'modules/staticWebAppBackend.bicep' = {
name: 'staticWebAppBackend'
params: {
backendBindedResourceId: appService.outputs.appServiceId
swaName: staticWebApp.outputs.swaName
location: location
}
}

output appServiceName string = appService.outputs.appServiceName // Export AppServiceName in order to deploy the API later on
output staticWebAppName string = staticWebApp.outputs.swaName // Export StaticWebAppName in order to deploy the Frontend late
24 changes: 24 additions & 0 deletions infrastructure/modules/appService.bicep
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
param location string
param project string
param identifier string

// App Service Plan identifier that will host our App Service
param planId string

resource app 'Microsoft.Web/sites@2022-03-01' = {
name: '${project}-app-${identifier}'
location: location

properties: {
serverFarmId: planId
reserved: true

siteConfig: {
linuxFxVersion: 'DOTNETCORE|9.0' // Specify to setup the .NET Core 9.0 runtime (used by our backend API) on the Linux machine under the hood
}
}
}

output appServiceName string = app.name // Export the App Service name for deployment

output appServiceId string = app.id
20 changes: 20 additions & 0 deletions infrastructure/modules/appServicePlan.bicep
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
param location string
param project string
param identifier string

resource plan 'Microsoft.Web/serverfarms@2022-09-01' = {
name: '${project}-plan-${identifier}'
location: location

sku: {
name: 'F1' // We use F1 pricing plan (free one) as we don't need specific features
}

kind: 'app,linux' // Allow to deploy on an App Service using Linux OS

properties: {
reserved: true // Specifity of App Service with Linux OS
}
}

output planId string = plan.id // Export the App Service identifier
17 changes: 17 additions & 0 deletions infrastructure/modules/staticWebApp.bicep
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
param location string
param project string
param identifier string

resource swa 'Microsoft.Web/staticSites@2024-04-01' = {
name: '${project}-swa-${identifier}'
location: location

sku: {
name: 'Standard'
tier: 'Standard'
}

properties: {} // Even empty, it's mandatory ...
}

output swaName string = swa.name // Expose Static Web App name as we did for App Service for deployment purpose
11 changes: 11 additions & 0 deletions infrastructure/modules/staticWebAppBackend.bicep
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
param backendBindedResourceId string
param swaName string
param location string

resource staticWebAppBackend 'Microsoft.Web/staticSites/linkedBackends@2022-03-01' = {
name: '${swaName}/backend'
properties: {
backendResourceId: backendBindedResourceId
region: location
}
}