diff --git a/Gemfile b/Gemfile index b1a320395a..049d478abd 100644 --- a/Gemfile +++ b/Gemfile @@ -11,3 +11,15 @@ end group :development, :test do gem 'rubocop', '1.20' end + +gem "pg", "~> 1.5" + +gem "rack-test", "~> 1.1" + +gem "sinatra", "~> 2.2" + +gem "webrick", "~> 1.7" + +gem "sinatra-contrib", "~> 2.2" + +gem "bcrypt", "~> 3.1" diff --git a/Gemfile.lock b/Gemfile.lock index 66064703c7..1087a945ea 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -3,11 +3,21 @@ GEM specs: ansi (1.5.0) ast (2.4.2) + bcrypt (3.1.18) diff-lcs (1.4.4) docile (1.4.0) + multi_json (1.15.0) + mustermann (2.0.2) + ruby2_keywords (~> 0.0.1) parallel (1.20.1) parser (3.0.2.0) ast (~> 2.4.1) + pg (1.5.3) + rack (2.2.7) + rack-protection (2.2.4) + rack + rack-test (1.1.0) + rack (>= 1.0, < 3) rainbow (3.0.0) regexp_parser (2.1.1) rexml (3.2.5) @@ -36,6 +46,7 @@ GEM rubocop-ast (1.11.0) parser (>= 3.0.1.1) ruby-progressbar (1.11.0) + ruby2_keywords (0.0.5) simplecov (0.21.2) docile (~> 1.1) simplecov-html (~> 0.11) @@ -46,18 +57,38 @@ GEM terminal-table simplecov-html (0.12.3) simplecov_json_formatter (0.1.3) + sinatra (2.2.4) + mustermann (~> 2.0) + rack (~> 2.2) + rack-protection (= 2.2.4) + tilt (~> 2.0) + sinatra-contrib (2.2.4) + multi_json + mustermann (~> 2.0) + rack-protection (= 2.2.4) + sinatra (= 2.2.4) + tilt (~> 2.0) terminal-table (3.0.1) unicode-display_width (>= 1.1.1, < 3) + tilt (2.1.0) unicode-display_width (2.0.0) + webrick (1.8.1) PLATFORMS ruby + x86_64-linux DEPENDENCIES + bcrypt (~> 3.1) + pg (~> 1.5) + rack-test (~> 1.1) rspec rubocop (= 1.20) simplecov simplecov-console + sinatra (~> 2.2) + sinatra-contrib (~> 2.2) + webrick (~> 1.7) RUBY VERSION ruby 3.0.2p107 diff --git a/README.md b/README.md index 465eda879b..87c36e6b6d 100644 --- a/README.md +++ b/README.md @@ -1,123 +1,57 @@ -Chitter Challenge -================= +# chitter-challenge -* Feel free to use Google, your notes, books, etc. but work on your own -* If you refer to the solution of another coach or student, please put a link to that in your README -* If you have a partial solution, **still check in a partial solution** -* You must submit a pull request to this repo with your code by 10am Monday morning +A Twitter clone built as part of the Makers Academy software development course. -Challenge: -------- +## Features -As usual please start by forking this repo. +Users can do the following: -We are going to write a small Twitter clone that will allow the users to post messages to a public stream. +- Sign up +- login/logout +- Post a message (peep) to chitter +- View all peeps in reverse chronological order +- See the timestamps on peeps +- See peeps their tagged in (tagging functionality still in progress) -Features: -------- +## How to run +```shell +bundle install +rackup ``` -STRAIGHT UP -As a Maker -So that I can let people know what I am doing -I want to post a message (peep) to chitter +## Database Structure -As a maker -So that I can see what others are saying -I want to see all peeps in reverse chronological order +Database Table design template can be found [here](chitter_two_tables_design_recipe.md). The database diagram below is generated by dbdiagram.io using DBML converted from the SQL seed code: +![db-diagram](./docs/db-diagram.png) -As a Maker -So that I can better appreciate the context of a peep -I want to see the time at which it was made +## Class Design -As a Maker -So that I can post messages on Chitter as me -I want to sign up for Chitter +Ruby Model and Repository class design template can be found [here](chitter_model_and_repository_design_recipe.md). -HARDER +## Webpage view -As a Maker -So that only I can post messages on Chitter as me -I want to log in to Chitter +![homepage screenshot](./docs/homepage-screenshot.png) +![userpage screenshot](./docs/userpage-screenshot.png) -As a Maker -So that I can avoid others posting messages on Chitter as me -I want to log out of Chitter +## Pending TODOs -ADVANCED +- User "tagging" functionality +- improve CSS design -As a Maker -So that I can stay constantly tapped in to the shouty box of Chitter -I want to receive an email if I am tagged in a Peep -``` - -Technical Approach: ------ - -In the last two weeks, you integrated a database using the `pg` gem and Repository classes. You also implemented small web applications using Sinatra, RSpec, HTML and ERB views to make dynamic webpages. You can continue to use this approach when building Chitter Challenge. - -You can refer to the [guidance on Modelling and Planning a web application](https://github.com/makersacademy/web-applications/blob/main/pills/modelling_and_planning_web_application.md), to help you in planning the different web pages you will need to implement this challenge. If you'd like to deploy your app to Heroku so other people can use it, [you can follow this guidance](https://github.com/makersacademy/web-applications/blob/main/html_challenges/07_deploying.md). - -If you'd like more technical challenge now, try using an [Object Relational Mapper](https://en.wikipedia.org/wiki/Object-relational_mapping) as the database interface, instead of implementing your own Repository classes. - -Some useful resources: -**Ruby Object Mapper** -- [ROM](https://rom-rb.org/) - -**ActiveRecord** -- [ActiveRecord ORM](https://guides.rubyonrails.org/active_record_basics.html) -- [Sinatra & ActiveRecord setup](https://learn.co/lessons/sinatra-activerecord-setup) - -Notes on functionality: ------- - -* You don't have to be logged in to see the peeps. -* Makers sign up to chitter with their email, password, name and a username (e.g. samm@makersacademy.com, password123, Sam Morgan, sjmog). -* The username and email are unique. -* Peeps (posts to chitter) have the name of the maker and their user handle. -* Your README should indicate the technologies used, and give instructions on how to install and run the tests. - -Bonus: ------ +## Built with -If you have time you can implement the following: +- Ruby +- Rspec +- Sinatra +- PostgreSQL +- BCrypt +- Render -* In order to start a conversation as a maker I want to reply to a peep from another maker. +## Deployment -And/Or: +Initially deployed on Render (https://chitter-the-chit-chat-app.onrender.com) but is currently inactive. -* Work on the CSS to make it look good. - -Good luck and let the chitter begin! - -Code Review ------------ - -In code review we'll be hoping to see: - -* All tests passing -* High [Test coverage](https://github.com/makersacademy/course/blob/main/pills/test_coverage.md) (>95% is good) -* The code is elegant: every class has a clear responsibility, methods are short etc. - -Reviewers will potentially be using this [code review rubric](docs/review.md). Referring to this rubric in advance may make the challenge somewhat easier. You should be the judge of how much challenge you want at this moment. - -Notes on test coverage ----------------------- - -Please ensure you have the following **AT THE TOP** of your spec_helper.rb in order to have test coverage stats generated -on your pull request: - -```ruby -require 'simplecov' -require 'simplecov-console' - -SimpleCov.formatter = SimpleCov::Formatter::MultiFormatter.new([ - SimpleCov::Formatter::Console, - # Want a nice code coverage website? Uncomment this next line! - # SimpleCov::Formatter::HTMLFormatter -]) -SimpleCov.start -``` +## Contact -You can see your test coverage when you run your tests. If you want this in a graphical form, uncomment the `HTMLFormatter` line and see what happens! +- [Caroline Evans](mailto:carolineevans261@gmail.com) diff --git a/app.rb b/app.rb new file mode 100644 index 0000000000..845ffcc6f1 --- /dev/null +++ b/app.rb @@ -0,0 +1,188 @@ +# file: app.rb +require 'sinatra' +require "sinatra/reloader" +require 'bcrypt' +require_relative 'lib/database_connection' +require_relative 'lib/user_repository' +require_relative 'lib/peep_repository' + +DatabaseConnection.connect + +class Application < Sinatra::Base + enable :sessions + configure :development do + register Sinatra::Reloader + also_reload 'lib/user_repository' + also_reload 'lib/peep_repository' + end + + # ------------- Homepage Route ---------------------------------------- + + get '/' do + if session[:user_id] != nil + # No user id in the session + # so the user is not logged in. + return redirect('/userpage') + end + peep_repo = PeepRepository.new + @user_repo = UserRepository.new + @peeps = peep_repo.all.sort_by! { |peep| peep.time }.reverse! + + return erb(:index) + end + + post '/' do + session[:user_id] = nil + return redirect('/') + + end + + # ------------- Sign Up Routes ---------------------------------------- + + get '/signup' do + return erb(:signup) + end + + post '/signup' do + + users = UserRepository.new + + if users.all.any? { |user| user.email == params[:email] } + status 400 + return "Email address already signed up." + end + + if users.all.any? { |user| user.username == params[:username] } + status 400 + return "Username already taken, please choose another." + end + + if invalid_signup_params? || invalid_user_params? + status 400 + return 'Invalid credentials, please try again.' + end + + encrypted_password = BCrypt::Password.create(params[:password]) + + user = User.new + user.name = params[:name] + user.email = params[:email] + user.username = params[:username] + user.password = encrypted_password + + users.create(user) + + user = users.find_by_username(user.username) + session[:user_id] = user.id + + redirect "/userpage" + + end + + # ------------- Peeps Routes ---------------------------------------- + + get '/peeps' do + peep_repo = PeepRepository.new + @user_repo = UserRepository.new + @peeps = peep_repo.all.sort_by! { |peep| peep.time }.reverse! + + return erb(:peeps) + end + + post '/peeps' do + if invalid_peep_params? + status 400 + return 'Invalid peep, please try again.' + end + + peep = Peep.new + peep_repo = PeepRepository.new + @user_repo = UserRepository.new + + peep.content = params[:content] + peep.time = DateTime.now + peep.user_id = params[:user_id] + + peep_repo.create(peep) + + @peeps = peep_repo.all.sort_by! { |peep| peep.time }.reverse! + + return erb(:peeps) + + end + + # ------------- User Page Routes ---------------------------------------- + + post '/userpage' do + + if invalid_user_params? + status 400 + return 'Invalid user details.' + end + + user_repo = UserRepository.new + + if user_repo.all.none? { |user| user.username == params[:username] } + status 400 + return "Username does not exist, please try again." + end + + username = params[:username] + entered_password = params[:password] + + user = user_repo.find_by_username(username) + + stored_password = BCrypt::Password.new(user.password) + if stored_password == entered_password + # Set the user ID in session + session[:user_id] = user.id + + redirect "/userpage" + + + else + status 400 + return 'Invalid user details.' + end + end + + get '/userpage' do + if session[:user_id] == nil + # No user id in the session + # so the user is not logged in. + return redirect('/') + else + # The user is logged in, display + # their account page. + + user_id = session[:user_id] + + @user_repo = UserRepository.new + peep_repo = PeepRepository.new + + @user = @user_repo.find(user_id) + @peeps = peep_repo.find_by_owner(@user.id).sort_by! { |peep| peep.time }.reverse! + @tagged_peeps = peep_repo.find_by_tagged_user(@user.id) + + return erb(:userpage) + end + + end + + # ------------- helper methods ---------------------------------------- + + def invalid_user_params? + params[:username] == nil || params[:username].match?(/[^a-zA-Z0-9 ]/) || + params[:password] == nil || params[:password] == "" + end + + def invalid_peep_params? + params[:content] == nil || params[:content] == "" + end + + def invalid_signup_params? + params[:name] == nil || params[:name].match?(/[^a-zA-Z0-9 ]/) || + params[:email] == nil || !params[:email].match?(/[@]/) + end + +end \ No newline at end of file diff --git a/chitter_model_and_repository_design_recipe.md b/chitter_model_and_repository_design_recipe.md new file mode 100644 index 0000000000..df593ae990 --- /dev/null +++ b/chitter_model_and_repository_design_recipe.md @@ -0,0 +1,365 @@ +# Chitter Peeps & Users - Model and Repository Classes Design Recipe + + +## 1. Design and create the Table + + +``` +Table: peeps + +Columns: +id | content | time | user_id + +Table: users + +Columns: +id | name | email | username + +``` + +## 2. Create Test SQL seeds + +Your tests will depend on data stored in PostgreSQL to run. + +If seed data is provided (or you already created it), you can skip this step. + +```sql +-- (file: spec/seeds_chitter.sql) + +-- First, you'd need to truncate the table - this is so our table is emptied between each test run, +-- so we can start with a fresh state. +-- (RESTART IDENTITY resets the primary key) + +TRUNCATE TABLE peeps, users, peeps_users RESTART IDENTITY; + +-- Below this line there should only be `INSERT` statements. +-- Replace these statements with your own seed data. + +INSERT INTO users (name, email, username) VALUES ('Caroline', 'carolinesemail@email.com', 'caro'); +INSERT INTO users (name, email, username) VALUES ('Philip', 'philsemail@email.com', 'phil'); + + +INSERT INTO peeps (content, time, user_id) VALUES ('This is the first Peep', '20230506 10:22:09 AM', 1); +INSERT INTO peeps (content, time, user_id) VALUES ('This is the second Peep', '20230507 03:35:35 PM', 1); +INSERT INTO peeps (content, time, user_id) VALUES ('This is the third Peep', '20230508 09:42:01 AM', 2); +INSERT INTO peeps (content, time, user_id) VALUES ('This is the forth Peep', '20230509 11:12:59 PM', 2); + +INSERT INTO peeps_users (peep_id, user_id) VALUES (1, 2); +INSERT INTO peeps_users (peep_id, user_id) VALUES (2, 2); +``` + +Run this SQL file on the database to truncate (empty) the table, and insert the seed data. Be mindful of the fact any existing records in the table will be deleted. + +```bash +psql -h 127.0.0.1 chitter_test < seeds_chitter.sql +``` + +## 3. Define the class names + +Usually, the Model class name will be the capitalised table name (single instead of plural). The same name is then suffixed by `Repository` for the Repository class name. + +```ruby +# Table name: peeps + +# Model class +# (in lib/peep.rb) +class Peep +end + +# Repository class +# (in lib/peep_repository.rb) +class PeepRepository +end + +# Table name: users + +# Model class +# (in lib/user.rb) +class User +end + +# Repository class +# (in lib/user_repository.rb) +class UserRepository +end +``` + +## 4. Implement the Model class + +Define the attributes of your Model class. You can usually map the table columns to the attributes of the class, including primary and foreign keys. + +```ruby + +# Table name: peeps + +# Model class +# (in lib/peep.rb) + +class Peep + # Replace the attributes by your own columns. + attr_accessor :id, :content, :time, :user_id +end + +# The keyword attr_accessor is a special Ruby feature +# which allows us to set and get attributes on an object, + +# Table name: users + +# Model class +# (in lib/user.rb) + +class User + # Replace the attributes by your own columns. + attr_accessor :id, :name, :email, :username +end + +``` + +*You may choose to test-drive this class, but unless it contains any more logic than the example above, it is probably not needed.* + +## 5. Define the Repository Class interface + +Your Repository class will need to implement methods for each "read" or "write" operation you'd like to run against the database. + +Using comments, define the method signatures (arguments and return value) and what they do - write up the SQL queries that will be used by each method. + +```ruby +# Table name: peeps + +# Repository class +# (in lib/peep_repository.rb) + +class PeepRepository + + # Selecting all records + # No arguments + def all + # Executes the SQL query: + # SELECT id, content, time, user_id FROM peeps; + + # Returns an array of Peep objects. + end + + def create(peep) + # Executes the SQL query: + # INSERT INTO peeps (content, time, user_id) VALUES ($1, $2, $3); + + # returns nil + end + + def find_by_owner(user_id) + # Executes the SQL query: + # SELECT id, content, time, user_id FROM peeps WHERE user_id = $1; + + # Returns an array of Peep objects. + end + + def find_by_tagged_user(user_id) + # Executes the SQL query: + # SELECT peeps.id, peeps.content, peeps.time, peeps.user_id FROM peeps JOIN peeps_users ON peeps.id = peeps_users.peep_id JOIN users ON users.id = peeps_users.user_id WHERE users.id = $1; + + # Returns an array of Peep objects. + end + +end + +# Table name: users + +# Repository class +# (in lib/user_repository.rb) + +class UserRepository + + def find(username) + # Executes the SQL query: + # SELECT id, name, email, username FROM users WHERE username = $1; + + # Returns a User object. + end + + def create(user) + # Executes the SQL query: + # INSERT INTO users (name, email, username) VALUES ($1, $2, $3); + + # returns nil + end +end +``` + +## 6. Write Test Examples + +Write Ruby code that defines the expected behaviour of the Repository class, following your design from the table written in step 5. + +These examples will later be encoded as RSpec tests. + +```ruby +# Peeps testing +# 1 +# Get all peeps +repo = PeepRepository.new + +peeps = repo.all + +peeps.length # => 4 + +peeps[0].id # => 1 +peeps[0].content # => 'This is the first Peep' +peeps[0].time # => '20230506 10:22:09 AM' +peeps[0].user_id # => 1 + +peeps[1].id # => 2 +peeps[1].content # => 'This is the second Peep' +peeps[1].time # => '20230507 03:35:35 PM' +peeps[1].user_id # => 1 + +peeps[2].id # => 3 +peeps[2].content # => 'This is the third Peep' +peeps[2].time # => '20230508 09:42:01 AM' +peeps[2].user_id # => 2 + +peeps[3].id # => 4 +peeps[3].content # => 'This is the forth Peep' +peeps[3].time # => '20230509 11:12:59 PM' +peeps[3].user_id # => 2 + +# 2 +# creates a new peep +peep = Peep.new +peep.content = 'This is the fifth Peep' +peep.time = '20230422 08:22:46 PM' +peep.user_id = 2 + +repo = PeepRepository.new + +repo.create(peep) + +last_peep = repo.all.last + +last_peep.id # => 5 +last_peep.content # => 'This is the fifth Peep' +last_peep.time # => '20230422 08:22:46 PM' +last_peep.user_id # => 2 + +# 3 +# find all peeps by the same user +repo = PeepRepository.new + +peeps = repo.find_by_owner(1) + +peeps.length # => 2 + +peeps[0].id # => 1 +peeps[0].content # => 'This is the first Peep' +peeps[0].time # => '20230506 10:22:09 AM' +peeps[0].user_id # => 1 + +peeps[1].id # => 2 +peeps[1].content # => 'This is the second Peep' +peeps[1].time # => '20230507 03:35:35 PM' +peeps[1].user_id # => 1 + +# 4 +# find all peeps that a user is tagged in + +repo = PeepRepository.new + +peeps = repo.find_by_tagged_user(1) + +peeps.length # => 0 + +peeps = repo.find_by_tagged_user(2) + +peeps.length # => 2 + +peeps[0].id # => 1 +peeps[0].content # => 'This is the first Peep' +peeps[0].time # => '20230506 10:22:09 AM' +peeps[0].user_id # => 1 + +peeps[1].id # => 2 +peeps[1].content # => 'This is the second Peep' +peeps[1].time # => '20230507 03:35:35 PM' +peeps[1].user_id # => 1 + + +# User testing +# 1 +# find a user + +repo = UserRepository.new + +user = repo.find('phil') + +user.id # => 2 +user.name # => 'Philip' +user.email # => 'philsemail@email.com' +user.username # => 'phil' + + +# 2 +# create a new user + +user = User.new +user.name = 'Pip' +user.email = 'pipsemail@email.com' +user.username = 'pip1' + +repo = UserRepository.new + +repo.create(user) + +last_user = repo.find('pip1') + +last_user.id # => 3 +last_user.name # => 'Pip' +last_user.email # => 'pipsemail@email.com' +last_user.username # => 'pip1' + + +``` + +Encode this example as a test. + +## 7. Reload the SQL seeds before each test run + +Running the SQL code present in the seed file will empty the table and re-insert the seed data. + +This is so you get a fresh table contents every time you run the test suite. + +```ruby +# file: spec/peep_repository_spec.rb + +def reset_chitter_tables + seed_sql = File.read('spec/seeds_chitter.sql') + connection = PG.connect({ host: '127.0.0.1', dbname: 'chitter_test' }) + connection.exec(seed_sql) +end + +describe PeepRepository do + before(:each) do + reset_chitter_tables + end + + # (your tests will go here). +end + +# file: spec/user_repository_spec.rb + +def reset_chitter_tables + seed_sql = File.read('spec/seeds_chitter.sql') + connection = PG.connect({ host: '127.0.0.1', dbname: 'chitter_test' }) + connection.exec(seed_sql) +end + +describe UserRepository do + before(:each) do + reset_chitter_tables + end + + # (your tests will go here). +end +``` + +## 8. Test-drive and implement the Repository class behaviour + +_After each test you write, follow the test-driving process of red, green, refactor to implement the behaviour._ \ No newline at end of file diff --git a/chitter_two_tables_design_recipe.md b/chitter_two_tables_design_recipe.md new file mode 100644 index 0000000000..9580f4d77a --- /dev/null +++ b/chitter_two_tables_design_recipe.md @@ -0,0 +1,159 @@ +# Chitter - Two Tables Design Recipe Template + +## 1. Extract nouns from the user stories or specification + +``` +STRAIGHT UP + +As a Maker +So that I can let people know what I am doing +I want to post a message (peep) to chitter + +As a maker +So that I can see what others are saying +I want to see all peeps in reverse chronological order + +As a Maker +So that I can better appreciate the context of a peep +I want to see the time at which it was made + +As a Maker +So that I can post messages on Chitter as me +I want to sign up for Chitter + +HARDER + +As a Maker +So that only I can post messages on Chitter as me +I want to log in to Chitter + +As a Maker +So that I can avoid others posting messages on Chitter as me +I want to log out of Chitter + +ADVANCED + +As a Maker +So that I can stay constantly tapped in to the shouty box of Chitter +I want to receive an email if I am tagged in a Peep +``` + +``` +Nouns: + +peep, time, sign up, log in, log out, tagged, email +``` + +## 2. Infer the Table Name and Columns + +Put the different nouns in this table. Replace the example with your own nouns. + +| Record | Properties | +| --------------------- | ------------------ | +| peep | content, time, tag +| user | name, email, username, tag + + +1. Name of the first table (always plural): `peeps` + + Column names: `content`, `time`, `user_id` + +2. Name of the second table (always plural): `users` + + Column names: `name`, `email` , `username` + +## 3. Decide the column types. + +[Here's a full documentation of PostgreSQL data types](https://www.postgresql.org/docs/current/datatype.html). + +Most of the time, you'll need either `text`, `int`, `bigint`, `numeric`, or `boolean`. If you're in doubt, do some research or ask your peers. + +Remember to **always** have the primary key `id` as a first column. Its type will always be `SERIAL`. + +``` +# EXAMPLE: + +Table: peeps +id: SERIAL +content: text +time: datetime +user_id: int + +Table: users +id: SERIAL +name: text +email: text +username: text +``` + +## 4. Design the Many-to-Many relationship + +Make sure you can answer YES to these two questions: + +1. Can one [TABLE ONE] have many [TABLE TWO]? (Yes/No) +2. Can one [TABLE TWO] have many [TABLE ONE]? (Yes/No) + +``` +1. Can one peep have many users? YES +2. Can one user have many peeps? YES +``` + +_If you would answer "No" to one of these questions, you'll probably have to implement a One-to-Many relationship, which is simpler. Use the relevant design recipe in that case._ + +## 5. Design the Join Table + +The join table usually contains two columns, which are two foreign keys, each one linking to a record in the two other tables. + +The naming convention is `table1_table2`. + +``` +Join table for tables: peeps and users +Join table name: peeps_users +Columns: peep_id, user_id +``` + +## 4. Write the SQL. + +```sql +-- file: peeps_users.sql + +DROP TABLE peeps_users; +DROP TABLE peeps; +DROP TABLE users; + + +-- Create the first table. +CREATE TABLE users ( + id SERIAL PRIMARY KEY, + name text, + email text, + username text +); + +-- Create the second table. +CREATE TABLE peeps ( + id SERIAL PRIMARY KEY, + content text, + time timestamp, + user_id int, + constraint fk_user foreign key(user_id) references users(id) on delete cascade +); + + +-- Create the join table. +CREATE TABLE peeps_users ( + peep_id int, + user_id int, + constraint fk_peeps foreign key(peep_id) references peeps(id) on delete cascade, + constraint fk_users foreign key(user_id) references users(id) on delete cascade, + PRIMARY KEY (peep_id, user_id) +); + +``` + +## 5. Create the tables. + +```bash +psql -h 127.0.0.1 chitter < peeps_users.sql +psql -h 127.0.0.1 chitter_test < peeps_users.sql +``` diff --git a/config.ru b/config.ru new file mode 100644 index 0000000000..18af4f3388 --- /dev/null +++ b/config.ru @@ -0,0 +1,3 @@ +# file: config.ru +require './app' +run Application \ No newline at end of file diff --git a/docs/db-diagram.png b/docs/db-diagram.png new file mode 100644 index 0000000000..33529e28df Binary files /dev/null and b/docs/db-diagram.png differ diff --git a/docs/homepage-screenshot.png b/docs/homepage-screenshot.png new file mode 100644 index 0000000000..a233d768d5 Binary files /dev/null and b/docs/homepage-screenshot.png differ diff --git a/docs/userpage-screenshot.png b/docs/userpage-screenshot.png new file mode 100644 index 0000000000..f501ce8ac9 Binary files /dev/null and b/docs/userpage-screenshot.png differ diff --git a/lib/database_connection.rb b/lib/database_connection.rb new file mode 100644 index 0000000000..bce8a207c2 --- /dev/null +++ b/lib/database_connection.rb @@ -0,0 +1,48 @@ +# file: lib/database_connection.rb + +require 'pg' + +# This class is a thin "wrapper" around the +# PG library. We'll use it in our project to interact +# with the database using SQL. + +class DatabaseConnection + # This method connects to PostgreSQL using the + # PG gem. We connect to 127.0.0.1, and select + # the database name given in argument. + + + def self.connect + # If the environment variable (set by Render) + # is present, use this to open the connection. + if ENV['DATABASE_URL'] != nil + @connection = PG.connect(ENV['DATABASE_URL']) + return + end + + if ENV['ENV'] == 'test' + database_name = 'chitter_test' + else + database_name = 'chitter' + end + @connection = PG.connect({ host: '127.0.0.1', dbname: database_name }) + end + + # def self.connect(database_name) + # @connection = PG.connect({ host: '127.0.0.1', dbname: database_name }) + # end + + + + # This method executes an SQL query + # on the database, providing some optional parameters + # (you will learn a bit later about when to provide these parameters). + def self.exec_params(query, params) + if @connection.nil? + raise 'DatabaseConnection.exec_params: Cannot run a SQL query as the connection to'\ + 'the database was never opened. Did you make sure to call first the method '\ + '`DatabaseConnection.connect` in your app.rb file (or in your tests spec_helper.rb)?' + end + @connection.exec_params(query, params) + end +end \ No newline at end of file diff --git a/lib/peep.rb b/lib/peep.rb new file mode 100644 index 0000000000..3192c84be1 --- /dev/null +++ b/lib/peep.rb @@ -0,0 +1,3 @@ +class Peep + attr_accessor :id, :content, :time, :user_id +end \ No newline at end of file diff --git a/lib/peep_repository.rb b/lib/peep_repository.rb new file mode 100644 index 0000000000..341ccaf827 --- /dev/null +++ b/lib/peep_repository.rb @@ -0,0 +1,78 @@ +require_relative 'database_connection' +require_relative 'peep' + + +class PeepRepository + + def all + # Executes the SQL query: + # SELECT id, content, time, user_id FROM peeps; + + sql = 'SELECT id, content, time, user_id FROM peeps;' + sql_params = [] + + result_set = DatabaseConnection.exec_params(sql, sql_params) + + # Returns an array of Peep objects. + return get_peeps(result_set) + + end + + def create(peep) + # Executes the SQL query: + # INSERT INTO peeps (content, time, user_id) VALUES ($1, $2, $3); + + sql = 'INSERT INTO peeps (content, time, user_id) VALUES ($1, $2, $3);' + sql_params = [peep.content, peep.time, peep.user_id] + + DatabaseConnection.exec_params(sql, sql_params) + + # returns nil + return nil + end + + def find_by_owner(user_id) + # Executes the SQL query: + # SELECT id, content, time, user_id FROM peeps WHERE user_id = $1; + + sql = 'SELECT id, content, time, user_id FROM peeps WHERE user_id = $1;' + sql_params = [user_id] + + result_set = DatabaseConnection.exec_params(sql, sql_params) + + # Returns an array of Peep objects. + return get_peeps(result_set) + end + + def find_by_tagged_user(user_id) + # Executes the SQL query: + # SELECT peeps.id, peeps.content, peeps.time, peeps.user_id FROM peeps JOIN peeps_users ON peeps.id = peeps_users.peep_id JOIN users ON users.id = peeps_users.user_id WHERE users.id = $1; + + sql = 'SELECT peeps.id, peeps.content, peeps.time, peeps.user_id FROM peeps JOIN peeps_users ON peeps.id = peeps_users.peep_id JOIN users ON users.id = peeps_users.user_id WHERE users.id = $1;' + sql_params = [user_id] + + result_set = DatabaseConnection.exec_params(sql, sql_params) + + # Returns an array of Peep objects. + return get_peeps(result_set) + end + + private + + def get_peeps(result_set) + peeps = [] + + result_set.each { |record| + peep = Peep.new + peep.id = record['id'].to_i + peep.content = record['content'] + peep.time = record['time'] + peep.user_id = record['user_id'].to_i + + peeps << peep + } + + return peeps + end + +end \ No newline at end of file diff --git a/lib/user.rb b/lib/user.rb new file mode 100644 index 0000000000..abd481f664 --- /dev/null +++ b/lib/user.rb @@ -0,0 +1,3 @@ +class User + attr_accessor :id, :name, :email, :username, :password +end \ No newline at end of file diff --git a/lib/user_repository.rb b/lib/user_repository.rb new file mode 100644 index 0000000000..1e932f6d89 --- /dev/null +++ b/lib/user_repository.rb @@ -0,0 +1,99 @@ +require_relative 'user' +require_relative 'database_connection' + + +class UserRepository + + def all + # Executes the SQL query: + sql = 'SELECT id, name, email, username, password FROM users;' + sql_params = [] + + users = [] + result_set = DatabaseConnection.exec_params(sql,sql_params) + + result_set.each {|record| + user = User.new + user.id = record['id'].to_i + user.name = record['name'] + user.email = record['email'] + user.username = record['username'] + user.password = record['password'] + users << user + } + + # Returns an array of User objects. + return users + end + + def find_by_username(username) + # Executes the SQL query: + sql = 'SELECT id, name, email, username, password FROM users WHERE username = $1;' + sql_params = [username] + + record = DatabaseConnection.exec_params(sql,sql_params)[0] + + user = User.new + user.id = record['id'].to_i + user.name = record['name'] + user.email = record['email'] + user.username = record['username'] + user.password = record['password'] + + + # Returns a User object. + return user + end + + def find(user_id) + # Executes the SQL query: + sql = 'SELECT id, name, email, username, password FROM users WHERE id = $1;' + sql_params = [user_id] + + record = DatabaseConnection.exec_params(sql,sql_params)[0] + + user = User.new + user.id = record['id'].to_i + user.name = record['name'] + user.email = record['email'] + user.username = record['username'] + user.password = record['password'] + + + # Returns a User object. + return user + end + + def create(user) + # Executes the SQL query: + sql = 'INSERT INTO users (name, email, username, password) VALUES ($1, $2, $3, $4);' + sql_params = [user.name, user.email, user.username, user.password] + + DatabaseConnection.exec_params(sql,sql_params) + + return nil + end + + def get_tagged_users(peep_id) + # Executes the SQL query: + sql = 'SELECT users.id, users.name, users.email, users.username, users.password FROM users JOIN peeps_users ON users.id = peeps_users.user_id JOIN peeps ON peeps.id = peeps_users.peep_id WHERE peeps.id = $1;' + sql_params = [peep_id] + + users = [] + result_set = DatabaseConnection.exec_params(sql,sql_params) + + result_set.each {|record| + user = User.new + user.id = record['id'].to_i + user.name = record['name'] + user.email = record['email'] + user.username = record['username'] + user.password = record['password'] + + users << user + } + + # Returns an array of User objects. + return users + end +end \ No newline at end of file diff --git a/peeps_users.sql b/peeps_users.sql new file mode 100644 index 0000000000..5dacfd8fb8 --- /dev/null +++ b/peeps_users.sql @@ -0,0 +1,31 @@ +-- file: peeps_users.sql + +DROP TABLE peeps_users; +DROP TABLE peeps; +DROP TABLE users; + +CREATE TABLE users ( + id SERIAL PRIMARY KEY, + name text, + email text, + username text, + password text +); + +CREATE TABLE peeps ( + id SERIAL PRIMARY KEY, + content text, + time timestamp, + user_id int, + constraint fk_user foreign key(user_id) references users(id) on delete cascade +); + + +-- Create the join table. +CREATE TABLE peeps_users ( + peep_id int, + user_id int, + constraint fk_peeps foreign key(peep_id) references peeps(id) on delete cascade, + constraint fk_users foreign key(user_id) references users(id) on delete cascade, + PRIMARY KEY (peep_id, user_id) +); \ No newline at end of file diff --git a/project_spec.md b/project_spec.md new file mode 100644 index 0000000000..465eda879b --- /dev/null +++ b/project_spec.md @@ -0,0 +1,123 @@ +Chitter Challenge +================= + +* Feel free to use Google, your notes, books, etc. but work on your own +* If you refer to the solution of another coach or student, please put a link to that in your README +* If you have a partial solution, **still check in a partial solution** +* You must submit a pull request to this repo with your code by 10am Monday morning + +Challenge: +------- + +As usual please start by forking this repo. + +We are going to write a small Twitter clone that will allow the users to post messages to a public stream. + +Features: +------- + +``` +STRAIGHT UP + +As a Maker +So that I can let people know what I am doing +I want to post a message (peep) to chitter + +As a maker +So that I can see what others are saying +I want to see all peeps in reverse chronological order + +As a Maker +So that I can better appreciate the context of a peep +I want to see the time at which it was made + +As a Maker +So that I can post messages on Chitter as me +I want to sign up for Chitter + +HARDER + +As a Maker +So that only I can post messages on Chitter as me +I want to log in to Chitter + +As a Maker +So that I can avoid others posting messages on Chitter as me +I want to log out of Chitter + +ADVANCED + +As a Maker +So that I can stay constantly tapped in to the shouty box of Chitter +I want to receive an email if I am tagged in a Peep +``` + +Technical Approach: +----- + +In the last two weeks, you integrated a database using the `pg` gem and Repository classes. You also implemented small web applications using Sinatra, RSpec, HTML and ERB views to make dynamic webpages. You can continue to use this approach when building Chitter Challenge. + +You can refer to the [guidance on Modelling and Planning a web application](https://github.com/makersacademy/web-applications/blob/main/pills/modelling_and_planning_web_application.md), to help you in planning the different web pages you will need to implement this challenge. If you'd like to deploy your app to Heroku so other people can use it, [you can follow this guidance](https://github.com/makersacademy/web-applications/blob/main/html_challenges/07_deploying.md). + +If you'd like more technical challenge now, try using an [Object Relational Mapper](https://en.wikipedia.org/wiki/Object-relational_mapping) as the database interface, instead of implementing your own Repository classes. + +Some useful resources: +**Ruby Object Mapper** +- [ROM](https://rom-rb.org/) + +**ActiveRecord** +- [ActiveRecord ORM](https://guides.rubyonrails.org/active_record_basics.html) +- [Sinatra & ActiveRecord setup](https://learn.co/lessons/sinatra-activerecord-setup) + +Notes on functionality: +------ + +* You don't have to be logged in to see the peeps. +* Makers sign up to chitter with their email, password, name and a username (e.g. samm@makersacademy.com, password123, Sam Morgan, sjmog). +* The username and email are unique. +* Peeps (posts to chitter) have the name of the maker and their user handle. +* Your README should indicate the technologies used, and give instructions on how to install and run the tests. + +Bonus: +----- + +If you have time you can implement the following: + +* In order to start a conversation as a maker I want to reply to a peep from another maker. + +And/Or: + +* Work on the CSS to make it look good. + +Good luck and let the chitter begin! + +Code Review +----------- + +In code review we'll be hoping to see: + +* All tests passing +* High [Test coverage](https://github.com/makersacademy/course/blob/main/pills/test_coverage.md) (>95% is good) +* The code is elegant: every class has a clear responsibility, methods are short etc. + +Reviewers will potentially be using this [code review rubric](docs/review.md). Referring to this rubric in advance may make the challenge somewhat easier. You should be the judge of how much challenge you want at this moment. + +Notes on test coverage +---------------------- + +Please ensure you have the following **AT THE TOP** of your spec_helper.rb in order to have test coverage stats generated +on your pull request: + +```ruby +require 'simplecov' +require 'simplecov-console' + +SimpleCov.formatter = SimpleCov::Formatter::MultiFormatter.new([ + SimpleCov::Formatter::Console, + # Want a nice code coverage website? Uncomment this next line! + # SimpleCov::Formatter::HTMLFormatter +]) +SimpleCov.start +``` + +You can see your test coverage when you run your tests. If you want this in a graphical form, uncomment the `HTMLFormatter` line and see what happens! diff --git a/public/chit_chat.png b/public/chit_chat.png new file mode 100644 index 0000000000..03d14916ed Binary files /dev/null and b/public/chit_chat.png differ diff --git a/public/styles.css b/public/styles.css new file mode 100644 index 0000000000..34109bc5c6 --- /dev/null +++ b/public/styles.css @@ -0,0 +1,90 @@ +/* styles.css */ + + +body { + font-family: Space Mono; + margin: 0; +} + +h1, +h2, +h3 { + color: #404042; + font-family: Space Mono; +} + +p { + color: #404042; + opacity: .7; +} + +.form { + border-style: solid; + padding: 10px; +} + +.container { + width: 800px; + margin: auto; +} + +.peep { + width: 600px; + margin: 20px auto; + border-style: solid; + padding: 5px; +} + +.banner { + background-image: linear-gradient(rgba(249, 153, 196, 0.4), rgba(246, 193, 233, 0.4)), url('/chit_chat.png'); + padding: 100px 0px 30px; + background-size: cover; + background-position: center; + text-align: center; +} + +.userbanner { + padding: 10px 0; + text-align: center; +} + +.banner h1 { + color: white; + font-size: 45px; + margin: 0; +} + +.banner h3 { + color: white; +} + +.banner p { + color: white; + font-size: 24px; +} + +.button-blue { + background-color: #1A48E3; + color: white; + padding: 16px 48px; + border-radius: 4px; + display: inline-block; + text-decoration: none; + transition: .3s; +} + +.button-blue:hover { + background-color: #0D1EC6; + color: white; +} + + +.footer { + background: #404042; + text-align: center; + padding: 24px 0; +} + +.footer p { + color: white; +} \ No newline at end of file diff --git a/spec/integration/application_spec.rb b/spec/integration/application_spec.rb new file mode 100644 index 0000000000..4f8b675f20 --- /dev/null +++ b/spec/integration/application_spec.rb @@ -0,0 +1,233 @@ +require "spec_helper" +require "rack/test" +require_relative '../../app' + + +def reset_test_database_tables + seed_sql = File.read('spec/seeds_chitter.sql') + connection = PG.connect({ host: '127.0.0.1', dbname: 'chitter_test' }) + connection.exec(seed_sql) +end + +describe Application do + # This is so we can use rack-test helper methods. + include Rack::Test::Methods + + # We need to declare the `app` value by instantiating the Application + # class so our tests work. + let(:app) { Application.new } + + before(:each) do + reset_test_database_tables + end + + # ------------- Homepage Route ---------------------------------------- + + context 'Homepage Route: ' do + it 'shows the two latest peeps' do + response = get('/') + expect(response.status).to eq 200 + + expect(response.body).to include '

