The Generated Optional Online Networking (GOON) card is a digital business card with emphasis on optional - now you can choose exactly which social media accounts you wish to share with the world! Dynamically select as many or as few of your linked accounts as you wish, and create a customized business card on the fly.
Created at TCNJ in the Fall 2022 semester by The Goon Squad:
- Payton shaltis
- Leah Kazenmayer
- Nisha Reddy
- Sterly Deracy
- Shane Matyi
- Project Structure
- Running the Application Locally
- Running the Application From the Cluster
- Development Process
- Usability and Features
- Routes
The GOON card is a web application at heart, and its file and directory structure attempts to reflect that. The general flow of file interaction in our project involves the following:
- The user accesses one of many routes that have been defined in our server.
- The server processes the user's request based on the current route and any information that has been sent by the user.
- The server responds with a view that is rendered by the client, including HTML, CSS, and JavaScript files.
The root directory contains the following files:
.gitignore
: Files to be ignored by git and GitHub..prettierrc
: Configuration file for the Prettier extension.LICENSE
: The license for the project.package.json
: The file that contains the project's metadata, as well as the list of dependencies. This file is used bynpm
, our dependency manager for the project, in order to install the dependencies. Read more about dependencies here.package-lock.json
: Created after installing all dependencies locally. Used only to specify the exact version of each dependency that was installed.README.md
: The file you are currently reading.server.js
: The most important file in the entire project. This is the entry point for the application, and is responsible for starting the server and listening for requests. It also contains the code for setting up the database connection, as well as the code for setting up the routes for the application. Because routes are defined in their own modules, this file simply imports the routes and sets them up. For this reason, our server file is very modular and easy to maintain at less than 60 lines of code.
The root directory contains the following subdirectories:
db
: Contains the SQLite database file, the original schema SQL file used for generating the database, as well as a number of JavaScript modules that define the interfaces for accessing data from each database table. Read more about the database and its schema here.docs
: Contains development documentation for the project, including the design diagrams for multiple iterations of the database schema.node_modules
: The application usesnpm
in order to manage dependencies. Cloning the project from GitHub will not create this directory by default, but it can be created by runningnpm install
in the root directory of the project. See more about dependencies here.public
: Contains the static assets and commonly shared files that make up the application, such as CSS styles and background images.routes
: Contains the JavaScript modules that define the routes for the application. Each set of routes has been logically divided into their own module according to type. Read more about routes here.util
: This folder contains any of the utility functions that are used throughout the application. These functions are not specific to any one module, and are instead used by multiple modules in the application. The main utility in this folder is the middleware used by routes to ensure that a user is logged in.views
: Contains the HTML Embedded Java Script (EJS), CSS, and JavaScript files for each page. Each page's set of files is logically separated into its own subdirectory based on the page's purpose. These are the files that get served when a user accesses a valid route. EJS was used over HTML in many cases since it can be templated to dynamically generate HTML based on the data passed to it.
There are technologies that we use that do not necessarily need to be installed by the user. The technologies that we are currently using in our project are:
- HTML & CSS
- Bootstrap
- SQLite
- Node.js & Express.js
- Google Charts API
- Web Speech API
Our project uses Node Package Manager (npm
) for managing and installing dependencies. The list of dependencies can be found in the package.json
file in the root directory of the project. The dependencies are listed in the dependencies
section of the file. The devDependencies
section contains the dependencies that are only used during development, such as the nodemon
package that automatically restarts the server when changes are made to the code or the prettier
package that formats the code to a consistent style.
Some of the more critical non-development dependencies are as follows:
express
andexpress-session
: These packages are used as the main NodeJS server framework and session management library, respectively.sqlite3
andknex
: These packages are used to connect to the SQLite database and execute SQL queries.bcrypt
: Used for hashing and salting passwords. All passwords are stored hashed in the database.body-parser
andejs
: These packages are used to parse the request body and render the EJS files, respectively.uuid
,address
: Utilities used in various places throughout the application to simplify IP address resolution and UUID generation.
In order to install dependencies, make sure that you have node
and npm
installed. Navigate to the root of the project and run the following command to install all of the dependencies listed in the package.json
file:
npm install
This will create the node_modules
subdirectory with all required dependencies installed locally. The package-lock.json
file will also be created, which specifies the exact version of each dependency that was installed.
The versions required for each dependency are specified in the package.json
file. This ensures that the same versions of each dependency are used by all developers working on the project. npm
does not always specify the exact version used (for example, ^5.1.0
indicates any version at or above 5.1.0
), so exact versions will vary based on when npm i
or npm install
is run to install the dependencies.
The project uses an SQLite database, which is a file-based database. The database file is not included in the GitHub repository, so it will need to be created after cloning the repo. A description of the schema and instructions on setting up the database are included below.
We are using an SQLite database in order to store users' preferred settings as well as their account information. Below is a diagram of the database's current schema.
- Because each user will have the same set of options, it makes more sense to store this as a JSON object within the
users
table and parse it later. - Because each user will likely have a different number of social media accounts, it makes more sense to store these in a seperate table and link the two with a foreign key.
- The
card_entries
table will be used to keep track of which social media accounts are linked to which card. This will allow us to quickly and efficiently generate new Goon Cards without needing to create a new file for each card.
In order to run the app locally, the database also needs to exist in the project directory. In order to prevent the database from being committed to the repo, it is ignored by git. To create the database, run npm run seed
in the main project directory. This will:
- Delete the database file if it already exists.
- Create a new database file.
- Create the tables.
- Seed the database with some sample data from
./seed.js
.
When testing the application locally, any data that you add will be deleted and the database reseeded with another run of npm run seed
. This configuration is just for testing the application while we develop; once we start testing with other users, we can change the configuration to persist data.
After understanding the structure of the application and how to install dependencies and set up the database, the server can be run to begin listening for requests. Follow the instructions below in order to begin running the application:
- Make sure that you have installed all dependencies as described in the dependencies section.
- (Required on first run) Make sure that you have configured the database as described in the database section.
- Run
npm start
to start the server locally on port 3000. Make sure that no other processes are running on port 3000 or this command will exit with an error. - You can access the server at
http://localhost:3000
, or on other computers on the same network if you know your local IP address.
As an alternative, you can run npm run cleanStart
to perform all of the above operations, including deleting old versions of existing dependencies, re-installing dependencies, recreating the database, and starting the server.
If the application is running on the TCNJ cluster, and you are connected to the eduroam
network, you should be able to access the application online using the link below. Note that any accounts that you made while running the application locally will not be available on the cluster, and vice versa:
http://gooncard.hpc.tcnj.edu:3000
If you get an error while trying to access the application on the cluster, there is a chance that the server may have crashed due to some HPC issue. If you think this has occurred, please contact one of us and we can restart the server!
GitHub allows us to develop in a mostly individual manner, but still allows us to collaborate on the same codebase. While there were plenty of times that we worked together on the same code at the same time, the issue of keeping things consistent naturally came up rather early in the process.
Throughout development of the project, a number of utilities were taken advantage of in order to streamline the contribution process. Below includes some of the main tools used for development:
The .prettierrc
file stores the formatting rules for the project. It is used by the prettier
package to format the code. You can format all code in the project by running npm run format
after writing code in order to adjust whitespace, indentation, and blank lines between code. This should hopefully avoid commits that include mostly formatting changes.
Currently, Prettier doesn't support .ejs
, but you can get around this by using the VSCode extension and setting your default formatter in VSCode to Prettier and adjust the .ejs
association to HTML
like in this link. The extension allows you to use the VSCode option to auto-format on save, which is definitely recommended.
There are 6 development scripts used in this project. These are used to run the server, format the code, and seed the database. The scripts are as follows:
The following scripts should only be run on your local machine:
npm start
- Starts up the server locally so that you can use the application.npm run cleanStart
- Runs a combination of the format, seed, and start commands in order.
The following scripts should only be run on the HPC cluster:
npm run start-vm
- Runs the server on the TCNJ VM with the appropriate gooncard.hpc.tcnj.edu web address.npm run cleanStart-vm
- Runs a combination of the format, seed, and start commands in order, but again, specifically for the VM.
The following scripts can be run on either your local machine or the HPC cluster:
npm run format
- Formats the application code using Prettier formatting as described in the Prettier formatting section.npm run seed
- Deletes the existing database file (if there is one), creates a new database file, creates the tables, and seeds the database with sample data for two GOON card users.
When you make a change (at least on the front-end side), you should just be able to refresh the page. Changes to routes or backend files require a restart of the server.
Before you push the code to the repo to make a Pull Request, make sure to run npm run format
to format the code so that it matches the repo's white space and indentation.
If you get a message about files being modified and changing from LF to CLF for example, do git add -A
to fix the issue.
- A user can register an account on this application.
- A user can input, edit, and delete their social media information.
- A user can select which social media information to share and generate a QR code associated with their selections to share with others.
Our goal as a team was to reach as broad of an audience as we can with our application. An application that is intended to promote networking should be easy to use and manage by everyone. That is why we put accessibility at the forefront of everything that we implemented. We followed numerous guidelines and articles (linked above) that discussed how developers can make an application more accessible or commonly known as, "a11y" friendly. Below are a number of accessible features that the GOON Card application currently includes:
- Settings Page allows users to customize their display. For example, a user can modify their name, font-size, font-family, contrast, and theme. This functionality was implemented to let users with visual or other limitations to have a more user-friendly accessible application.
- The web application is responsive on both mobile versions and desktop versions for users with a wide variety of devices.
- There is a speech recognition feature where users can dictate a command into the device's microphone after selecting the microphone icon and have the action be carried out. This feature is to help users with physical impairments or restrictions.
- Visual displays such as icons or codes or potentially unrecognizable entities will have alt tags for users with screen readers to be able to identify.
- There is a help icon on pages that require user interactivity. This highly accessible feature provides a user with guidance on how to interact with the application.
- Check out this link for accessiblity guidelines to consider.
- Check out this link for color schemes that follow accessibility guidelines.
- Check out this link for an overview of important accessibility guidelines to follow.
- Check out this link and this link for information about accessible fonts.
As mentioned in the file and directory structure section, routes are the lifeblood of the application's backend. Each route defines a certain set of actions that the user can take, as well as the appropriate page view files that will be sent back as a response. The section linked above includes the location for all routes if closer inspection is required, but a more general outline of the routes is included below:
GET /
- Loads the home page if logged in, otherwise redirects to the login page. The home page is passed the user object for displaying information about the user.GET /home/style
- Serves the home page's stylesheet.GET /home/main
- Serves the home page's JavaScript.
GET /account/login
- Loads the login page, passing it an error that can be optionally displayed.GET /account/login/style
- Serves the login page's stylesheet.GET /account/login/main
- Serves the login page's JavaScript.POST /account/login
- Attempts to log the user in. If successful, redirects to the home page. If unsuccessful, redirects to the login page with an error message. Used in form submission.
GET /account/logout
- Logs the user out and redirects to the login page. This is implemented as a GET method in order to use it in an<a>
tag.
GET /account/signup
- Loads the signup page, passing it an error that can be optinally displayed.GET /account/signup/style
- Serves the signup page's stylesheet.GET /account/signup/main
- Serves the signup page's JavaScript.POST /account/signup
- Attempts to create a new user. If successful, redirects to the home page. If unsuccessful, redirects to the signup page with an error message. Used in form submission.
GET /account/profile
- Loads the profile page, passing it the user object for displaying information about the user.GET /account/profile/style
- Serves the profile page's stylesheet.GET /account/profile/main
- Serves the profile page's JavaScript.GET /account/profile/getsettings
- Returns the users selected settingsGET /account/profile/getall
- Returns all the user_accounts associated with the user or null if the user has no accounts.POST /account/profile/update
- Updates the user's settings. Used in form submission.
GET /account/settings
- Loads the settings page, passing it the user object for displaying information about the user.GET /account/settings/style
- Serves the settings page's stylesheet.GET /account/settings/main
- Serves the settings page's JavaScript.GET /account/settings/update
- Updates the user's settings. Used in form submission.GET /account/settings/delete
- Deletes the user's Goon Card account, all of their linked accounts, and all cards associated with their account that were previously generated.
GET /aboutus
- Loads the about us page if logged in, otherwise redirects to the login page.GET /aboutus/style
- Serves the about us page's stylesheet.GET /aboutus/main
- Serves the about us page's JavaScript.
GET /displaycard
- Loads the page to display a single GOON card. Requires the query parameterid
to be set to the ID of the card to display. If the card does not exist, the user is redirected to the 404 page.GET /displaycard/style
- Serves the display card page's stylesheet.GET /displaycard/main
- Serves the display card page's JavaScript.
GET /notfound
- Loads the 404 page. This page should only be directed to if the user tries accessing a GOON card that does not exist.GET /notfound/style
- Serves the 404 page's stylesheet.GET /notfound/main
- Serves the 404 page's JavaScript.