Skip to content

eddieantonio/oq

Repository files navigation

oq

The Project Obscure Questionnaire platform.

Architecture

The oq platform currently has the following parts:

  • webapp — a SvelteKit full-stack application that is what users see when they interact with the application. The webapp is also responsible for maintaining the database schema.
  • piston — a customized version of Piston — code execution engine.

In the future, I might add the following:

  • db — a relational database management system. As of 2023-11-14, I am using SQLite3, but if I need more concurrency, I plan to switch to PostgreSQL.

How user compilation works

Here is the flow of a users' code can be compiled and run:

%%{init: { 'sequence': {'mirrorActors': false} } }%%
sequenceDiagram
  actor User
  participant webapp
  participant piston

  User ->> webapp: press "Run code" button
  activate webapp
  webapp ->>+ piston: HTTP POST /api/v2/execute
  deactivate webapp
  piston ->> piston: Compile code
  piston ->> piston: Run code
  piston -->>- webapp: Run results
  
  activate webapp
  webapp -->> User: Output or error message
  deactivate webapp
Loading

System requirements

You will need:

  • Docker (I am using v4.25.0)
  • Node (I am using Node v20.4.0)

Piston

To get Piston setup, I had to do a bunch of stuff manually (automating the steps is quite annoying and not worth it right now). See piston/README.md for details, but make sure to have this setup before continuing.

Development

Important

See Installing new npm packages as things are not straightforward.

This is my development setup:

  • I setup all of my Node.js stuff on my host machine (for me, that's my Mac).
    • go into ./webapp and run npm install
  • I use Visual Studio Code with the Prettier extension to edit files.
  • I use Docker Compose to run the development server.

To setup the dev environment for the first time, you must do the following after clone:

ln -s docker-composer.{dev,override}.yml

Subsequently, you can bring up everything using the following command:

docker-compose up --build

Note: I do not daemonize docker-compose (i.e., provide the -d option) during development.

After this, you should have a live-reloading version of the site on http://localhost:3000/.

Installing new npm packages

The way that I have a live reloading dev environment within Docker compose is a bit of a hack. As a result, you can't just install an npm package with npm install --save example and expect it to work. Here's what you need to do:

tl;dr

cd webapp
npm install «PACKAGE» --save-dev
cd ..
docker compose down --volumes
docker volume prune --filter all=1
docker compose up --build

Install your package normally

Within the ./webapp directory, npm install your package. Remember to provide --save or --save-dev to commit the package to package.json.

How to choose between --save or --save-dev

If the module will be imported at runtime by the webapp server (anything used in $lib/server for instance), then it should be installed via --save:

For example,

npm install http-status-codes --save

Anything else should use --save-dev. With --save-dev, the package will only be required at build time, so it does not need to be in node_packages when the server is deployed, saving space in deployed image!

Bring docker compose down

Run the following command:

docker compose down --volumes

This will shut down the entire system and remove the currently used containers. Importantly, --volumes removes the anonymous volume used to store node_modules in the development container. If you don't remove this volume, then the container will act as if you never ran npm install at all!

(optional) Remove unused anonymous volumes

The anonymous volume stores the full dev version of node_modules — you probably want to get rid of it.

Heaviest objects in the universe (in increasing order of mass): sun, neutron star, black hole, node_modules

Docker does not delete the anonymous volume with the previous command, even though no container will ever use it anymore.

As of 2023-11-14, this is the command to remove unused, anonymous volumes:

docker volume prune --filter all=1

Rebuild the image and bring it all back up

Now you can bring everything back up, but you will need to rebuild the image.

docker compose up --build

This will:

  1. Rebuild all Docker images needed for the entire application — not just the webapp
    • Rebuilding the webapp image will also install all of node_modules
  2. Create a new anonymous volume for node_modules.
  3. Copy the brand new node_modules from the image into the new anonymous volume.

In practice, I always run docker compose up --build to bring everything up, even if I'm not messing around with npm packages.

Database migrations

See webapp/README.md for details on how to run the database migrations.

Deployment on beckett4

To deploy, you must run:

ln -s docker-composer.{prod,override}.yml

The following directory is bind-mounted to the webapp container:

/home/eddie/oq/run

What this means is that the database can be found in there!

TODO: Eddie needs to write more documentation here!

  1. how to build
  2. how to migrate

Creating the database

Make sure all containers are running, then use the following command:

docker compose exec webapp knex-wrapper migrate:latest

The very first time you deploy, you also need to insert initial data (seed data):

docker compose exec webapp knex-wrapper seed:run

It will ask you to create a password which will be the "participantion code" for the classroom.