Electronic voting system for the annual general meeting (AGM), or "generalforsamlingen/genfors" of Linjeforeningen Online
Check out the todo list.
- Node.JS(8.9 or higher)
- npm
- Start by cloning the repository, then running
npm install
(oryarn
) from the root directory. - Run backend with
npm run backend
(oryarn backend
) - Run frontend with
npm run frontend
(oryarn frontend
)
This is an example of a docker command to run super-duper-fiesta in production.
Build image: docker build -f Dockerfile.prod -t sdf_prod .
docker run \
-p 3000:3000 \
-e DATABASE_URL \
-e SDF_OIDC_PROVIDER \
-e SDF_OIDC_CLIENT_ID \
-e SDF_BACKEND_HOST=0.0.0.0 \
--name sdf_prod \
--rm \
sdf_prod
npm install
npm run build:prod
- Make sure the relevant environment variables are set
NODE_ENV=production
PRODUCTION=true
- Any other required variables, see the configuration section
npm start
(ornode server/app.js
)
- All code is to be linted according to the specification listed in
.eslintrc
. You can verify this by runningyarn lint
.
- The project uses React.JS for the frontend code, with redux mixed in for state management.
- The backend is an ExpressJS web server which exposes WebSockets in addition to a simple API. WebSockets is the primary source of communication in this project.
The backend and frontend are connected by socket.io, following the ideology of redux-socket.io, namely that any action to be dispatched to the backend starts is on the form server/<action>
, where action
is a unique name for the action to be performed. The backend will emit events on the action
channel, and every event will contain a type
which is parsed by the client side of the library, and passed through to the correct reducer in the frontend.
If the frontend dispatches an action server/hello
, redux-socket.io will automatically dispatch this event to the backend. It's now up to the backend to handle this event.
Check out this example. Basically much like any other action as defined by redux.
If the backend dispatches an event on the channel action
, it is required to have a type
for the frontend to be able to parse it as an action. Import and use the helper function emit
from utils.js
to automate some of the repetetive tasks as demonstrated below.
// Emit an action to be handled by the frontend
emit(socket, 'OPEN_ISSUE', { description: "Issue number 1" });
This will be emitted on the action
channel and be passed on to the reducers in the front end.
Most of the configuration of this app is done using environment variables.
(Tip: Use a tool like "autoenv" to automatically set environment variables upon entering a project directory.)
Linux, MacOS:
Current terminal:
export VARIABLE_NAME=VARIABLE_VALUE
To persist the changes, look into a tool like "autoenv" as mentioned earlier, or set them in ~/.profile
(not recommended...).
Windows:
Current terminal:
set VARIABLE_NAME=VARIABLE_VALUE
Current user (permanent):
setx VARIABLE_NAME VARIABLE_VALUE
These variables are required in development. The others are safe to skip.
DATABASE_URL
(example:sqlite:///path/to/working/directory/db.db
(note the three/
-es))SDF_OIDC_PROVIDER
(example:https://online.ntnu.no/openid/
)SDF_OIDC_CLIENT_ID
(example:123456
) You can get this ID from the adminstration panel in Django for OW4.SDF_OIDC_REDIRECT_URI
(example:http://127.0.0.1:8080/auth
) Make sure this value is included in the allowed redirect uris in the OpenID Client.SDF_BACKEND_HOST
(default:'backend'
) should be set (example:127.0.0.1
) if not running in docker.SDF_BACKEND_PORT
(default:3000
) should be set if running OW4 with webpack locally.
Key | Description | Example | Default |
---|---|---|---|
PRODUCTION |
Run the app in production mode | true |
'' |
SDF_HOST |
Host to run the webpack dev server on | 127.0.0.1 |
127.0.0.1 |
SDF_PORT |
Port to run the webpack dev server on | 8080 |
8080 |
Key | Description | Example | Default |
---|---|---|---|
SDF_BACKEND_HOST |
Host to run the app on | 127.0.0.1 |
127.0.0.1 |
SDF_BACKEND_PORT |
Port to run the app on | 3000 |
3000 |
SDF_SCHEME |
HTTP Scheme to run the app on | http |
http |
The database connection can be configured using DATABASE_URL, which is a connection URI exposed to ENV on the format:
protocol://[user:password@]address[:port]/database_name
Example PostgreSQL connection URI:
postgresql://sdf:secretpassword@127.0.0.1:5432/sdf
Or SQLite: sqlite:///path/to/database.db
(note the three /
, the first two are part of the protocol separator and the final one denotes the root of your file system)
For further configuration of Sequelize, check out .sequelizerc
. This contains information about paths to model, migration and seed folders.
The project is bootstrapped and expected to use migrations for model changes. Migrations can be applied by running
yarn db:migrations
(which executessequelize db:migrate
)
When running migrations, you have to specify --url $DATABASE_URL
for Sequelize-cli to understand that you want to connect using the DATABASE_URL configuration.
The (fairly lacking) documentation for Sequelize migrations and how to create them can be found here or by executing sequelize --help
.
Authentication can be done through OpenID Connect.
This requires an OpenID Client ID as well as an OpenID Provider capable of providing the "onlineweb4" claim. source
Key | Description | Example | Default |
---|---|---|---|
SDF_OIDC_PROVIDER |
OpenID Connect Provider (Issuer) | http://127.0.0.1:8000/openid |
`` |
SDF_OIDC_CLIENT_ID |
ClientID of an OIDC client on OIDC provider | 123456 |
`` |
SDF_OIDC_REDIRECT_URI |
Redirect URI back to SDF | http://127.0.0.1:8080/auth |
`` |
If running the OpenID Connect provider in onlineweb4 locally, remember that webpack uses port 3000 by default, so you'll likely have to use another port for super-duper-fiesta.