Skip to content

Plea-Justice/researcher-console

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Plea Simulation Researcher Console

The researcher console is a web-based interface for configuring the plea bargain simulation. A project overview and detailed documentation are available at pleajustice.org.

Dependencies

A Unix platform is recommended. The live server and test system run Ubuntu Server 20.04.

Node may be installed with nvm. Additional Node based software will be automatically installled by npm.

The npm package, cross-zip requires the host system to support zip or Windows Powershell. Note that zip must be manually installed on some Unix systems such as Ubuntu Server.

Running a live server on HTTP port 80 or with HTTPS on port 443 may require reverse proxy, especially if other sites must also be served on the same host. The Nginx webserver supports reverse proxy. Let's Enycrypt's Certbot can be used to obtain SSL certificates. Note that a webserver may impose restrictions such as upload filesize limits that may interfere with the console. Check the software's manual for configuration options to lift such restrictions.

MongoDB is necessary to support the backend. After installing the database, start the Mongo service and set the URI of the database in server/config.js. By default, Mongo will run on port 27017. The database may be given any name. No further configuration is needed, however it is recommended that the database is set up with credentials and that its port is not accessible from the network.

Client

The console frontend is built with Nuxt.js, which provides additional features to Vue.js for single page applications. Vue components are from Buefy, based on the Bulma CSS framework. Axios handles XHR requests to the server.

Building the Frontend

# Switch to the client's directory.
$ cd client/

# Install dependencies.
$ npm install

# Configure client options.
$ vim nuxt.config.js

# Serve the client for development with hot reload at localhost:3001.
$ npm run dev

# Build for production and serve just the client.
$ npm run build
$ npm run start

# Build static project for production and output to client/dist/.
# This Nuxt command changes frequently. Check the most recent docs.
$ npm run generate

Most configuration options are in client/nuxt.config.js. Double check build options, that the Axios baseURL points to the backend, and that the auth middleware is enabled for authenticated routes.

Nuxt provides its own improved versions of many common packages as modules. These modules may have different options than the original packages (e.g. the Axios module). Nuxt also provides its own implementation of Vuex the Vue state management library, and configures other Vue features, such as routes automatically.

Server

The application's backend is built with Express. It interfaces with MongoDB using Mongoose. Filesystem interactions are performed using the fs-extra package, an extension of Node's built in fs module that adds async/await functionality and recursive copy among other functions. express-session handles user sessions, automatically issuing session cookies and tracking them in MongoDB.

Running the Backend

# Switch to the server's directory.
$ cd server/

# Install dependencies.
$ npm install

# Configure server options.
$ vim config.js

# Start the server for development with hot reload using nodemon.
$ npm install -g nodemon
$ nodemon

# Start the server for production.
$ npm run start

Configuration is stored in server/config.js. The client must be enabled in this file. Make sure to define separate databases for development and production and select the appropriate database in this file. Additionally, make sure that any file path options point to their respective and desired locations on the host system. In production, set session_secret to a new, random, and secure string.

Express creates bin/www, the server entry point. This file imports the application definition at app.js which requires middleware dependencies and sets up routes, mounting the client at root. Additional routes are defined in routes/ including api_v1, where the API is defined.

Note that there is not currently a systemd or other service in place to autorestart the server. After reboot it must be started manually.

For backend testing, curl, a browser extension, or graphical app such as Postman are recommended.

Running Side-By-Side

It is sometimes helpful to run both the client and the server separately during development, especially when hot-reload on the client is desired. In this situation, the frontend and backend must be run on separate ports. To allow for this, several features must be enabled.

In the client's nuxt.config.js, the Axios module baseURL must point to the server and credentials must be enabled so that cookies are sent cross-site. The port should be different than that of the server.

server: {
    // Note that 'server' here refers to Nuxt serving the frontend.
    port: 3001,
    host: 'localhost'
}
axios: {
    // If the backend server is on port 3000.
    baseURL: 'http://localhost:3000',
    credentials: true
}

Similarly, in the server's config.js, the port must be different from that of the client. Cross-origin requests must be enabled, and secure cookies disabled so that the client will know it is safe to share cookies issued by the server between the server and the client.