This is the forth Peep

' + expect(response.body).to include '

This is the third Peep

' + + end + end + + # ------------- Sign Up Routes ---------------------------------------- + + context 'Sign Up Routes: ' do + it 'displays a form with entries for name, email, username, password' do + response = get('/signup') + expect(response.status).to eq 200 + + expect(response.body).to include '
' + end + + it 'rejects a signup if an email address is already in the database' do + response = post('/signup', + name:'Test Name', + email:'carolinesemail@email.com', + username: 'testusername', + password: 'test') + expect(response.status).to eq 400 + + expect(response.body).to include 'Email address already signed up.' + end + + it 'rejects a signup if a username is already in the database' do + response = post('/signup', + name:'Test Name', + email:'testemail@email.com', + username: 'caro', + password: 'test') + expect(response.status).to eq 400 + + expect(response.body).to include 'Username already taken, please choose another.' + end + + it 'rejects a signup if the name includes special characters' do + response = post('/signup', + name:'TestName!', + email:'testemail@email.com', + username: 'testusername', + password: 'test') + expect(response.status).to eq 400 + + expect(response.body).to include 'Invalid credentials, please try again.' + end + + it 'rejects a signup if the username includes special characters' do + response = post('/signup', + name:'TestName', + email:'testemail@email.com', + username: 'testusername!', + password: 'test') + expect(response.status).to eq 400 + + expect(response.body).to include 'Invalid credentials, please try again.' + end + + it 'rejects a signup if the password is empty' do + response = post('/signup', + name:'TestName', + email:'testemail@email.com', + username: 'testusername!', + password: '') + expect(response.status).to eq 400 + + expect(response.body).to include 'Invalid credentials, please try again.' + end + + it 'rejects a signup if the email does not have an @ sign' do + response = post('/signup', + name:'TestName', + email:'testemailemail.com', + username: 'testusername', + password: 'test') + expect(response.status).to eq 400 + + expect(response.body).to include 'Invalid credentials, please try again.' + end + + it 'successfully adds a new user' do + response = post('/signup', + name:'TestName', + email:'testemail@email.com', + username: 'testusername1', + password: 'test') + + expect(response.status).to eq 302 + expect(response.original_headers['Location']).to include '/userpage' + + users = UserRepository.new.all + + expect(users[2].id).to eq 3 + expect(users[2].name).to eq 'TestName' + expect(users[2].email).to eq 'testemail@email.com' + expect(users[2].username).to eq 'testusername1' + expect(BCrypt::Password.new(users[2].password)).to eq 'test' + + end + + end + + # ------------- Peeps Routes ---------------------------------------- + + context 'Peeps Routes' do + it 'displays all peeps' do + response = get('/peeps') + expect(response.status).to eq 200 + + expect(response.body).to include('

