The request process workflow tool for the RedHat SSO Dev Exchange service
This is a tool to streamline the single-sign-on request process for new keycloak clients. It consists of:
- A react frontend hosted on github pages. See the /app subdirectory for more.
- A serverless AWS backend, including lambda functions, API gateway, and an RDS database using postgres. All AWS resources are defined in terraform and can be found in the /terraform subdirectory, while the code for the lambda functions are in the /lambda subdirectory.
- Github actions scripts that handle the creation of clients through terraform, as well as communicate events to the backend.
keycloak: New keycloak clients are created and version controlled through terraform. For production, the terraform scripts to create clients are in the github sso-terraform repository, and for development they are in sso-terraform-dev. To create new clients, this application adds terraform files to those repositories (for the prod and dev versions respectively) and merges them to have terraform cloud add the new clients to keycloak.
Github: This application uses github actions to trigger client creation, as well as github pages to host the frontend application. The following three repositories are used in the workflow:
- sso-requests: Source code of the application. Github pages hosts the production app.
- sso-terraform-dev: Contains terraform scripts defining the dev keycloak clients. Contains actions this application triggers to add new clients through terraform cloud. Github pages is also used to host the dev frontend.
- sso-terraform: Same logic as sso-terraform-dev, but for the production environment.
AWS: The backend application is provisioned through AWS. It uses API Gateway, AWS Lambda, and RDS.
Terraform: All infrastructure for this application is created and version controlled through terraform. The application is launched into an existing AWS zone, and uses the predefined VPC and subnets.
The general workflow for new client creation is:
- A user authenticates with their IDIR and fills in a form for their new client
- Data is validated in the backend lambda function (see
lambda/app
) and a new github workflow is kicked off (in the sso-terraform repository). See the repositories yaml file for details. - The request generates terraform files for the new client and creates a pull request to the default branch. The event (pr creation event) is logged to to the lambda backend. The pull request is labelled as
auto-generated
. - New auto-generated pull requests start the terraform workflow. This workflow
will run
terraform plan
, and validate that the resources created follow reasonable constraints. If the plan is valid, it will auto-merge the pull request to the main branch. A terraform-plan event is logged to the backend. - Merge events to main also trigger the terraform workflow.
Terraform will attempt to apply the changes, creating a new client. An
apply
event will be logged to the backend.
The process for updating clients is similar (see the github scripts for more details).
See the below diagram for more information:
Front-end: The front end application is built with nextjs (a react framework). It is a static application hosted on github pages. A static single-page-application was chosen to deliver a fast user experience, take advantage of pre-built bcgov components, and to interact as an independant layer from the backend.
Back-end: The backend uses AWS serverless technologies, which was chosen to reduce cost (with serverless, you only pay for what you use) since this application is expected to have low traffic. It uses:
- Lambda: Lambda functions are used to handle backend logic on API events
- API Gateway: API Gateway supports serverless infrastructure, and is used to proxy API events to the lambda functions.
- RDS: An amazon relational database that supports serverless infrastructure.
The following events are logged with timestamps as a request workflow runs:
- Pull Requests: When a new pull request workflow is triggered, a success or failure event is logged
- Terraform Plan: When a terraform plan completes, a success or failure event is logged.
- Terraform Apply: When a terraform apply completes, a success or failure event is logged.
These events are saved in an event table and can be used to monitor the cycle a request has gone through. In addition, requests are timestamped for time created and time updated.
In the /app and /lambda directories install dependencies with npm i
or yarn
.
Backend
To run this app, you will need to setup your terraform infrastructure. The code in the terraform directory
assumes one available VPC exists with available subnets in two zones. To set the names for your subnets the terraform variables
subnet_a
and subnet_b
should be set.
The lambda functions are written in typescript, and to compile and bundle them you can run make lambda-all
from the lambda directory.
Once they are compiled, from the terraform directory you can run terraform apply
to build the backend.
Frontend See the example env file for an example environment variable setup. You will require:
- API_URL: The API endpoint of your API Gateway
- SSO_URL= The URL of the keycloak realm
- SSO_CLIENT_ID: The ID of your keycloak client
- SSO_REDIRECT_URI: Your redirect url for after users have authenticated. For local development use
localhost:3000
- SSO_AUTHORIZATION_RESPONSE_TYPE: Set to
code
- SSO_AUTHORIZATION_SCOPE: Set to
openid
- SSO_TOKEN_GRANT_TYPE: Set to
authorization_code
The app can then be run locally by running yarn dev
This repository requires the following secrets:
DEV_KEYCLOAK_DEV_CLIENT_SECRET
: The client secret for the sandbox keycloaks dev environment.DEV_KEYCLOAK_TEST_CLIENT_SECRET
: The client secret for the sandbox keycloaks test environment.DEV_KEYCLOAK_PROD_CLIENT_SECRET
: The client secret for the sandbox keycloaks prod environment.PROD_KEYCLOAK_DEV_CLIENT_SECRET
: The client secret for the production keycloaks dev environment.PROD_KEYCLOAK_TEST_CLIENT_SECRET
: The client secret for the production keycloaks test environment.PROD_KEYCLOAK_PROD_CLIENT_SECRET
: The client secret for the production keycloaks prod environment.GH_ACCESS_TOKEN
: Access token for the github service account (used to trigger workflows insso-terraform
).GH_SECRET
: The secret used to authorize requests between github actions (fromsso-terraform
) to the API.TFC_TEAM_TOKEN
: The token used to run terraform cloud.
In Addition, the following secrets are required for sso-terraform
and sso-terraform-dev
to work with this app:
GH_ACCESS_TOKEN
: Access token for the github service account (used to run github API calls).GH_SECRET
: The secret used to authorize requests between github actions (fromsso-terraform
) to the API.
This repository has frontend tests using jest and react testing library, and backend tests run with jest.
Tests are run in CI in the test.yml
file.
To run the frontend tests, from the app
directory run yarn test
To run the backend integration tests, you willneed to have a local postgres database running.
Ensure you have started the server with pg_ctl start
.
For the first time running the tests, the database will need to be created.
Run ./setup.sh ssodb
from the /db
directory to initialize it Note: you may need to run chmod +x
to give necessary permissions.
To provide the local database connection string, in your console run export DATABASE_URL=postgresql://localhost:5432/ssodb
.
Then running yarn test
from the lambda directory will run the necessary migrations and test suites.
Error codes we use for the application:
E01
: There is an application in thesso-terraform
repository that cannot be applied, blocking new requestsE02
: The user has a token in their session storage that is invalid
For now, reporting information can be gathered directly from the AWS Data API. To use, you will need access to the AWS platform as well as the database credentials.
- In the AWS console, navigate to the
RDS
dashboard and select DB Clusters from the resources panel. - Select
aurora-db-postgres
. From the top rightactions
dropdown select query. - You will be prompted for the db host, db name, username and password. Enter these and select
connect
(it may take some time to connect) - From the saved queries, select the one you would like to use (refer to descriptions for information)
Queries
-- Count of clients in draft
select count(*) from requests where status='draft';
-- Count of clients awaiting approval (note: currently auto-approve but will apply when bceid is added)
select count(*) from requests where status='submitted';
-- Count of clients completed
select count(*) from requests where status='applied';
-- Time request was initially submitted
select events.created_at from events join requests on requests.id = events.request_id where requests.id=1 and events.event_code = 'request-pr-success' order by events.created_at asc limit 1;
-- (initial )time request was fulfilled (dev, test, and prod)
select events.created_at from events join requests on requests.id = events.request_id where requests.id={id_here} and events.event_code = 'request-apply-success';