{
    port: 3000,
    host: 'localhost',

    cors_enabled: true,
    cors_origin: 'http://localhost:3001',

    secure_cookies: false,

    ...
}

The server and client can then be run separately with:

$ cd client
$ npm run dev

and in a separate terminal...

$ cd server
$ nodemon

Ensure that when it comes to production, credentials, cors_enabled and secure_cookies are false, false, and true respectively so as not to introduce security issues.

Managing the Database

The database is handled completely by Mongoose, which provides an interface to MongoDB that includes models, schemas, and validation. Refer to the Mongoose documentation when adding functionality to the backend code.

It is not recommended to manually modify the database, however it is occasionally necessary for development purposes. This is dangerous as there is no confirmation before you delete an entire database.

Mongo organizes data into collections of documents (JavaScript objects), assigning each object an ObjectID.

The server will create three collections in whatever database has been defined in config.js. If the database does not exist, it will be automatically created on first reference.

At the Mongo shell, use will switch the context to the desired database. From there, the collections can be listed.

$ mongo

> use researcher
> show collections

The scenarios collection is where all users' data is stored. users and sessions hold authentication information and express-session information respectively.

The contents of any collection may be listed or searched with find.

// List all documents in a collection.
> db.sessions.find()

// Find documents matching a Query object.
> db.scenarios.find({ _id: ObjectID("5f21d0b62c6778ec05fcc108") })
> db.users.find({ username: TestUser })

Mongo uses JavaScript as a scripting language and so common development queries can be easily scripted and run with mongo script.js.

Managing the Filesystem

The server will store and retrieve data from the directories specified in config.js. The server requires a directory containing the default copy of the simulation code as well as a directory containing default assets for new users. New directories may be created to host simulations and store uploaded assets and other user data. Temporary files may also be created in the host's /tmp filesystem under "sim-prev/" to generate and serve simulation previews and zip files.

Ensure that a path exists for the server's data and that the server has write permissions. Any files in this path may be overwritten by the server. See the example structure in server/data/.

Server API (v1)

API endpoints will always respond with a JSON object of the following format.

{
    success: Boolean,
    message: String,
    result: Object
}

The API defines endpoints for managing scenarios, assets, and user authentication.

Method Endpoint Function
GET /api/v1/scenarios Get the list of scenarios.
POST /api/v1/scenarios Create a new scenario.
GET /api/v1/scenarios/{scenario_id} Get a scenario.
PUT /api/v1/scenarios/{scenario_id} Save a scenario.
POST /api/v1/scenarios/{scenario_id} Copy a scenario.
DELETE /api/v1/scenarios/{scenario_id} Delete a scenario.
POST /api/v1/scenarios/{scenario_id}/generate Generate a simulation.
POST /api/v1/scenarios/{scenario_id}/zip Prepare a ZIP of a generated simulation.
GET /api/v1/assets Get the list of assets.
POST /api/v1/assets Upload an asset.
POST /api/v1/assets/{asset_id} Copy an asset.
DELETE /api/v1/assets/{asset_id} Delete an asset.
GET /api/v1/assets/{asset_id}/thumbnail Get an asset's image thumbnail.
POST /api/v1/auth/login Begin a logged in session.
POST /api/v1/auth/logout Destroy a logged in session.
POST /api/v1/auth/register Register new credentials.
GET /api/v1/auth/user Get the user of the current session.
GET /api/v1/admin/users Get the list of user information.
PUT /api/v1/admin/users/{user_id}/permissions Set the permissions of a user.
PUT /api/v1/admin/users/{user_id}/password Set the permissions of a user.
PUT /api/v1/admin/users/{user_id}/attributes Set other properties of a user.
DELETE /api/v1/admin/users/{user_id} Delete a user.

Copyright (C) 2021, The Plea Justice Project. Licensed under GPLv3.

About

Configuration interface for researchers interested in creating psychological studies with https://github.com/Plea-Justice/pleabargain-simulation.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Contributors 3

  •  
  •  
  •