A problem that we, the members of Team Pelican, constantly face is that of not being able to decide what to watch online. With so many video streaming services at our disposal, it is easy to forget that one service might offer something much more appealing to you given your current mood. QueueUp aims to grab all of your streaming video content and assemble it into an easy to use streaming media dashboard to help you decide what to watch.
Clone the QueueUp repository to a location on your machine by running the following command in terminal:
git clone https://github.com/TeamPelican/QueueUp.git
QueueUp is a node application, so you'll need the command line tools to install necessary files. Install Node here.
Once you have Node installed, navigate to the QueueUp project directory and run the following to install necessary Node packages:
npm install
This might take a few seconds to finish installing QueueUp's dependencies based on how far along we are in the development process.
QueueUp also makes use of a Node package called Bower in order to easily install front end resources to a public directory.
First, you need to install Bower:
npm install -g bower
Note that you may need to prefix the above command with sudo
in order to install Bower at the global level.
With Bower now installed, you need to install the front-end dependencies through it's own package managing syntax, similar to npm above:
bower install
At this point, you should now have all dependencies required to start QueueUp on your machine.
QueueUp implements data persistence through use of MongoDB. In order to successfully use QueueUp, you must first install MongoDB. Once you have MongoDB installed, make sure that both the mongo
and mongod
commands work in your terminal.
QueueUp also requires a password.json file to be used for encrypting data in your database. The password specifed in this file can be anything you like. Just be sure that you do not change it. Below is an example password.json file:
{
"password" : "thisisonlyanexample"
}
Here you will need to place your api tokens. We have provided a specific api.json for our instructors in our submission, but for developers who would like to extend QueueUp, a new api.json will need to be created. The api.json file will be formatted as follows:
{
"web": {
"client_id": "<your_oauth_token>",
"auth_uri": "https://accounts.google.com/o/oauth2/auth",
"token_uri": "https://accounts.google.com/o/oauth2/token",
"auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs",
"client_secret": "<your_client_secret>",
"redirect_uris": [
"http://localhost:3000/user/callback",
"http://localhost:3000/oauth2/oauth2callback",
"http://localhost:3000/oauth2/youtube"
]
}
}
In order to create <your_oauth_token> and <your_client_secret> from the JSON above, you will need to manually create credentials at the Google Developers Console.
First, you should open a new terminal and run the mongod
command in order to start the local MongoDB server for QueueUp to connect to. This should start a MongoDB server on port 27017.
Once you have all dependencies correctly installed, you can run QueueUp by navigating to the root level of the project directory and running
node app.js
This will start the QueueUp web application on port 3000 if you do not have an environment variable specified.
Note: It is important that you view the dependencies of the following libraries should you have any trouble setting up and/or running QueueUp after cloning this repository.
Body-parser is currently used to create middleware that parses either URL-encoded or JSON bodies of HTTP requests.
Bower is QueueUp's package manager for front-end resources. We use it to quickly and easily import front-end libraries like jQuery, Bootstrap, and sweetalert.
QueueUp uses connect-flash to store messages to be used by the session. Specifically, we store a message using connect-flash in one context of QueueUp, and then redirect to another context which will access that message.
QueueUp saves session information to our database using connect-mongo.
Express wins the contest for Team Pelican's favorite npm package. This package allows us to easily manage HTTP requests to the server, and is widely supported so we could easily find answers online whenever we ran into something we did not know how to tackle.
Express-handlebars allows Express.js to use Handlebars as the view engine for QueueUp. That is, instead of using one of Express' default view engines, we get to use Handlebars for templating.
This package allows QueueUp to keep track of a user session when the user logs in. How we manage this session determines which state a user will navigate to as they use QueueUp.
This package allows us to use Google's OAuth2 protocol to connect to their servers, as well as interface and query YouTube.
QueueUp uses the mongodb package to connect to our MongoDB instance through JavaScript. With this package, we are able to perform insertions, updates, and removals of data in our database.
Morgan is a middleware that allows us to easily change how much logging to the server's console we receive upon each HTTP request made to the server. We currently use its 'tiny' amount of logging, but we could use many other options as seen at the link to the GitHub above.
The request package allows the server to send GET/POST requests directly without redirecting, which we needed to send deauthorization requests to revoke access tokens.
QueueUp uses Bootstrap as a foundation for CSS styling. Bootstrap allows us to easily get a nice UI with minimal additional CSS.
JQuery allows us to easily add functionality to our webpages and make special HTTP requests to the server.
Slick provides a visually appealing means of displaying the data we grab from YouTube.
QueueUp makes use of sweetalert to have visually appealing alert messages that get triggered when special events occur.
In this section, we summarize each of the currently functioning views of QueueUp.
This is a minimal view that acts as the first point of interaction with QueueUp. The purpose of this view is not to overwhelm the user immediately but to easily show the product logo, a brief phrase describing QueueUp, other useful links at the bottom of the page, and the abilities to login or signup.
Keep in mind that the footer is visible on every view, regardless of whether the user is logged in. The functionality of the navbar will change when the user logs in.
This is a very simple login view. QueueUp offers the functionality for a user's login to be "remembered", meaning their logged in session will be preserved for 7 days. A user needs to use this view in order to gain access to QueueUp's main functionality.
The layout to the Signup view is almost identical to that of Login's. In order to use QueueUp, a user must create an account at this page if they have not already. They will be automatically logged in after submitting the form, but if any errors are present in the form (E.g. the user name they have chosen already exists), an error dialog will be presented for them to fix those errors.
This page contains two sections: an area displaying info about the user (currently only their username) and an option to change their password, and an area displaying which streaming services the user has authorized. Users will use the Profile page in order to manage which streaming services they have currently activated for their QueueUp account.
This is one of the main views that demonstrates QueueUp's main functionality. Here you will see your main queue in a carousel pulling content from whichever streaming services you have enabled over at the Profile page. Clicking on an element in the queue of the main view will deliver you to view that content in a new tab at that provider's website.
Only users with admin privileges can access this page. When you first start the QueueUp server, an admin user with username="admin" and password="admin" is automatically created with admin privileges. At this page, admin users can modify admin privileges of other users. In the future, we plan to add to this page to possibly contain user statistics, such as how many users have authorized specific services.
This page contains more verbose info about QueueUp's purpose and use cases.
Here you will find information about the developers that make up Team Pelican. This view was adopted from one of our early CS326 projects, and still contains the functionality from that project to query for a team member with ?user=<team_members_username
. Try it out with ?user=nfuller
.
This page contains links to many of the early mockups of QueueUp. These serve as a starting point for us when we develop the front-end of QueueUp.
QueueUp implements statefulness similarly to how CS326 students were expected to do so in our 3rd individual project assignment. We use the express-session npm package to maintain a session variable and add a "user" object to that variable when a user logs into QueueUp.
For many of QueueUp's routes, we perform a check to see if the user session variable exists before performing the expected logic at that route. For example, for most of the routes defined in /routes/user.js
, we redirect users to the Login page if they attempt to navigate to a route while not logged in.
Sometimes, we also feel it is necessary to deliver messages when redirecting users. We use the connect-flash
middleware to attach a useful message to a request for a resource that we eventually redirect to. When we redirect to that resource, the message is extracted and delivered to the view.
An example of a stateful flow of QueueUp would be that which a user takes when changing their password. Assuming the user is already at the Profile page, they will initiate an HTTP POST request upon submission of the change-pass form which gets handled in /routes/user.js
. If for some reason the user's session expired before they submitted the form, they will be redirected to the login page. Otherwise, we make use of our MongoDB-user library to check to see that the current password they have entered is correct in order to avoid malicious entities attempting changing passwords of an already logged in user. If the incorrect password is entered too many times, they are redirected to /user/logout which destroys the user session and redirects to /user/login. Otherwise, the database is updated with the new password and the user gets redirected to the Profile page and is shown a message stating that they have successfully changed their password.
To get a much simpler idea of how statefulness is implemented in QueueUp, please take a look into /routes/index.js
which has much less complex logic than the user routes file, but you may see that it redirects to many of the routes defined in /routes/user.js
.
QueueUp uses MongoDB for data persistence. Currently, we only make requests to a local MongoDB instance, but do plan to move to a remote instance with MongoLab. To access our MongoDB instance through JavaScript, we use the mongodb npm package.
/lib/MongoDB-basic.js
is our file that performs basic operations on the MongoDB database, such as adding, updating, finding, and removing data. We use this lib file in /lib/MongoDB-user.js
in order to interact with our database.
/lib/MongoDB-user.js
is used to perform QueueUp specific accesses to the database. For example, the login
function will take a username and password to see if the username/password pair exists in the database. In order to do this, it hashes the password entered (using bcrypt) and compares it to the password returned by the findData
function in /lib/MongoDB-basic.js
.
As you can see, we do not use an object modeling service like Mongoose to manage how data is saved to our database, however there has been discussion of refactoring QueueUp to do so.
Our database contains solely a Users collection, where each document in that collection contains information about that user and the API tokens that will be used for pulling data respective to their accounts for those services. The User document is outlined as follows:
{
_id : ObjectId("<unique_id_generated_by_mongo>"),
"user" : {
"username" : "<the_users_username>",
"password" : "<the_users_hashed_password>",
"admin" : <boolean_whether_user_has_admin_privileges>
},
"APIS" : {
"YouTube" : "<encrypted_youtube_access_token>",
"Twitch" : "<encrypted_twitch_access_token>",
"Netflix" : "<encrypted_netflix_access_token>",
"HBOGo" : "<encrypted_hbogo_access_token>",
"Amazon" : "<encrypted_amazon_prime_instant_access_token>"
}
}
Note: You see that the list of APIs above implies that several services plan to be implemented, but it is likely that only a subset of these services will be implemented if any of those services do not currently offer public APIs.