Skip to content

Commit

Permalink
Initial commit
Browse files Browse the repository at this point in the history
  • Loading branch information
arturogonzalezm committed Feb 4, 2024
1 parent 67af481 commit 9e65b37
Show file tree
Hide file tree
Showing 16 changed files with 826 additions and 1 deletion.
8 changes: 8 additions & 0 deletions .funcignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
.git*
.vscode
__azurite_db*__.json
__blobstorage__
__queuestorage__
local.settings.json
test
.venv
29 changes: 29 additions & 0 deletions .github/workflows/pylint.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
name: Pylint

on: [ push ]

jobs:
build:
runs-on: ubuntu-latest
strategy:
matrix:
python-version: [ "3.11" ]
steps:
- uses: actions/checkout@v3
- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v3
with:
python-version: ${{ matrix.python-version }}
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install -r requirements.txt
- name: Code Scores
run: find . -name '*.py' -print -exec pylint {} \;
- name: Analysing the code with PyLint
run: |
pylint $(find . -name '*.py') --fail-under=8.0
- name: Upload coverage reports to Codecov
uses: codecov/codecov-action@v3
env:
CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
146 changes: 146 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
.idea
.idea/*
*.iml
*.ipr
venv/*
venv
.venv
.venv/*

# Byte-compiled / optimized / DLL files
__pycache__/
*.py[cod]
*$py.class

# C extensions
*.so

# Distribution / packaging
.Python
build/
develop-eggs/
dist/
downloads/
eggs/
.eggs/
lib/
lib64/
parts/
sdist/
var/
wheels/
pip-wheel-metadata/
share/python-wheels/
*.egg-info/
.installed.cfg
*.egg
MANIFEST

# PyInstaller
# Usually these files are written by a python script from a template
# before PyInstaller builds the exe, so as to inject date/other infos into it.
*.manifest
*.spec

# Installer logs
pip-log.txt
pip-delete-this-directory.txt

# Unit test / coverage reports
htmlcov/
.tox/
.nox/
.coverage
.coverage.*
.cache
nosetests.xml
coverage.xml
*.cover
.hypothesis/
.pytest_cache/
.idea/*
.idea/

# Translations
*.mo
*.pot

# Django stuff:
*.log
local_settings.py
db.sqlite3

# Flask stuff:
instance/
.webassets-cache

# Scrapy stuff:
.scrapy

# Sphinx documentation
docs/_build/

# PyBuilder
target/

# Jupyter Notebook
.ipynb_checkpoints

# IPython
profile_default/
ipython_config.py

# pyenv
.python-version

# pipenv
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
# However, in case of collaboration, if having platform-specific dependencies or dependencies
# having no cross-platform support, pipenv may install dependencies that don’t work, or not
# install all needed dependencies.
#Pipfile.lock

# celery beat schedule file
celerybeat-schedule

# SageMath parsed files
*.sage.py

# Environments
.env
.venv
env/
venv/
ENV/
env.bak/
venv.bak/

# Spyder project settings
.spyderproject
.spyproject

# Rope project settings
.ropeproject

# mkdocs documentation
/site

# mypy
.mypy_cache/
.dmypy.json
dmypy.json

# Pyre type checker
.pyre/

# Azure Functions artifacts
bin
obj
appsettings.json
local.settings.json

# Azurite artifacts
__blobstorage__
__queuestorage__
__azurite_db*__.json
.python_packages
200 changes: 199 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
@@ -1 +1,199 @@
# azure_function
[![codecov](https://codecov.io/gh/arturogonzalezm/azure_function_iac/graph/badge.svg?token=V0XK325ILC)](https://codecov.io/gh/arturogonzalezm/azure_function_iac)
[![License: MIT](https://img.shields.io/badge/License-MIT-purple.svg)](https://github.com/arturogonzalezm/azure_function_iac/blob/master/LICENSE)
[![Read the Docs](https://img.shields.io/readthedocs/:packageName)](https://github.com/arturogonzalezm/azure_function_iac/blob/master/README.md)
[![Python test with Github Actions](https://github.com/arturogonzalezm/azure_function_iac/actions/workflows/pylint.yml/badge.svg)](https://github.com/arturogonzalezm/azure_function_iac/actions/workflows/pylint.yml)



# Azure Function and Storage Account #

This project is a proof of concept to create an Azure Function App, Key Vault and a Storage Account using ARM Templates.

## SaaS Overview

- The Azure Function App will consume an external REST API
- THE Azure Function App will retrieve the REST API secrets from the Key Vault.
- The Azure Function App will run in micro-batches every minute.
- The Azure Function App will ingest the JSON data into the Storage Account.
- The JSON files stored in the Storage Account will be stored in Snowflake.

```mermaid
sequenceDiagram
participant Process as Process
participant ExternalAPI as External REST API
participant KeyVault as Key Vault
participant FunctionApp as Azure Function App
participant StorageAccount as Storage Account
participant Snowflake as Snowflake
Process->>ExternalAPI: Consume
ExternalAPI->>FunctionApp: Return data
FunctionApp->>KeyVault: Retrieve REST API secrets
KeyVault->>FunctionApp: Return secrets
loop Every Minute
FunctionApp->>FunctionApp: Run in micro-batches
end
FunctionApp->>StorageAccount: Ingest JSON data
StorageAccount->>Snowflake: Store JSON files
```

- This project is designed to integrate with Azure Functions and WebEOC API.
- This project integrates with Azure Functions to interact with WebEOC API and Azure Blob Storage.
- It includes a logging utility, a client for WebEOC API, and functionalities to store data and logs in Azure Blob Storage.
- The main application is an Azure Function that periodically pulls data from the WebEOC API, processes it,
and then stores both the data and associated logs.

## Workflow

![workflow](images/flow_chart.drawio.png)

### Features
- Custom logging setup compatible with Azure Functions and Application Insights.
- Data pulling from WebEOC API.
- Data and log ingestion to Azure Blob Storage.
- Environment-specific configuration and secret management.

## Installation
Provide instructions on how to set up and install your project. This might include:
- Cloning the repository.
- Setting up a virtual environment.
- Installing dependencies.

```bash
git clone azure_function_iac
cd azure_function_iac
git init
git remote add origin
git pull origin master
```

```bash
python -m venv .venv
.\.venv\Scripts\Activate.ps1
```

```bash
python -m pip install --upgrade pip setuptools wheel
pip install -r requirements.txt
```

## Key Components

```class LoggerUtility```: Provides logging functionalities tailored for Azure Functions.

```configure_logger()```: Configures the logger with default settings suitable for Azure Functions.

```get_logger(name)```: Returns a configured logger instance.

![logging](images/logging_diagram.drawio.png)

```class WebEOCBlobStorage()```: Manages the storage of data and logs in Azure Blob Storage.

```ingest_to_blob(data, blob_name, container_name)```: Uploads data to Azure Blob Storage.

```write_logs_to_blob(log, blob_name, container_name)```: Uploads logs to Azure Blob Storage.

```create_output_filename(log, blob_name, container_name)```: Generates a filename based on a specified format and current time in Perth time zone.

```class WebEOCClient()```: Client for interacting with the WebEOC API.

```pull_data()```: Pulls data from the WebEOC API.

![function](images/function_app.drawio.png)

## WebEOCClient Class
The WebEOCClient class is designed to interface with the WebEOC API. It manages authentication and data retrieval specific to a deployment environment. This class uses Azure services for secure credential storage and retrieval, ensuring safe access to the WebEOC API.

### WebEOCClient Key Features
Initialization
- Environment Parameter: The client is initialized with a environment parameter, allowing it to adapt its behavior based on the deployment environment (e.g., production, development, or user acceptance testing).
- Logging: Utilizes the LoggerUtility for logging, aiding in monitoring and debugging.

Credential Management
- Azure Integration: Leverages Azure Key Vault for secure credential storage and retrieval.
- Credential Caching: Utilizes class-level caching for Azure credentials to optimize performance.

Authentication
- Method: authenticate
- Functionality: Authenticates the session with the WebEOC API using credentials specific to the deployment environment.
- Error Handling: Includes comprehensive error handling to manage exceptions during the authentication process.

Configuration Retrieval
- Several methods (get_username, get_password, get_position, etc.) retrieve configuration details like username, password, and position, based on the deployment environment.
- Error Handling: Includes comprehensive error handling to manage exceptions during the configuration retrieval process.

Data Retrieval
- Method: pull_data
- Functionality: Pulls data from the WebEOC API based on the specified board and display names relevant to the deployment environment.
- Error Handling: Robust error handling to address issues during the data retrieval process.

Usage
```python
client = WebEOCClient(environment='uat')
data = client.pull_data()
```

![api_client](images/API_client.drawio.png)

## WebEOCBlobStorage Class

The WebEOCBlobStorage class is designed to handle interactions with Azure Blob Storage within the context of the WebEOC project. It primarily focuses on uploading data and log files to Azure Blob Storage, ensuring that data is stored securely and logs are kept for monitoring and debugging purposes.

### WebEOCBlobStorage Key Features

Initialization
- Connection String: Retrieves the Azure Blob Storage connection string from environment variables.
- Logger: Utilizes LoggerUtility for logging, aiding in tracking the process and handling errors.
- Blob Service Client: Initializes a Blob Service Client with the given connection string.

Output Filename Generation
- Method: create_output_filename
- Functionality: Generates a unique output filename based on the current date and time in the 'Australia/Perth' timezone.
- Usage: This filename is used when uploading data to Azure Blob Storage.
- Error Handling: Includes comprehensive error handling for various scenarios like missing data, blob already exists, container/blob not found, and general Azure storage errors.

Data Ingestion
- Method: ingest_to_blob
- Parameters: Accepts the data to be uploaded.
- Functionality: Uploads the given data to Azure Blob Storage. If the data is not a string, it converts it into JSON format.
- Error Handling: Includes comprehensive error handling for various scenarios like missing data, blob already exists, container/blob not found, and general Azure storage errors.

Log Writing to Blob
- Method: write_logs_to_blob
- Parameters: Start time, end time, duration, status, records pulled, log level, and log message.
- Functionality: Creates a structured log file and uploads it to Azure Blob Storage. The log file includes details such as the filename, process duration, status, number of records pulled, and a log message.
- Conditional Logging: Includes log level and log message only for "INFO" or "ERROR" levels.

Usage
```python
storage = WebEOCBlobStorage()
storage.ingest_to_blob(data)
storage.write_logs_to_blob(start_time, end_time, duration, status, records_pulled, log_level, log_message)
```

![blob_client](images/blob_client.drawio.png)

## TODO: Azure Resources(IaaS) Overview

- Azure Function App will be deployed by using ARM Templates.
- Key Vault will be deployed by using ARM Templates.
- Storage Account will be deployed by using ARM Templates.
- The ARM Templates will be deployed by using Azure DevOps.

```mermaid
sequenceDiagram
participant AzureDevOps as Azure DevOps
participant ARM as ARM Templates
participant KeyVault as Key Vault
participant StorageAccount as Storage Account
participant FunctionApp as Azure Function App
AzureDevOps->>ARM: Deploy
ARM->>KeyVault: Deploy Key Vault
ARM->>StorageAccount: Deploy Storage Account
ARM->>FunctionApp: Deploy Azure Function App
```

-------------------------------------------------------------------------------------------------------------------
### Author: ###
This project was made by [**Arturo Gonzalez**]
Loading

0 comments on commit 9e65b37

Please sign in to comment.