Victor, Jack, Reuben, Tony
Installation
- git clone git@github.com:fac-17/Week6-DFIO.git
- cd into folder
- npm install
- npm start
- Use ES6
- Branches named Feature/branch
- Work on remote branches together
- Never push to master
- Use const for any variables that don't change
- Use === not ==
- Ensure roughly equal commits
- Make sure everyone know what they're doing on the presentation
- Login form with 2 fields - username and password
- Users only have to log in once (i.e. implement a cookie-based session on login)
- Username is visible on each page of the site after logging in
- Any user-submitted content should be labelled with the authors username
- 2 Options
- Refund functionality
- Admins that can add new items
- There should be protected routes and unprotected routes that depend on the user having a cookie or not (or what level of access they have).
- Website content should be stored in a database
- Include thorough tests on the back-end, testing pure functions and testing routes using Supertest. If you make external API calls, use Nock to mock the response for your tests.
- Test front-end logic, we don't expect tests on the DOM.
relating to authentication
- Create user with login fields and appropriate authentication
- Login with authentication
- Add login to cookies
- Redirect login if logged in to the inventory page
- Redirect inventory page if not logged in to the inventory page (protected routes covered by this?) other tasks - do if we have time after authentication work is completed
- Get satchel working
- Coins can't go below 0
- Inventory and satchel table dom update functions need to be updated so that titles aren't deleted
- LeaderBoard page
-
Database: Add new column to USERS table to store hashed password
-
Route 1: Create new account
- Client side Validation
- Add password fields
- Password long enough/short enough
- Completed all the fields?
- Passwords complex enough
- Check username pattern, setcustomvalidity to say only lowercase letters and numbers (no spaces)
- Check password pattern, set custom validity to check there is a lowercase letter, uppercase letter and number
- Client side Validation
-
Route 2: Log in with existing account
- Hash, salt and store password.
-
Route 1: Create new account
- Pass details (username, password) to the back-end, and make sure they are stored correctly on the backend
- Create 'logged-in' token.
- Redirect to user to inventory, logged-in as new account.
-
Route 2: Log in with existing account
- Check log in details against USERS table
- Generate token and pass to front end
-
Check existing token and redirect to relevant endpoint.
- If no token > to Login/Create account page
- Set timeout on token
- If token > to Inventory page as user
- Remove front end validation on password on login
- Setup alert if password is wrong
- Setup alert if username exists(response is being sent back to /login request, which is sent by form, not front-end XMLHttpRequest. figure out how to access this response.)
- Logout button on inventory page that removes cookies, redirect to home page
- Add button to inventory page
- Remove token
- Redirect to home page
- Rename index.js file to inventory.js and update wherever necessary
- Redirect to login if you attempt to hit the inventory endpoint without a token
- Add column names to price and quantity
- Not allow items to be bought when not enough gold
- Call back end when item is bought with user gold
- Function called checkEnoughGold(item name, username)
- Check DB with checkEnoughGold to deterine if user can buy item (return True/False)
- If true, buy item
- If false, response.end(Can't buy item)
- Promise > checkEnoughGold (Error > Break) > Buy Item
- Different permissions (user, admin that can POST new items)
- Leaderboard
- Generate secure secret key
- The column name for us was '?column?'
const checkEnoughGold = (userName, itemName, cb) => {
decodedItemName = decodeURI(itemName);
databaseConnection.query(
`SELECT (SELECT gold_pieces FROM users WHERE name = '${userName}') - (SELECT item_price FROM inventory WHERE inventory.item_name = '${decodedItemName}')`,
(err, res) => {
if (err) cb(err);
else {
const valueDiff = res.rows[0]["?column?"];
const enoughGold = valueDiff >= 0 ? true : false;
cb(null, enoughGold);
}
}
);
};
User your tokens to stay logged in and check them every time you want to access something on the server
Victor, Jack, Reuben, Tony
Installation
- git clone git@github.com:fac-17/Week6-DFIO.git
- cd into folder
- npm install
- npm start
- Use ES6
- Branches named Feature/branch
- Work on remote branches together
- Never push to master
- Use const for any variables that don't change
- use === not ==
- Ensure roughly equal commits
-
Simple web app with a node server and a database.
-
Content dynamic, but DOM manipulation kept to a minimum
-
Mobile-first design
-
Your database comes with a schema, which should be documented in your readme (along with any other architectural decisions).
-
Database hosted on Heroku, or locally.
-
Build script for your database.
-
Security concerns appropriately considered (you must protect against script injections!).
-
Clear user journey (even if you take one of our suggested ideas, document the user journey in your readme)
-
Test your server routes with supertest
-
Test your pure functions both server and client side
-
Set up a test database so that you can test your database queries
- Interesting articles
- Railway system
- Upload your favourite worst movies (or albums), everyone else can comment and/or upvote it
- Fantasy shop
- First screen: Enter your name
- Second screen: Shop with a list of item + price. Also shows current user and coins left
- Third screen: Leaderboard(stretch goal)
https://www.lucidchart.com/documents/edit/c79acf78-839b-4481-b6c8-f4475aff4a22/0
- User can enter username and gets assigned coin allocation.
- User can access the shop and select items.
- User coin allocation gets updated accordingly.
- Security against SQL injections.
- Include test database to test queries.
- User can view full list of items owned.
const populateSatchel = (arr) => {
const satchelTable = document.querySelector(".satchel_table");
while (satchelTable.firstChild) {
satchelTable.firstChild.remove()
}
arr.forEach(item => {
let newItem = document.createElement('tr');
// Item details filled in
let newItemName = document.createElement('th');
newItemName.innerText = item.item_name;
newItem.appendChild(newItemName);
let newItemDescription = document.createElement('th');
newItemDescription.innerText = item.item_description;
newItem.appendChild(newItemDescription);
let newItemPower = document.createElement('th');
newItemPower.innerText = item.item_power;
newItem.appendChild(newItemPower);
// Full new row added
satchelTable.appendChild(newItem);
})
}
- User can buy multiple items.
- User can get refunds or exchange items.
- Items get (hidden) power value to compare with other users.
- Log back in and play again (keep the score)
- Build File Structure - separate front-end backend
- Install dependencies and link to package.json
- Skeleton HTML, CSS
- Setup server files - server router handler
- Setup database connection js file
- Get heroku setup
- Setup Travis
- Add env variables inside travis
- Build database on heroku
- Write build script
- Setup test DATABASE -
- do the following on your local terminal CREATE DATABASE final_factasy_db_test; CREATE USER final_factasy WITH SUPERUSER PASSWORD 'waterbottle'; ALT
- Create env file and add links
- Split up: pair 1: setup testing, pair 2: setup basic server files. Whoever finishes first finish off deploying to heroku/travis
- Come together to create database files and database testing
Write sample tests for
- Front end pure functions (tape)
- Server endpoints (supertest)
- Test database (getData function)
- Write backend JS + SQL to GET and POST info to database
- Create new user in user table
- Buy items.
- Reduce gold in user table
- Reduce quantity in inventory table
- Create new row in ownership table
- Get score of user from from all three tabls
- Return what specific user owns from ownership table
- convert sql queries to use variables from user input
- Write -front end JS to render call back end and render DOM
- Back end server call
- Create new user with post
- 'Log in' with user
- Render current inventory
- Render current user ownership
- Item with quantity = 0 becomes non-available
- Helper function to stream chunks of data
- Plan breaks
- PROTECT AGAINST SQL INJECTION.
- Do Readme
- Installation instructions
- User journey
- Schema
t.error(err, "No Error");
//Assert that db_build finished
//successfully with no errors
<form class="form" autocomplete="off" action="/newuser" method="post">
Tests require the results to be in a specific order even if we don't. We had to order our results on an arbitray value just to get the tests to pass.
Update a user's wallet after they've purchased an item:
databaseConnection.query(`UPDATE users SET gold_pieces = gold_pieces - 1 WHERE name = 'Jon';
UPDATE inventory
SET item_quantity = item_quantity - 1
WHERE item_name = 'Cape';
INSERT INTO ownership(owner_id, item_id)
VALUES ((SELECT id FROM users WHERE name = 'Jon'), (SELECT id FROM inventory WHERE item_name = 'Cape') )`
Get score for all users by joining three tables
databaseConnection.query(`SELECT users.name, SUM(inventory.item_power)
FROM users
INNER JOIN ownership
ON users.id = ownership.owner_id
INNER JOIN inventory
ON inventory.id = ownership.item_id
GROUP BY users.id`
CREATE TABLE inventory (
id SERIAL PRIMARY KEY,
...
item_quantity INTEGER DEFAULT 0,
item_price INTEGER DEFAULT 20,
item_power INTEGER DEFAULT 0
);
Add an onclick
attribute
let newItemButton = document.createElement('button');
newItemButton.innerText = 'Buy now';
newItemButton.classList.add('buy-now');
newItemButton.setAttribute('onclick', `buyItem('${item.item_name}')`);
newItem.appendChild(newItemButton);
Which translates in html to:
<button class="buy-now" onclick="buyItem('Dagger')">Buy now</button>
-
Create a local test database
-
Run your build script with \i and route to build file
-
also run the build script once on the remote database
Here testing the buy item
function:
test('Reduce quantity in inventory table after purchase', (t) => {
runDbBuild((err, res) => {
t.error(err, 'No error');
let expected = [{ item_name: 'Dagger', item_quantity: 3, item_price: 8 }, { item_name: 'Cape', item_quantity: 4, item_price: 3 }, { item_name: 'Walking stick', item_quantity: 10, item_price: 2 }, { item_name: 'Toothbrush', item_quantity: 2, item_price: 10 }]
queries.buyItem((err, result) => {
if (err) console.log(err);
//select inventory inside callback for buyItem(wait for completion of buyItem before trying to select the inventory)
queries.getInventory((err, result) => {
if (err) console.log(err);
t.deepEqual(result, expected, 'Cape should have a quantity of 4')
t.end();
})
})
})
})
- Still need direction to make sure you are building the right functions...
- ..and they take the right arguments etc. e.g. username vs user id
- How to test SQL functions that only update - don't return values - nest a select query inside, and test the response from that.
queries.buyItem((err, result) => {
if (err) console.log(err);
queries.getUsers((err, result) => {
if (err) console.log(err);
t.deepEqual(result, expected, 'Jon should have 1 less gold piece')
t.end()
})
... approximately 10 minutes per hour 💪
module.exports = {
handleHome,
handlePublic,
handleDbNewUser,
handleGetInventory,
handleDbLogin,
handleGetUser,
handleBuyItem
};
...headache game getting even stronger 🤯
else if (endpoint.includes('public')) {
handlers.handlePublic(request, response);
} else if (endpoint.includes('/newuser')) {
handlers.handleDbNewUser(request, response);
} else if (endpoint.includes('/login')) {
handlers.handleDbLogin(request, response);
} else if (endpoint.includes('/getinventory')) {
handlers.handleGetInventory(request, response);
} else if (endpoint.includes('/getitemsowned')) {
handlers.handleGetItemsOwned(request, response);
} else if (endpoint.includes('/buyitem')) {
handlers.handleBuyItem(request, response);
} else if (endpoint.includes('/getuser')) {
handlers.handleGetUser(request, response);
...