https://github.com/hvalfangst/azure-static-react-website-triggering-functions
Static web app written in React which uploads CSV files to Azure storage blobs via HTTP-triggered Azure function, which is then processed by a blob-triggered function and its results stored in yet another blob.
https://github.com/hvalfangst/azure-static-react-website-triggering-functions
az-204 az-cli azure azure-functions azure-storage-account azure-storage-blob blob-trigger http-trigger javascript python react static-website
Last synced: 2 days ago
JSON representation
Static web app written in React which uploads CSV files to Azure storage blobs via HTTP-triggered Azure function, which is then processed by a blob-triggered function and its results stored in yet another blob.
- Host: GitHub
- URL: https://github.com/hvalfangst/azure-static-react-website-triggering-functions
- Owner: hvalfangst
- Created: 2024-11-21T22:25:57.000Z (3 months ago)
- Default Branch: main
- Last Pushed: 2025-02-05T09:19:33.000Z (16 days ago)
- Last Synced: 2025-02-05T10:26:46.734Z (16 days ago)
- Topics: az-204, az-cli, azure, azure-functions, azure-storage-account, azure-storage-blob, blob-trigger, http-trigger, javascript, python, react, static-website
- Language: Shell
- Homepage:
- Size: 596 KB
- Stars: 0
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Static web app invoking Azure functions
The aim of this repository is to demonstrate how to deploy a [static website](client/src/App.js) written in React to a [Storage Blob](https://learn.microsoft.com/en-us/azure/storage/blobs/storage-blob-static-website). The
hosted SPA allows users to upload CSV files containing demographic and financial data about individuals. The files are uploaded to a storage blob by calling an HTTP-triggered Azure Function with the appropriate output bindings.
Once the CSV has been uploaded to the storage blob, another, blob-triggered Azure Function calculates correlations between various variables, such as experience, state, gender, and income.
The computed statistics are then stored in a new blob container, which is used to serve the results to the user.
These two functions are defined in the python script [function_app.py](hvalfangst_function/function_app.py) - which is the main entrypoint of our Azure Function App instance.The SPA is protected with Oauth2.0 authorization code flow with PKCE and OIDC. The user is redirected to the Azure AD login page, where they must authenticate before being redirected back to the SPA.
The associated Azure infrastructure is deployed with a script (more on that below).
A branch-triggered pipeline has been set up to deploy our code to the respective Azure resources using a GitHub Actions Workflows [script](.github/workflows/deploy_to_azure.yml).
The two functions are deployed using the Function App's associated **publish profile**, whereas the static web app is deployed using a **Service Principal** configured with a **Federated Credential**.
Note that the static website is hosted directly from a storage blob, as our associated storage container has been configured to serve static websites in our resource provisioning script.
Thus, deploying the website is simply a matter of uploading the static files to the designated blob container.## Requirements
- **Platform**: x86-64, Linux/WSL
- **Programming Languages**: [React](https://reactjs.org/docs/getting-started.html), [Python 3](https://www.python.org/downloads/)
- **Cloud Account**: [Azure](https://azure.microsoft.com/en-us/pricing/purchase-options/azure-account)
- **Resource provisioning**: [Azure CLI](https://learn.microsoft.com/en-us/cli/azure/)## Allocate resources
The shell script [allocate_resources](infra/allocate_resources.sh) creates Azure resources using the Azure CLI in conjunction with a
[Bicep](https://learn.microsoft.com/en-us/azure/azure-resource-manager/bicep/overview?tabs=bicep) template [file](infra/main.bicep).It will create the following hierarchy of resources:
```mermaid
graph TD
A[Subscription]
A --> B[Resource Group]
B --> C[Storage Account]
C --> D[Blob Container]
D -->|Static Website Hosting| H[index.html]
B --> E[App Service Plan]
E -->|Hosts| G[Function App]
G -->|Uses| F[Application Insights]A -->|Contains| B
B -->|Contains| C
C -->|Contains| D
B -->|Contains| E
B -->|Contains| F
```## Registrations
In addition to the resources listed above, the script will also create a **service principal** and two Microsoft Entra ID **app registrations.**### Service Principal for GitHub Actions
The service principal has been assigned contributor role to our resource group, which is sufficient in order to deploy the static web app to the storage blob.
It has been assigned a federated credential configured to work with this repository as it is utilized in our CI/CD [GitHub Actions Workflow script](.github/workflows/deploy_to_azure.yml).### App Registration for Azure Function App
Exposes the scopes **Csv.Writer** and **Csv.Reader** under URI **api://hvalfangst-function-app**
data:image/s3,"s3://crabby-images/9b279/9b279be16050531fd092f9ba24585671cbd0a6c9" alt="img.png"
### App Registration for SPA
Has a redirect URI configured to the static web app's URL and the permissions **Csv.Writer** and the OIDC ones.
data:image/s3,"s3://crabby-images/abf64/abf64f06f6fa41f66abba2018a55b82d04ed142c" alt="img_1.png"
## GitHub secrets
Four secrets are required in order for the GitHub Actions Workflow script to deploy the code to the Azure resources.
As may be observed in the [script](.github/workflows/deploy_to_azure.yml), these are:- **AZURE_GITHUB_SP_CLIENT_ID**: Used to authenticate the service principal in order to deploy the static web app
- **AZURE_SUBSCRIPTION_ID**: Used to authenticate the service principal in order to deploy the static web app
- **AZURE_TENANT_ID**: Used to authenticate the service principal and for the OIDC flow in the React SPA
- **PUBLISH_PROFILE**: Used to deploy our two functions to the Azure Function App
- **STATIC_WEB_APP_CLIENT_ID**: Used in the React SPA for OIDC authentication### Subscription and Tenant ID
The **subscription ID** and **tenant ID** is found by running the following Azure CLI command:```bash
az account show --query id
az account show --query tenantId
```### Publish Profile
The publish profile may be obtained by navigating to the Azure Portal, selecting the Azure Function App, and clicking on **Get publish profile**.### Azure GitHub Service Principal
The service principal used for GitHub Actions is created as part of our resource provisioning script and
as thus should be displayed in the terminal output as such:data:image/s3,"s3://crabby-images/78a74/78a748d5d4f669fc6c7cf075647b733b47b1e344" alt="sp_terminal.png"
### Static Web App Client ID
Similarly, the **client ID** for the static web app is created as part of our resource provisioning script and outputted to the terminal as such:data:image/s3,"s3://crabby-images/244a5/244a57b3b5b4ebb3fb8d3611e8b6546d43210ffd" alt="spa_terminal.png"
## Usage
After provisioning resources, setting up secrets, and pushing the code to the repository, one
may access the static web app by navigating to the following URL:https://hvalfangststorageaccount.z6.web.core.windows.net, which results in the following.
data:image/s3,"s3://crabby-images/f11f4/f11f476321a9d1459868377d4377c712b27f7920" alt="csv_uploader.png"
Click on **Sign In** to initiate the OIDC flow - which redirects to the Azure AD permission consent screen.
data:image/s3,"s3://crabby-images/379eb/379eb96af59577e37e68e1f5b06cbce390e04cfa" alt="oidc.png"
Clik on **Accept** and check off the **Consent on behalf of your organization** box to be redirected back to the SPA, where you will be greeted with the following.
data:image/s3,"s3://crabby-images/ffef9/ffef907e779516f28373122a0e47a5455cb22ab5" alt="authenticated_user.png"
Proceed to click on **Upload** to choose a file to upload. Pick the CSV file named [input](input.csv) which has been provided for this purpose.
data:image/s3,"s3://crabby-images/382fe/382fe395a77def63219609012c0b012e3cc595e7" alt="choose_file.png"
data:image/s3,"s3://crabby-images/5d02d/5d02d36f87becc00fd94c43c67ae8634da5fa8b7" alt="file_chosen.png"
The file name will be displayed in the input field. Click on **Upload** to attempt to upload the file to the storage blob.
If the upload was successful, the following message will be displayed.
data:image/s3,"s3://crabby-images/f5a6d/f5a6d69919ba8e1f71e8a852bd664eeb14884722" alt="upload_successful.png"