This is the forth Peep

') + expect(response.body).to include('

This is the third Peep

') + expect(response.body).to include('

This is the second Peep

') + expect(response.body).to include('

This is the first Peep

') + + end + + it 'rejects a peep post if content is empty' do + response = post('/peeps', + content:'', + user_id: 1) + expect(response.status).to eq 400 + + expect(response.body).to include 'Invalid peep, please try again.' + + end + + it 'creates a new peep' do + response = post('/peeps', + content:'Test Peep 1', + user_id: 1) + + expect(response.status).to eq 200 + + peeps = PeepRepository.new.all + + expect(peeps[4].id).to eq 5 + expect(peeps[4].content).to eq 'Test Peep 1' + expect(peeps[4].user_id).to eq 1 + + end + + end + + + # ------------- User Page Routes ---------------------------------------- + + context 'User Page Routes: ' do + it 'rejects a log-in if password is empty' do + response = post('/userpage', + username:'caro', + password: '') + expect(response.status).to eq 400 + + expect(response.body).to include 'Invalid user details.' + + end + + it 'rejects a log-in if password is wrong' do + response = post('/signup', + name:'TestName', + email:'testemail@email.com', + username: 'testusername1', + password: 'test') + + response = post('/userpage', + username:'testusername1', + password: 'testing') + expect(response.status).to eq 400 + + expect(response.body).to include 'Invalid user details.' + + end + + it 'rejects a log-in if username is not in the database' do + response = post('/userpage', + username:'test', + password: 'testing') + expect(response.status).to eq 400 + + expect(response.body).to include 'Username does not exist, please try again.' + + end + + it 'redirects the user on a successful log-in' do + response = post('/signup', + name:'TestName', + email:'testemail@email.com', + username: 'testusername1', + password: 'test') + + response = post('/userpage', + username:'testusername1', + password: 'test') + expect(response.status).to eq 302 + end + + + end + + +end \ No newline at end of file diff --git a/spec/peep_repository_spec.rb b/spec/peep_repository_spec.rb new file mode 100644 index 0000000000..e098ef15b2 --- /dev/null +++ b/spec/peep_repository_spec.rb @@ -0,0 +1,102 @@ +require 'peep_repository' + +def reset_chitter_tables + seed_sql = File.read('spec/seeds_chitter.sql') + connection = PG.connect({ host: '127.0.0.1', dbname: 'chitter_test' }) + connection.exec(seed_sql) +end + +describe PeepRepository do + before(:each) do + reset_chitter_tables + end + + it 'gets all peeps' do + repo = PeepRepository.new + + peeps = repo.all + + expect(peeps.length).to eq 4 + + expect(peeps[0].id).to eq 1 + expect(peeps[0].content).to eq 'This is the first Peep' + expect(peeps[0].time).to eq '2023-05-06 10:22:09' + expect(peeps[0].user_id).to eq 1 + + expect(peeps[1].id).to eq 2 + expect(peeps[1].content).to eq 'This is the second Peep' + expect(peeps[1].time).to eq '2023-05-07 15:35:35' + expect(peeps[1].user_id).to eq 1 + + expect(peeps[2].id).to eq 3 + expect(peeps[2].content).to eq 'This is the third Peep' + expect(peeps[2].time).to eq '2023-05-08 09:42:01' + expect(peeps[2].user_id).to eq 2 + + expect(peeps[3].id).to eq 4 + expect(peeps[3].content).to eq 'This is the forth Peep' + expect(peeps[3].time).to eq '2023-05-09 23:12:59' + expect(peeps[3].user_id).to eq 2 + + end + + it 'creates a new peep' do + peep = Peep.new + peep.content = 'This is the fifth Peep' + peep.time = '20230422 08:22:46 PM' + peep.user_id = 2 + + repo = PeepRepository.new + + repo.create(peep) + + last_peep = repo.all.last + + expect(last_peep.id).to eq 5 + expect(last_peep.content).to eq 'This is the fifth Peep' + expect(last_peep.time).to eq '2023-04-22 20:22:46' + expect(last_peep.user_id).to eq 2 + end + + it 'finds all peeps made by the same user' do + repo = PeepRepository.new + + peeps = repo.find_by_owner(1) + + expect(peeps.length).to eq 2 + + expect(peeps[0].id).to eq 1 + expect(peeps[0].content).to eq 'This is the first Peep' + expect(peeps[0].time).to eq '2023-05-06 10:22:09' + expect(peeps[0].user_id).to eq 1 + + expect(peeps[1].id).to eq 2 + expect(peeps[1].content).to eq 'This is the second Peep' + expect(peeps[1].time).to eq '2023-05-07 15:35:35' + expect(peeps[1].user_id).to eq 1 + end + + it 'finds all peeps that a given user has been tagged in' do + repo = PeepRepository.new + + peeps = repo.find_by_tagged_user(1) + + expect(peeps.length).to eq 0 + + peeps = repo.find_by_tagged_user(2) + + expect(peeps.length).to eq 2 + + expect(peeps[0].id).to eq 1 + expect(peeps[0].content).to eq 'This is the first Peep' + expect(peeps[0].time).to eq '2023-05-06 10:22:09' + expect(peeps[0].user_id).to eq 1 + + expect(peeps[1].id).to eq 2 + expect(peeps[1].content).to eq 'This is the second Peep' + expect(peeps[1].time).to eq '2023-05-07 15:35:35' + expect(peeps[1].user_id).to eq 1 + end + + +end \ No newline at end of file diff --git a/spec/seeds_chitter.sql b/spec/seeds_chitter.sql new file mode 100644 index 0000000000..1079767ee0 --- /dev/null +++ b/spec/seeds_chitter.sql @@ -0,0 +1,23 @@ +-- (file: spec/seeds_chitter.sql) + +-- First, you'd need to truncate the table - this is so our table is emptied between each test run, +-- so we can start with a fresh state. +-- (RESTART IDENTITY resets the primary key) + +TRUNCATE TABLE peeps, users, peeps_users RESTART IDENTITY; + +-- Below this line there should only be `INSERT` statements. +-- Replace these statements with your own seed data. + +INSERT INTO users (name, email, username, password) VALUES ('Caroline', 'carolinesemail@email.com', 'caro', '$2a$12$RWbFtjHnA3kC2Gt31m7/l.N4f8ISipDp9T7KIyvSHhww/sGqohGHS'); +INSERT INTO users (name, email, username, password) VALUES ('Philip', 'philsemail@email.com', 'phil', '$2a$12$BV.yJVJm0fDOX0ikKe/6V.DjV5T4KqfMzWUh5B0Yd.QDIOGdyLNIK'); + +-- the two passwords are 'pwtest1' and 'pwtest2' + +INSERT INTO peeps (content, time, user_id) VALUES ('This is the first Peep', '20230506 10:22:09 AM', 1); +INSERT INTO peeps (content, time, user_id) VALUES ('This is the second Peep', '20230507 03:35:35 PM', 1); +INSERT INTO peeps (content, time, user_id) VALUES ('This is the third Peep', '20230508 09:42:01 AM', 2); +INSERT INTO peeps (content, time, user_id) VALUES ('This is the forth Peep', '20230509 11:12:59 PM', 2); + +INSERT INTO peeps_users (peep_id, user_id) VALUES (1, 2); +INSERT INTO peeps_users (peep_id, user_id) VALUES (2, 2); \ No newline at end of file diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index 252747d899..4de41563fc 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -15,3 +15,10 @@ puts "\e[33mTry it now! Just run: rubocop\e[0m" end end + + +require 'database_connection' + +ENV['ENV'] = 'test' + +DatabaseConnection.connect diff --git a/spec/user_repository_spec.rb b/spec/user_repository_spec.rb new file mode 100644 index 0000000000..82583a3dc2 --- /dev/null +++ b/spec/user_repository_spec.rb @@ -0,0 +1,92 @@ +require "user_repository" + +def reset_chitter_tables + seed_sql = File.read('spec/seeds_chitter.sql') + connection = PG.connect({ host: '127.0.0.1', dbname: 'chitter_test' }) + connection.exec(seed_sql) +end + +describe UserRepository do + before(:each) do + reset_chitter_tables + end + + it 'gets all users' do + repo = UserRepository.new + + users = repo.all + + expect(users.length).to eq 2 + + expect(users[0].id).to eq 1 + expect(users[0].name).to eq 'Caroline' + expect(users[0].email).to eq 'carolinesemail@email.com' + expect(users[0].username).to eq 'caro' + expect(BCrypt::Password.new(users[0].password)).to eq 'pwtest1' + + + expect(users[1].id).to eq 2 + expect(users[1].name).to eq 'Philip' + expect(users[1].email).to eq 'philsemail@email.com' + expect(users[1].username).to eq 'phil' + expect(BCrypt::Password.new(users[1].password)).to eq 'pwtest2' + + + end + + it 'finds a user by username' do + repo = UserRepository.new + + user = repo.find_by_username('phil') + + expect(user.id).to eq 2 + expect(user.name).to eq 'Philip' + expect(user.email).to eq 'philsemail@email.com' + expect(user.username).to eq 'phil' + end + + it 'finds a user by user ID' do + repo = UserRepository.new + + user = repo.find(2) + + expect(user.id).to eq 2 + expect(user.name).to eq 'Philip' + expect(user.email).to eq 'philsemail@email.com' + expect(user.username).to eq 'phil' + end + + it 'creates a new user' do + user = User.new + user.name = 'Pip' + user.email = 'pipsemail@email.com' + user.username = 'pip1' + + repo = UserRepository.new + + repo.create(user) + + last_user = repo.find_by_username('pip1') + + expect(last_user.id).to eq 3 + expect(last_user.name).to eq 'Pip' + expect(last_user.email).to eq 'pipsemail@email.com' + expect(last_user.username).to eq 'pip1' + + end + + it 'finds all tagged users for a given peep' do + repo = UserRepository.new + + users = repo.get_tagged_users(1) + + expect(users.length).to eq 1 + + expect(users[0].id).to eq 2 + expect(users[0].name).to eq 'Philip' + expect(users[0].email).to eq 'philsemail@email.com' + expect(users[0].username).to eq 'phil' + + end + +end \ No newline at end of file diff --git a/views/index.erb b/views/index.erb new file mode 100644 index 0000000000..2e5e0c8bd9 --- /dev/null +++ b/views/index.erb @@ -0,0 +1,87 @@ + + + + + + Chitter - The ChitChat App + + + + + + + + + + + + + +
+

