The application consists of three subprojects: backend, frontend and admin-frontend.
- Admin-frontend is used by the youth workers to handle registered members and their info. The QR code reading is also part of this.
- Frontend is the end user web application used by the youth to see information about the youth clubs and generate a personal QR code for checking in.
- Backend has endpoints for both frontends and for Suomi.fi identification. It uses PostgreSQL for database.
More detailed documentation is found in a README in respective directories of each project.
Mobiilinutarkotti by default uses local user management via database. With environment variables, a single-tenant Microsoft Entra ID login is possible. Mobiilinutakortti uses SMS service by Telia, and email service by Amazon.
- NodeJS - v22 preferred
- PostgreSQL - v16 preferred
- Docker (optional)
Only Docker is needed to run the app. If you want to run backend locally, you'll need NodeJS and PostgreSQL installed. More info in ./backend/README.md.
For production use, Telia SMS service is required. Note that there are two endpoints (see environment variables section in backend documentation). The use of the batch endpoint requires permit separately from Telia. Ensure your credentials have the permit.
- Frontend : React (running on port 3001)
- Backend : NestJS (running on port 3000)
- Admin-Frontend : React Admin (running on port 3002)
- Database : PostgreSQL (running on port 5432)
Each subproject may be run individually, with or without docker - see README.md files of the projects.
To start up everything using Docker compose, run docker compose -f docker-compose.yml.local up
in this directory. Since there might be problems with Docker caching, the run-locally.sh
helper script is a good alternative.
To make sure everything is working, navigate to:
- http://localhost:3001 - frontend
- http://localhost:3002 - admin-frontend (default port will be 3000 if running without Docker)
- http://localhost:3000/api - backend
If you see the webpage for frontend and admin-frontend, and "API is running" message for backend, you're good.
NOTE:
-
If you have PostgreSQL running locally, it is probably using port 5432 and will conflict with the Docker setup. Bring the local instance down or reconfigure it to solve the issue.
-
At some point during npm install you might get a weird error like this:
npm ERR! code EAI_AGAIN npm ERR! errno EAI_AGAIN npm ERR! request to https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz failed, reason: getaddrinfo EAI_AGAIN registry.npmjs.org registry.npmjs.org:443
This is because Docker might have some problems using IPv6 DNS servers. Force the use of IPv4 DNS in your localhost.
NB: this section only applies if you are not using Microsoft Entra ID to login users. By default, Entra ID is not used. It can be enabled via environment variables.
The application needs at least one youth worker user to work properly. The backend must be running when executing this step. The endpoint that we call is only open if the environment variable SETUP_ENDPOINTS
equals "yes", so set it when launching the backend. You can do this temporarily for example by editing the docker-compose.yml.local
file.
Run the following curl
command to create a youth worker with admin rights:
curl --location --request POST 'http://localhost:3000/api/youthworker/registerAdmin' \
--header 'Content-Type: application/json' \
--data-raw '{
"email": "[email protected]",
"password": "test",
"firstName": "admin",
"lastName": "admin",
"isAdmin": "true"
}'
Alternatively, you can use GUI tools such as Postman or Insomnia to create a youth worker user.
POST to http://localhost:3000/api/youthworker/registerAdmin with following body:
{
"email":"[email protected]",
"password": "test",
"firstName": "admin",
"lastName": "admin",
"isAdmin": "true"
}
Now you can login to admin-frontend with given credentials.
When deploying application to production, endpoint should initially be open, and after creation of youth worker, it should be closed ASAP. The endpoint is toggled by environment variable SETUP_ENDPOINTS
. Set its value to "yes" to allow registering youth workers via the endpoint and unset the variable (or set as "no") to disable the endpoint afterwards.
Currently, there's no user interface for creating youth clubs. You can insert them directly to the database to the clubs
table. For example, the list of youth clubs for Vantaa would be something like:
insert into public.club (name) values
('Hakunilan nuorisotila'),
('Havukosken nuorisotila'),
('Hiekkaharjun nuorisotila'),
('Kivistön nuorisotila'),
('Kolohongan nuorisotila'),
('Korson nuorisotila'),
('Länsimäen nuorisotila'),
('Martinlaakson nuorisotila'),
('Mikkolan nuorisotila'),
('Myyrmäen nuorisotila'),
('Pakkalan nuorisotila'),
('Pähkinärinteen nuorisotila'),
('Tikkurilan nuorisotila');
commit;
To test SMS functionality locally, rename .env.template
file to .env
in /backend and update the Telia username/password/user fields with right values (check in Microsoft Teams - Vantaan Kaupunki Wiki page to see whom to contact to get the values)
With the SETUP_ENDPOINTS
enabled and the backend running, use these two to create and remove test youth data:
- Create 100 test cases:
curl --location --request POST 'http://localhost:3000/api/junior/createTestDataJuniors' --header 'Content-Type: application/json' --data-raw '{ "numberOfCases": "100" }'
- Delete all created test cases:
curl --location --request POST 'http://localhost:3000/api/junior/deleteTestDataJuniors' --header 'Content-Type: application/json'
QR-code check-in endpoint is open by default, and should be accessible without authentication. This is due the removal of session-token when entering to QR-code screen, to prevent end-user to navigate to other parts of the application.
In the admin-frontend there is a possibility to enable extra entry registry via an environment variable (see its README.md). The extra entries are hidden in the UI by default. The extra entries enable permissions and markings to entries of type <what> <expiry age>
. For example, if a junior has a permission to participate into a gym intro course, there could be an extra entry of type <Gym course> <21>
, and the junior would have a permission for it. After completing the intro course, he could be given a permanent marking for the gym course. The marking would expire when the junior turns 21 years.
Docker volumes sometimes get messed up and database won't work, often indicated by login not working. This might be indicated by error message such as:
Failed Password Authentication for user 'postgres'
Bring down the Docker containers with: docker compose down
To nuke the database, remove Docker volume from the PostgreSQL container, and bring the application up again.
When running "docker compose up" you might get an error like this:
admin-frontend_1 | events.js:174
admin-frontend_1 | throw er; // Unhandled 'error' event
admin-frontend_1 | ^
admin-frontend_1 |
admin-frontend_1 | Error: ENOSPC: System limit for number of file watchers reached, watch '/admin-frontend/public'
or your build may error randomly.
There's a lot of files under node_modules
and they are all being watched, reaching the system limit. Each file watcher takes up some kernel memory, and therefore they are limited to some reasonable number by default. On a Ubuntu Linux the limit can be increased for example like this:
echo fs.inotify.max_user_watches=524288 | sudo tee -a /etc/sysctl.conf && sudo sysctl -p
Increasing memory limits for Docker might also help if for example you are using the Docker Desktop app to constrain them in the first place.
Application runs in Elastic Beanstalk in a single container (using Dockerfile via docker-compose.yml) and is deployed via command-line manually. The name is nutakortti-vantaa-dev.
Application runs in Elastic Beanstalk in a single container (using Dockerfile via docker-compose.yml) and is deployed via command-line manually. The name is nutakortti-vantaa-prod. See next section for updating the production environment using EB CLI tools.
Production logs are found in AWS CloudWatch under /aws/elasticbeanstalk/nutakortti-vantaa-prod/var/log/
(just go to CloudWatch and select Log groups from the left panel). The current app log and nginx access/error logs are of most interest.
Install the tools (for quick setup, follow the README in GitHub):
- AWS docs
- GitHub
- Remember to add EB CLI to PATH (e.g.
export PATH="/home/username/.ebcli-virtual-env/executables:$PATH"
).
Configure the EB CLI:
- AWS docs
- Note: this process only initializes the current directory/repository on your computer. The relevant files have been added to gitignore.
- Go to project directory root (where this file is). Type:
eb init
. - Select
eu-central-1
as the location (unless something's been changed). - If you haven't set up your AWS credentials yet, provide your personal Access key ID and Secret access key. You got them when receiving the AWS credentials (you should have got the following: User name,Password,Access key ID,Secret access key,Console login link). On Linux/OS X, the credentials will be stored in
~/.aws/config
. - Select the
Nutakortti
as application. Don't continue with CodeCommit (defaults to N). - Ensure the environment is set up by typing
eb list
. You should see nutakortti-vantaa-prod.
Deploy a new version to production:
- While in the project root directory, type:
eb deploy nutakortti-vantaa-prod
- To see how things are progressing, type:
eb events -f
Nutakortti has been configured to use AWS CloudWatch for log streaming.
Since the AWS browser GUI doesn't offer the means to directly download full logs, CLI tools might be needed when browsing logs.
The EB CLI tool has a command to download all logs from a specific CloudWatch log group, but it does not work: it only downloads what would be visible in AWS management console at that moment. For older events, the tool will create 0-byte files.
The AWS CLI tool can be used to get CloudWatch log events, but that would require the use of the --next-token
parameter in a script to actually download all the logs.
There exists a nice tool to solve the problem. So, in case one needs all the logs for whatever reason, use the AWS CLI directly by scripting, or use for example the awslogs for an easy solution. To use the tool:
- Install awslogs using pip:
pip install awslogs
- Set up AWS CLI (command:
aws configure
) - Set up default region for awslogs (environment variable
AWS_REGION
) or give it as a command line parameter. - Get the logs from a specific time window, e.g.
awslogs get /aws/elasticbeanstalk/nutakortti-vantaa-prod/var/log/eb-docker/containers/eb-current-app/stdouterr.log --start='52 weeks' > logs_past_year.txt
Normally there's only the certificates to update. TLS certificates are updated to AWS (or wherever the service is running).
For Suomi.fi certificate updates, see the file ./backend/certs/README.md
.