Welcome to the codebase for BC's Housing Permit Portal
The codebase uses Rails on the back-end, and React/MST stack on the front-end.
Ensure you have the following:
- Ruby 3.2.2
- Postgres 13+
- Redis
- Node 20.10+
(Alternatively run this locally with docker compose
see section further down)
Please enable git LFS in order to properly clone the repo.
To make things easier to develop on various platforms locally, there is a docker compose
script for the app that can be run. This will setup all critical dependencies locally and bring the app up mounted to your filesystem for live development. There are a few caveats to this to note which will are mentioned at the end of this section
Prerequisites
- Download and install Docker Desktop for your system
- Make sure that you can run both
docker
anddocker compose
in your terminal and this is working correctly - Since some prefer to run this locally without Docker (see section below) this setup is aimed to preserve the ability to run this application both ways. As such, this uses a separate
Dockerfile.dev
and ENV file.env.docker_compose
that we will use for the purposes of running locally only - Ensure that you have a local copy of
.env.docker_compose
- an example is provided in.env_example.docker_compose
which provides the minimal ENV configuration to start the app without some services (see first point in Caveats section for more info) - We currently ONLY allow keycloak login have removed the ability to login via e-mail. To ensure you can login to the app make sure you have a valid Keycloak development ENV secrets set up so that you can properly log in. (See
.env_example.docker_compose
and What is Keycloak at BC Government)
Instructions
- Clone this repo with git to your local machine
- Run:
docker compose up
(this will start up all related services including Vite for HMR) - (If this is the first run of the application, create the database)
docker compose exec app bundle exec rails db:create
- (If this is the first run of the application OR if there are new migrations to run) Run migrations:
docker compose exec app bundle exec rails db:migrate
- (If this is the first run of the app or seeds have changed, you can rerun the seeder). Generate seed data:
docker compose exec app bundle exec rails db:seed
- Things should now be running. You can isolate / look at logs for various containers using
docker compose logs -f app
for example for the app logs (or any other service) - The app should reflect changes live if you edit the files in the folder (both Ruby and JS) since its mounted live on the filesystem
- If you want to run a
binding.pry
ordebugger
you can do so in the code, and then rundocker compose attach app
to interact with the debugging interface.
NOTES / Caveats
- A minimal set of ENV vars that reference various services (eg. Redis, Postgres, etc.) are defaulted in the
docker-compose.yml
for dev purposes, the app also loads other ENV vars from.env.docker_compose
so ensure that you have those setup if you are trying to use functionality related to that (eg. CHES keys for email sending, BCEID / keycloak stuff, etc.) - Local development (see above) uses
letter_opener
/launchy
to view emails locally. These do not work within the dockerized environment, and we haven't yet put in a workaround for this - This local dockerized version does not contain a service for
Consigno Verifio Notarius Server
which is proprietary licensed software that helps with validating PDFs. Therefore all functionality around this will not work. You can either run the service manually outside Docker and refer to it using the ENV vars, or find a private repository that has the image that you can then run indocker-compose
- The local dockerized version does not setup Minio (which can be used to locally to mock Object Storage as it is S3 compatible - see instructions for local setup below). You can set this up manually outside the docker setup or add your own docker compose service for it. Just hook it up via the ENV vars (see
.env.example
) or simply switch the app to use actualBCGOV_OBJECT_STORAGE
if you have a bucket allocation already.
- Install Dependencies:
bundle install
andnpm install
- Ensure you have a
.env
file with required variables (reference.env.example
) - Set up Database credentials following the official Rails guide. One of the ways is to have an environment variable called
DATABASE_URL
- (Only first time) Create a database:
rails db:create
- (Only first time or if there are changes) Run migrations:
rails db:migrate
- (Only first time or if there are changes) Generate seed data:
rails db:seed
- Start the server:
rails s
- Start the front-end dev server for hot-reloading:
npm run dev
The app uses the Sidekiq library for background job processing. To run this locally:
- Ensure you have
Redis
installed locally - Set ENV var
REDIS_URL=localhost:6379/0
- Run
bundle exec sidekiq
(add a-q queue_name
) to start that particular queue, you can see which queues are currently in the app inconfig/initializers/sidekiq.rb
underconfig.queues
- Note that the Openshift deployed versions make use of HA-Redis via Sentinels so the environment variables required for that are different
The app uses Anycable to serve websockets in a scalable way. To run the websocket server locally:
- Ensure to
bundle install
- Download the
anycable-go
websocket server on OSX:brew install anycable-go
- Add an ENV var
ANYCABLE_REDIS_URL=redis://localhost:6379/2
. It is recommended to set this value to a different Redis db than Sidekiq.
After installation run both the anycable-go
websocket server as well as the RPC server
- RPC Server:
bundle exec anycable
- Anycable-Go Sockets:
anycable-go --port=8080
- For local development, ensure you have the right test endpoint. The dev / test / production environments are set up with a licensed version running in its own container.
- For the local environment, if you want to test the full file upload process, you would want to run a local version of minio to simulate the object storage environment.
- Install minio:
brew install minio
minio server --address 127.0.0.1:9001 ~/Documents/whatever-your-folder-for-storage-is
- Log into minio with the admin/password provided and set up the environment.
- Set up a bucket (in our example we call it
hous-local
) - Set up a user (in our example, we call it
hous-formio-user
) - Set up a policy (in our example, we call it
formioupload
)
{
"Version": "2012-10-17",
"Statement": [
{
"Sid": "ListBucket",
"Effect": "Allow",
"Action": [
"s3:ListBucket"
],
"Resource": [
"arn:aws:s3:::hous-local"
]
},
{
"Sid": "UploadFile",
"Effect": "Allow",
"Action": [
"s3:GetObject",
"s3:ListMultipartUploadParts",
"s3:PutObject",
"s3:AbortMultipartUpload",
"s3:DeleteObject"
],
"Resource": [
"arn:aws:s3:::hous-local/*"
]
},
{
"Sid": "crossdomainAccess",
"Effect": "Allow",
"Action": [
"s3:GetObject"
],
"Resource": [
"arn:aws:s3:::hous-local/crossdomain.xml"
]
}
]
}
-
note that "s3:PutObjectAcl" is ignored at the moment since we are at the root bucket, but we may need it later
-
You will need to add the following to your .env files:
BCGOV_OBJECT_STORAGE_ENDPOINT=http://127.0.0.1:9001(or_whatever_you want)
BCGOV_OBJECT_STORAGE_BUCKET=your-local-bucket-name
BCGOV_OBJECT_STORAGE_ACCESS_KEY_ID=your-local-user-access-key
BCGOV_OBJECT_STORAGE_SECRET_ACCESS_KEY=your-local-user-secret-access
BCGOV_OBJECT_STORAGE_REGION=us-east-1
Our folder structure for our bucket is designed as follows:
- :bucket_name/permit-applications/:id/*
The overall process of a file upload is as follow:
- The front-end form allows you to upload files
- Upon selecting files, it calls the backend to get a presigned url for upload
- The files are directly uploaded into the /cache/ folder
- Upon saving the file, it persists gainst the application as a 'supporting_document'
See the /devops
folder for Dockerfiles and Helm-Charts
All unit tests should be run like the following:
bundle exec rspec
Some other notes
- For digital seal tests, the we have not uploaded documents with the real document to the repo, but captured the request responses
- If you are running in the dockerized version, you may need to reset your database or you may want to create a second instance of the dockerized postgre service to run tests.
- To generate up to date erd diagrams, make sure you have graphiviz installed locally
bundle exec erd
Frontend linting is performed by Prettier. The backend files are set up to use syntax_tree for formatting and Rubocop for linting. The only recommended setup required for this is to install the RubyLSP plugin for VSCode, and select syntax_tree as the formatter in the exension settings. Binaries for these should be installed as shims automatically upon running bundle install. All other configuration should be handled by the included workspace settings and .rubocop.yml file.
At the moment getting intellisense to work out of the box on Mac is difficult but can be accomplished with some configuration. Install the Solargraph extension and then in your User VSCode settings add:
"solargraph.commandPath": "/Users/YOUR-USERNAME/.rbenv/shims/solargraph", "solargraph.bundlerPath": "/Users/YOUR-USERNAME/.rbenv/shims/bundler", "solargraph.useBundler": true
ensure that you are using rbenv and it is set up correctly.