Log in to Chitter

+
+ +
+ + +
+
+ + +
+ + + +
+ +
+ + + +
+

Latest Peeps

+
+

<%= @user_repo.find(@peeps[0].user_id).name%> (@<%= @user_repo.find(@peeps[0].user_id).username%>):

+

<%= @peeps[0].content %>

+

<%= @peeps[0].time %>

+ <% if @user_repo.get_tagged_users(@peeps[0].id).length > 0 then %> +

tagged users: <% @user_repo.get_tagged_users(@peeps[0].id).each do |user| %> + @<%= user.username %> + <% end %> +

+ <% end %> +
+
+

<%= @user_repo.find(@peeps[1].user_id).name%> (@<%= @user_repo.find(@peeps[1].user_id).username%>):

+

<%= @peeps[1].content %>

+

<%= @peeps[1].time %>

+ <% if @user_repo.get_tagged_users(@peeps[1].id).length > 0 then %> +

tagged users: <% @user_repo.get_tagged_users(@peeps[1].id).each do |user| %> + @<%= user.username %> + <% end %> +

+ <% end %> +
+ +
+ +
+

View all peeps

+
+ + + + + + + + + + \ No newline at end of file diff --git a/views/peeps.erb b/views/peeps.erb new file mode 100644 index 0000000000..abb6bcb87f --- /dev/null +++ b/views/peeps.erb @@ -0,0 +1,47 @@ + + + + + + Chitter - Peeps + + + + + + + + + + +
+

All Peeps

+ <% @peeps.each do |peep| %> +
+

<%=@user_repo.find(peep.user_id).name%> (@<%= @user_repo.find(peep.user_id).username%>):

+

<%= peep.content %>

+

<%= peep.time %>

+ <% if @user_repo.get_tagged_users(peep.id).length > 0 then %> +

tagged users: <% @user_repo.get_tagged_users(peep.id).each do |user| %> + @<%= user.username %> + <% end %> +

+ <% end %> +
+ <% end %> +
+ + + + + + + + + + \ No newline at end of file diff --git a/views/signup.erb b/views/signup.erb new file mode 100644 index 0000000000..4a985d8f81 --- /dev/null +++ b/views/signup.erb @@ -0,0 +1,56 @@ + + + + + + Chitter - Sign Up + + + + + + + + + + +
+

Sign up to Chitter!

+
+
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+ + +
+
+ +
+ + + + + + + + + \ No newline at end of file diff --git a/views/userpage.erb b/views/userpage.erb new file mode 100644 index 0000000000..69ddee6182 --- /dev/null +++ b/views/userpage.erb @@ -0,0 +1,100 @@ + + + + + + Chitter - User Homepage + + + + + + + + + + +
+

Hello, <%=@user.name%>!

+

Welcome to your chitter homepage

+
+ +
+ +
+ + + +
+

Username: <%=@user.username%>

+

Email: <%=@user.email%>

+
+ + +
+

Post a new Peep

+
+
+
+ + +
+ + > + + +
+
+
+ + + +
+

<%=@user.name%>'s Peeps

+ <% @peeps.each do |peep| %> +
+

<%= peep.content %>

+

<%= peep.time %>

+
+ <% end %> +
+ + +
+

Tagged Peeps

+ <% if @tagged_peeps.length == 0 then %> +

You haven't been tagged in any peeps yet!

+ <% end %> + <% @tagged_peeps.each do |peep| %> +
+

<%= @user_repo.find(peep.user_id).username %>:

+

<%= peep.content %>

+

<%= peep.time %>

+
+ <% end %> +
+ + +
+

View all Peeps

+ +
+ + + + + + + + + + + \ No newline at end of file