diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..31eeee0 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,7 @@ +# See https://git-scm.com/docs/gitattributes for more about git attribute files. + +# Mark the database schema as having been generated. +db/schema.rb linguist-generated + +# Mark any vendored files as having been vendored. +vendor/* linguist-vendored diff --git a/.github/workflows/linters.yml b/.github/workflows/linters.yml new file mode 100644 index 0000000..595f85e --- /dev/null +++ b/.github/workflows/linters.yml @@ -0,0 +1,30 @@ +name: Linters + +on: pull_request + +env: + FORCE_COLOR: 1 + +jobs: + rubocop: + name: Rubocop + runs-on: ubuntu-22.04 + steps: + - uses: actions/checkout@v2 + - uses: actions/setup-ruby@v1 + with: + ruby-version: 3.1.x + - name: Setup Rubocop + run: | + gem install --no-document rubocop -v '>= 1.0, < 2.0' # https://docs.rubocop.org/en/stable/installation/ + [ -f .rubocop.yml ] || wget https://raw.githubusercontent.com/microverseinc/linters-config/master/ror/.rubocop.yml + - name: Rubocop Report + run: rubocop --color + nodechecker: + name: node_modules checker + runs-on: ubuntu-22.04 + steps: + - uses: actions/checkout@v2 + - name: Check node_modules existence + run: | + if [ -d "node_modules/" ]; then echo -e "\e[1;31mThe node_modules/ folder was pushed to the repo. Please remove it from the GitHub repository and try again."; echo -e "\e[1;32mYou can set up a .gitignore file with this folder included on it to prevent this from happening in the future." && exit 1; fi diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..c4c0f86 --- /dev/null +++ b/.gitignore @@ -0,0 +1,33 @@ +# See https://help.github.com/articles/ignoring-files for more about ignoring files. +# +# If you find yourself ignoring temporary files generated by your text editor +# or operating system, you probably want to add a global ignore instead: +# git config --global core.excludesfile '~/.gitignore_global' + +# Ignore bundler config. +/.bundle + +# Ignore all logfiles and tempfiles. +/log/* +/tmp/* +!/log/.keep +!/tmp/.keep + +# Ignore pidfiles, but keep the directory. +/tmp/pids/* +!/tmp/pids/ +!/tmp/pids/.keep + +# Ignore uploaded files in development. +/storage/* +!/storage/.keep +/tmp/storage/* +!/tmp/storage/ +!/tmp/storage/.keep + +/public/assets + +# Ignore master key for decrypting credentials and more. +/config/master.key + +/config/database.yml diff --git a/.rspec b/.rspec new file mode 100644 index 0000000..c99d2e7 --- /dev/null +++ b/.rspec @@ -0,0 +1 @@ +--require spec_helper diff --git a/.rubocop.yml b/.rubocop.yml new file mode 100644 index 0000000..07baeb4 --- /dev/null +++ b/.rubocop.yml @@ -0,0 +1,60 @@ +AllCops: + NewCops: enable + Exclude: + - "db/**/*" + - "bin/*" + - "config/**/*" + - "Guardfile" + - "Rakefile" + - "node_modules/**/*" + + DisplayCopNames: true + +Layout/LineLength: + Max: 120 +Metrics/MethodLength: + Include: + - "app/controllers/*" + - "app/models/*" + Max: 20 +Metrics/AbcSize: + Include: + - "app/controllers/*" + - "app/models/*" + Max: 50 +Metrics/ClassLength: + Max: 150 +Metrics/BlockLength: + AllowedMethods: ['describe'] + Max: 30 + +Style/Documentation: + Enabled: false +Style/ClassAndModuleChildren: + Enabled: false +Style/EachForSimpleLoop: + Enabled: false +Style/AndOr: + Enabled: false +Style/DefWithParentheses: + Enabled: false +Style/FrozenStringLiteralComment: + EnforcedStyle: never + +Layout/HashAlignment: + EnforcedColonStyle: key +Layout/ExtraSpacing: + AllowForAlignment: false +Layout/MultilineMethodCallIndentation: + Enabled: true + EnforcedStyle: indented +Lint/RaiseException: + Enabled: false +Lint/StructNewOverride: + Enabled: false +Style/HashEachMethods: + Enabled: false +Style/HashTransformKeys: + Enabled: false +Style/HashTransformValues: + Enabled: false \ No newline at end of file diff --git a/.ruby-version b/.ruby-version new file mode 100644 index 0000000..944880f --- /dev/null +++ b/.ruby-version @@ -0,0 +1 @@ +3.2.0 diff --git a/Gemfile b/Gemfile new file mode 100644 index 0000000..1cdc1f1 --- /dev/null +++ b/Gemfile @@ -0,0 +1,80 @@ +source 'https://rubygems.org' +git_source(:github) { |repo| "https://github.com/#{repo}.git" } + +ruby '3.1.3' + +gem 'bcrypt', '~> 3.1.7' +gem 'jwt', '~> 2.2.3' +gem 'rack-cors' + +gem 'rubocop', '>= 1.0', '< 2.0' + +# Bundle edge Rails instead: gem "rails", github: "rails/rails", branch: "main" +gem 'rails', '~> 7.0.4', '>= 7.0.4.3' + +# The original asset pipeline for Rails [https://github.com/rails/sprockets-rails] +gem 'sprockets-rails' + +# Use postgresql as the database for Active Record +gem 'pg', '~> 1.1' + +# Use the Puma web server [https://github.com/puma/puma] +gem 'puma', '~> 5.0' + +# Use JavaScript with ESM import maps [https://github.com/rails/importmap-rails] +gem 'importmap-rails' + +# Hotwire's SPA-like page accelerator [https://turbo.hotwired.dev] +gem 'turbo-rails' + +# Hotwire's modest JavaScript framework [https://stimulus.hotwired.dev] +gem 'stimulus-rails' + +# Build JSON APIs with ease [https://github.com/rails/jbuilder] +gem 'jbuilder' + +# Use Redis adapter to run Action Cable in production +# gem "redis", "~> 4.0" + +# Use Kredis to get higher-level data types in Redis [https://github.com/rails/kredis] +# gem "kredis" + +# Use Active Model has_secure_password [https://guides.rubyonrails.org/active_model_basics.html#securepassword] +# gem "bcrypt", "~> 3.1.7" + +# Windows does not include zoneinfo files, so bundle the tzinfo-data gem +gem 'tzinfo-data', platforms: %i[mingw mswin x64_mingw jruby] + +# Reduces boot times through caching; required in config/boot.rb +gem 'bootsnap', require: false + +# Use Sass to process CSS +# gem "sassc-rails" + +# Use Active Storage variants [https://guides.rubyonrails.org/active_storage_overview.html#transforming-images] +# gem "image_processing", "~> 1.2" + +group :development, :test do + # See https://guides.rubyonrails.org/debugging_rails_applications.html#debugging-with-the-debug-gem + gem 'debug', platforms: %i[mri mingw x64_mingw] + gem 'rspec-rails' +end + +group :development do + # Use console on exceptions pages [https://github.com/rails/web-console] + gem 'web-console' + + # Add speed badges [https://github.com/MiniProfiler/rack-mini-profiler] + # gem "rack-mini-profiler" + + # Speed up commands on slow machines / big apps [https://github.com/rails/spring] + # gem "spring" +end + +group :test do + # Use system testing [https://guides.rubyonrails.org/testing.html#system-testing] + gem 'capybara' + gem 'rails-controller-testing' + gem 'selenium-webdriver' + gem 'webdrivers' +end diff --git a/Gemfile.lock b/Gemfile.lock new file mode 100644 index 0000000..7841c1b --- /dev/null +++ b/Gemfile.lock @@ -0,0 +1,297 @@ +GEM + remote: https://rubygems.org/ + specs: + actioncable (7.0.4.3) + actionpack (= 7.0.4.3) + activesupport (= 7.0.4.3) + nio4r (~> 2.0) + websocket-driver (>= 0.6.1) + actionmailbox (7.0.4.3) + actionpack (= 7.0.4.3) + activejob (= 7.0.4.3) + activerecord (= 7.0.4.3) + activestorage (= 7.0.4.3) + activesupport (= 7.0.4.3) + mail (>= 2.7.1) + net-imap + net-pop + net-smtp + actionmailer (7.0.4.3) + actionpack (= 7.0.4.3) + actionview (= 7.0.4.3) + activejob (= 7.0.4.3) + activesupport (= 7.0.4.3) + mail (~> 2.5, >= 2.5.4) + net-imap + net-pop + net-smtp + rails-dom-testing (~> 2.0) + actionpack (7.0.4.3) + actionview (= 7.0.4.3) + activesupport (= 7.0.4.3) + rack (~> 2.0, >= 2.2.0) + rack-test (>= 0.6.3) + rails-dom-testing (~> 2.0) + rails-html-sanitizer (~> 1.0, >= 1.2.0) + actiontext (7.0.4.3) + actionpack (= 7.0.4.3) + activerecord (= 7.0.4.3) + activestorage (= 7.0.4.3) + activesupport (= 7.0.4.3) + globalid (>= 0.6.0) + nokogiri (>= 1.8.5) + actionview (7.0.4.3) + activesupport (= 7.0.4.3) + builder (~> 3.1) + erubi (~> 1.4) + rails-dom-testing (~> 2.0) + rails-html-sanitizer (~> 1.1, >= 1.2.0) + activejob (7.0.4.3) + activesupport (= 7.0.4.3) + globalid (>= 0.3.6) + activemodel (7.0.4.3) + activesupport (= 7.0.4.3) + activerecord (7.0.4.3) + activemodel (= 7.0.4.3) + activesupport (= 7.0.4.3) + activestorage (7.0.4.3) + actionpack (= 7.0.4.3) + activejob (= 7.0.4.3) + activerecord (= 7.0.4.3) + activesupport (= 7.0.4.3) + marcel (~> 1.0) + mini_mime (>= 1.1.0) + activesupport (7.0.4.3) + concurrent-ruby (~> 1.0, >= 1.0.2) + i18n (>= 1.6, < 2) + minitest (>= 5.1) + tzinfo (~> 2.0) + addressable (2.8.1) + public_suffix (>= 2.0.2, < 6.0) + ast (2.4.2) + bcrypt (3.1.18) + bindex (0.8.1) + bootsnap (1.16.0) + msgpack (~> 1.2) + builder (3.2.4) + capybara (3.38.0) + addressable + matrix + mini_mime (>= 0.1.3) + nokogiri (~> 1.8) + rack (>= 1.6.0) + rack-test (>= 0.6.3) + regexp_parser (>= 1.5, < 3.0) + xpath (~> 3.2) + concurrent-ruby (1.2.2) + crass (1.0.6) + date (3.3.3) + debug (1.7.2) + irb (>= 1.5.0) + reline (>= 0.3.1) + diff-lcs (1.5.0) + erubi (1.12.0) + globalid (1.1.0) + activesupport (>= 5.0) + i18n (1.12.0) + concurrent-ruby (~> 1.0) + importmap-rails (1.1.5) + actionpack (>= 6.0.0) + railties (>= 6.0.0) + io-console (0.6.0) + irb (1.6.3) + reline (>= 0.3.0) + jbuilder (2.11.5) + actionview (>= 5.0.0) + activesupport (>= 5.0.0) + json (2.6.3) + jwt (2.2.3) + loofah (2.19.1) + crass (~> 1.0.2) + nokogiri (>= 1.5.9) + mail (2.8.1) + mini_mime (>= 0.1.1) + net-imap + net-pop + net-smtp + marcel (1.0.2) + matrix (0.4.2) + method_source (1.0.0) + mini_mime (1.1.2) + minitest (5.18.0) + msgpack (1.7.0) + net-imap (0.3.4) + date + net-protocol + net-pop (0.1.2) + net-protocol + net-protocol (0.2.1) + timeout + net-smtp (0.3.3) + net-protocol + nio4r (2.5.8) + nokogiri (1.14.2-x64-mingw-ucrt) + racc (~> 1.4) + nokogiri (1.14.2-x86-mingw32) + racc (~> 1.4) + nokogiri (1.14.2-x86_64-darwin) + racc (~> 1.4) + nokogiri (1.14.2-x86_64-linux) + racc (~> 1.4) + parallel (1.22.1) + parser (3.2.2.0) + ast (~> 2.4.1) + pg (1.4.6) + pg (1.4.6-x64-mingw-ucrt) + pg (1.4.6-x86-mingw32) + public_suffix (5.0.1) + puma (5.6.5) + nio4r (~> 2.0) + racc (1.6.2) + rack (2.2.6.4) + rack-cors (2.0.1) + rack (>= 2.0.0) + rack-test (2.1.0) + rack (>= 1.3) + rails (7.0.4.3) + actioncable (= 7.0.4.3) + actionmailbox (= 7.0.4.3) + actionmailer (= 7.0.4.3) + actionpack (= 7.0.4.3) + actiontext (= 7.0.4.3) + actionview (= 7.0.4.3) + activejob (= 7.0.4.3) + activemodel (= 7.0.4.3) + activerecord (= 7.0.4.3) + activestorage (= 7.0.4.3) + activesupport (= 7.0.4.3) + bundler (>= 1.15.0) + railties (= 7.0.4.3) + rails-controller-testing (1.0.5) + actionpack (>= 5.0.1.rc1) + actionview (>= 5.0.1.rc1) + activesupport (>= 5.0.1.rc1) + rails-dom-testing (2.0.3) + activesupport (>= 4.2.0) + nokogiri (>= 1.6) + rails-html-sanitizer (1.5.0) + loofah (~> 2.19, >= 2.19.1) + railties (7.0.4.3) + actionpack (= 7.0.4.3) + activesupport (= 7.0.4.3) + method_source + rake (>= 12.2) + thor (~> 1.0) + zeitwerk (~> 2.5) + rainbow (3.1.1) + rake (13.0.6) + regexp_parser (2.7.0) + reline (0.3.3) + io-console (~> 0.5) + rexml (3.2.5) + rspec-core (3.12.1) + rspec-support (~> 3.12.0) + rspec-expectations (3.12.2) + diff-lcs (>= 1.2.0, < 2.0) + rspec-support (~> 3.12.0) + rspec-mocks (3.12.5) + diff-lcs (>= 1.2.0, < 2.0) + rspec-support (~> 3.12.0) + rspec-rails (6.0.1) + actionpack (>= 6.1) + activesupport (>= 6.1) + railties (>= 6.1) + rspec-core (~> 3.11) + rspec-expectations (~> 3.11) + rspec-mocks (~> 3.11) + rspec-support (~> 3.11) + rspec-support (3.12.0) + rubocop (1.48.1) + json (~> 2.3) + parallel (~> 1.10) + parser (>= 3.2.0.0) + rainbow (>= 2.2.2, < 4.0) + regexp_parser (>= 1.8, < 3.0) + rexml (>= 3.2.5, < 4.0) + rubocop-ast (>= 1.26.0, < 2.0) + ruby-progressbar (~> 1.7) + unicode-display_width (>= 2.4.0, < 3.0) + rubocop-ast (1.28.0) + parser (>= 3.2.1.0) + ruby-progressbar (1.13.0) + rubyzip (2.3.2) + selenium-webdriver (4.8.6) + rexml (~> 3.2, >= 3.2.5) + rubyzip (>= 1.2.2, < 3.0) + websocket (~> 1.0) + sprockets (4.2.0) + concurrent-ruby (~> 1.0) + rack (>= 2.2.4, < 4) + sprockets-rails (3.4.2) + actionpack (>= 5.2) + activesupport (>= 5.2) + sprockets (>= 3.0.0) + stimulus-rails (1.2.1) + railties (>= 6.0.0) + thor (1.2.1) + timeout (0.3.2) + turbo-rails (1.4.0) + actionpack (>= 6.0.0) + activejob (>= 6.0.0) + railties (>= 6.0.0) + tzinfo (2.0.6) + concurrent-ruby (~> 1.0) + tzinfo-data (1.2023.3) + tzinfo (>= 1.0.0) + unicode-display_width (2.4.2) + web-console (4.2.0) + actionview (>= 6.0.0) + activemodel (>= 6.0.0) + bindex (>= 0.4.0) + railties (>= 6.0.0) + webdrivers (5.2.0) + nokogiri (~> 1.6) + rubyzip (>= 1.3.0) + selenium-webdriver (~> 4.0) + websocket (1.2.9) + websocket-driver (0.7.5) + websocket-extensions (>= 0.1.0) + websocket-extensions (0.1.5) + xpath (3.2.0) + nokogiri (~> 1.8) + zeitwerk (2.6.7) + +PLATFORMS + x64-mingw-ucrt + x86-mingw32 + x86_64-darwin-19 + x86_64-linux + +DEPENDENCIES + bcrypt (~> 3.1.7) + bootsnap + capybara + debug + importmap-rails + jbuilder + jwt (~> 2.2.3) + pg (~> 1.1) + puma (~> 5.0) + rack-cors + rails (~> 7.0.4, >= 7.0.4.3) + rails-controller-testing + rspec-rails + rubocop (>= 1.0, < 2.0) + selenium-webdriver + sprockets-rails + stimulus-rails + turbo-rails + tzinfo-data + web-console + webdrivers + +RUBY VERSION + ruby 3.2.0p0 + +BUNDLED WITH + 2.4.7 diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..60d4a6c --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2023 Vanel Nwaba + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..2aa83c2 --- /dev/null +++ b/README.md @@ -0,0 +1,208 @@ +# 📗 Table of Contents + +- [📗 Table of Contents](#-table-of-contents) +- [📖 Book my Ride App ](#-car-rental-app-) + - [Kanban Board](#kanban-board) + - [API Documentation](#api-documentation) + - [Frontend](#frontend) + - [🛠 Built With ](#-built-with-) + - [Tech Stack ](#tech-stack-) + - [Key Features ](#key-features-) + - [🚀 Live Demo ](#-live-demo-) + - [💻 Getting Started ](#-getting-started-) + - [Prerequisites](#prerequisites) + - [Setup](#setup) + - [Install](#install) + - [Usage](#usage) + - [Run tests](#run-tests) + - [Deployment](#deployment) + - [👥 Authors ](#-authors-) + - [🔭 Future Features ](#-future-features-) + - [🤝 Contributing ](#-contributing-) + - [⭐️ Show your support ](#️-show-your-support-) + - [🙏 Acknowledgments ](#-acknowledgments-) + - [📝 License ](#-license-) + +# 📖 Book my Ride App + +**Book my Ride App** is a car rental application that provides users with the ability to create an account, log in, and reserve a car for a designated time frame. It also allows users to browse through all available vehicles and view specific cars. Additionally, users can check their own booking history and view all reservations made by them. + +## Kanban Board + +Collaboration team is made of 5 members: + +- Abeera Tahir +- Vanel Nwaba +- Ezema Anthony Sunday +- Khomotso Mkansi +- Zia Bakhteyari + +### Initial State of Kanban: + +![kanban](https://user-images.githubusercontent.com/38879488/228954839-ecab1c29-7bad-4338-8b62-68f996f6ce7d.png) + +[Kanban Board Link](https://github.com/users/vanelnw/projects/4) + +## API Documentation + +[Link to API Documentation](https://documenter.getpostman.com/view/24548009/2s93XsYS33) + +## Frontend + +The frontend is made with react and can be found [here] (https://github.com/vanelnw/appointment-capstone-frontend.git) + +## 🛠 Built With + +### Tech Stack + +
+ Server + +
+ +
+Database + +
+ +### Key Features + +- **Deploy application API on render** +- **Update Api documentation with deployed url** + +

(back to top)

+ +## 🚀 Live Demo + +- [Live Demo Link](https://rentyourcar.onrender.com) + +

(back to top)

+ +## 💻 Getting Started + +To get a local copy up and running, follow these steps. + +### Prerequisites + +In order to run this project you need: + +```sh + gem install rails +``` + +### Setup + +Clone this repository to your desired folder: + +```sh + cd my-folder + git clone https://github.com/vanelnw/Appointment-Capstone-Backend.git +``` + +### Install + +```sh + cd Appointment-Capstone-Backend + bundle install +``` + +### Usage + +To run the project, execute the following command: + +``` + rails db:create +``` + +``` + rails db:migrate +``` + +``` + rails db:seed +``` + +``` + rails server +``` + +### Run tests + +To run tests, run the following command: + +```sh + rspec +``` + +

(back to top)

+ +## 👥 Authors + +👤 **Zia Bakhteyari** + +- GitHub: [@zia123456](https://github.com/Zia123456) +- Twitter: [@zia_bakhteyari](https://twitter.com/Zia_Bakhteyari) +- LinkedIn: [Zia bakhteyari](https://www.linkedin.com/in/zia-bakhteyari) + +👤 **Khomotso Mkansi** + +- GitHub: [@githubhandle](https://momotsow.github.io/microverse-portfolio/) +- Twitter: [@twitterhandle](#) +- LinkedIn: [LinkedIn](https://www.linkedin.com/in/khomotso-prudence-mkansi/) + +👤 **Vanel Nwaba** + +- GitHub: [@vanelnw](https://github.com/vanelnw) +- LinkedIn: [Vanel Nwaba](https://www.linkedin.com/in/va-nw/) +- Twitter: [@vanelnw](#) + +👤 **Ezema Anthony Sunday** + +- GitHub: [@sonyco-4u](https://github.com/sonyco-4u) +- Twitter: [@EZEMASUN](https://twitter.com/EZEMASUN) +- LinkedIn: [ezema-anthony-sunday](https://www.linkedin.com/in/sunday-anthony-ezema/) + +👤 **Abeera Tahir** + +- GitHub: [@AbeeraTahir](https://github.com/AbeeraTahir) +- Twitter: [@AbeeraTahir8](https://twitter.com/AbeeraTahir8?t=z5CjMpmHMZmS98i09gUpYA&s=08) +- LinkedIn: [Abeera Tahir](https://www.linkedin.com/in/abeera-tahir/) + +

(back to top)

+ +## 🔭 Future Features + +- [ ] **Integrate Admin action.** +- [ ] **Integrate super users.** + +

(back to top)

+ +## 🤝 Contributing + +Contributions, issues, and feature requests are welcome! + +Feel free to check the [issues page](../../issues/). + +

(back to top)

+ +## ⭐️ Show your support + +If you like this project consider giving it a star ⭐️. + +

(back to top)

+ +## 🙏 Acknowledgments + +Special thank to [Murat Korkmaz](https://www.behance.net/muratk) for providing the design guidelines. + +

(back to top)

+ +## 📝 License + +This project is [MIT](./LICENSE) licensed. + +

(back to top)

diff --git a/Rakefile b/Rakefile new file mode 100644 index 0000000..9a5ea73 --- /dev/null +++ b/Rakefile @@ -0,0 +1,6 @@ +# Add your own tasks in files placed in lib/tasks ending in .rake, +# for example lib/tasks/capistrano.rake, and they will automatically be available to Rake. + +require_relative "config/application" + +Rails.application.load_tasks diff --git a/app/assets/config/manifest.js b/app/assets/config/manifest.js new file mode 100644 index 0000000..ddd546a --- /dev/null +++ b/app/assets/config/manifest.js @@ -0,0 +1,4 @@ +//= link_tree ../images +//= link_directory ../stylesheets .css +//= link_tree ../../javascript .js +//= link_tree ../../../vendor/javascript .js diff --git a/app/assets/images/.keep b/app/assets/images/.keep new file mode 100644 index 0000000..e69de29 diff --git a/app/assets/stylesheets/application.css b/app/assets/stylesheets/application.css new file mode 100644 index 0000000..288b9ab --- /dev/null +++ b/app/assets/stylesheets/application.css @@ -0,0 +1,15 @@ +/* + * This is a manifest file that'll be compiled into application.css, which will include all the files + * listed below. + * + * Any CSS (and SCSS, if configured) file within this directory, lib/assets/stylesheets, or any plugin's + * vendor/assets/stylesheets directory can be referenced here using a relative path. + * + * You're free to add application-wide styles to this file and they'll appear at the bottom of the + * compiled file so the styles you add here take precedence over styles defined in any other CSS + * files in this directory. Styles in this file should be added after the last require_* statement. + * It is generally better to create a new file per style scope. + * + *= require_tree . + *= require_self + */ diff --git a/app/channels/application_cable/channel.rb b/app/channels/application_cable/channel.rb new file mode 100644 index 0000000..d672697 --- /dev/null +++ b/app/channels/application_cable/channel.rb @@ -0,0 +1,4 @@ +module ApplicationCable + class Channel < ActionCable::Channel::Base + end +end diff --git a/app/channels/application_cable/connection.rb b/app/channels/application_cable/connection.rb new file mode 100644 index 0000000..0ff5442 --- /dev/null +++ b/app/channels/application_cable/connection.rb @@ -0,0 +1,4 @@ +module ApplicationCable + class Connection < ActionCable::Connection::Base + end +end diff --git a/app/controllers/api/v1/application_controller.rb b/app/controllers/api/v1/application_controller.rb new file mode 100644 index 0000000..b3db151 --- /dev/null +++ b/app/controllers/api/v1/application_controller.rb @@ -0,0 +1,19 @@ +require_relative './json_web_token' + +class Api::V1::ApplicationController < ActionController::API + before_action :authenticate_request + before_action :set_default_format + + private + + def set_default_format + request.format = :json + end + + def authenticate_request + header = request.headers['Authorization'] + header = header.split.last if header + decoded = JsonWebToken.jwt_decode(header) + @current_user = User.find(decoded[:user_id]) + end +end diff --git a/app/controllers/api/v1/authentication_controller.rb b/app/controllers/api/v1/authentication_controller.rb new file mode 100644 index 0000000..c762a45 --- /dev/null +++ b/app/controllers/api/v1/authentication_controller.rb @@ -0,0 +1,43 @@ +class Api::V1::AuthenticationController < Api::V1::ApplicationController + skip_before_action :authenticate_request + + def login + @user = User.find_by_email(params[:email]) + if @user&.authenticate(params[:password]) + token = JsonWebToken.jwt_encode(user_id: @user.id) + render json: { user: @user, token:, message: 'Logged in successfully!' }, status: :ok + else + render json: { token: 'unauthorized', message: 'Invalid email or password!' }, status: :unauthorized + end + end + + def logout + session.delete(:user_id) + render json: { message: 'Logged out successfully!' } + end + + def register + @user = User.find_by(email: params[:email]) + + if @user.present? + render json: { message: 'Email already exists!' }, status: :unprocessable_entity + else + @user = User.new(user_params) + + @user.role = 'admin' if @user.email == 'admin@gmail.com' + + if @user.save + token = JsonWebToken.jwt_encode(user_id: @user.id) + render json: { user: @user, token:, message: 'Registered successfully!' }, status: :created + else + render json: { message: @user.errors.full_messages.join(', ') }, status: :unprocessable_entity + end + end + end + + private + + def user_params + params.permit(:name, :email, :password, :role) + end +end diff --git a/app/controllers/api/v1/cars_controller.rb b/app/controllers/api/v1/cars_controller.rb new file mode 100644 index 0000000..4ab875a --- /dev/null +++ b/app/controllers/api/v1/cars_controller.rb @@ -0,0 +1,71 @@ +class Api::V1::CarsController < Api::V1::ApplicationController + before_action :set_car, only: %i[show edit update destroy] + before_action :authenticate_request + + # GET /cars or /cars.json + def index + @cars = Car.all + render json: @cars + end + + # GET /cars/1 or /cars/1.json + def show + render json: @car + end + + # GET /cars/new + def new + @car = Car.new + end + + # GET /cars/1/edit + def edit; end + + # POST /cars or /cars.json + def create + @car = Car.new(car_params) + @car.user_id = @current_user.id + + if @car.save + render json: { car: @car, message: 'Car added successfully!' }, status: :created + else + render json: { message: @car.errors.full_messages.join(', ') }, status: :unprocessable_entity + end + end + + # PATCH/PUT /cars/1 or /cars/1.json + def update + respond_to do |format| + if @car.update(car_params) + format.html { redirect_to car_url(@car), notice: 'Car was successfully updated.' } + format.json { render :show, status: :ok, location: @car } + else + format.html { render :edit, status: :unprocessable_entity } + format.json { render json: @car.errors, status: :unprocessable_entity } + end + end + end + + # DELETE /cars/1 or /cars/1.json + + def destroy + @car = Car.find(params[:id]) + if @car.destroy + render json: { message: 'Car successfully deleted' }, status: :ok + else + render json: { message: @car.errors.full_messages.join(', ') }, status: :unprocessable_entity + end + end + + private + + # Use callbacks to share common setup or constraints between actions. + def set_car + @car = Car.find(params[:id]) + end + + # Only allow a list of trusted parameters through. + def car_params + params.require(:car).permit(:make, :model, :year, :image, :description, :daily_rate) + end +end diff --git a/app/controllers/api/v1/json_web_token.rb b/app/controllers/api/v1/json_web_token.rb new file mode 100644 index 0000000..800c695 --- /dev/null +++ b/app/controllers/api/v1/json_web_token.rb @@ -0,0 +1,15 @@ +require 'jwt' +class JsonWebToken + extend ActiveSupport::Concern + SECRET_KEY = Rails.application.secret_key_base + + def self.jwt_encode(payload, exp = 7.days.from_now) + payload[:exp] = exp.to_i + JWT.encode(payload, SECRET_KEY) + end + + def self.jwt_decode(token) + decoded = JWT.decode(token, SECRET_KEY)[0] + HashWithIndifferentAccess.new decoded + end +end diff --git a/app/controllers/api/v1/reservations_controller.rb b/app/controllers/api/v1/reservations_controller.rb new file mode 100644 index 0000000..bbe5d66 --- /dev/null +++ b/app/controllers/api/v1/reservations_controller.rb @@ -0,0 +1,71 @@ +class Api::V1::ReservationsController < Api::V1::ApplicationController + before_action :set_reservation, only: %i[show edit update destroy] + before_action :authenticate_request + + # GET /reservations or /reservations.json + def index + @reservations = Reservation.includes(:car).all + render json: { reservations: @reservations.as_json(methods: + %i[car_make car_image car_model car_year car_daily_rate]) } + end + + # GET /reservations/1 or /reservations/1.json + def show; end + + # GET /reservations/new + def new + @reservation = Reservation.new + end + + # GET /reservations/1/edit + def edit; end + + # POST /reservations or /reservations.json + def create + @reservation = Reservation.new(reservation_params) + @reservation.user_id = @current_user.id + + if @reservation.save + render json: { reservation: @reservation, success: true, message: 'Car reserved successfully!' }, + status: :created + else + render json: { success: false, message: @reservation.errors.full_messages.join(', '), + status: :unprocessable_entity } + end + end + + # PATCH/PUT /reservations/1 or /reservations/1.json + def update + respond_to do |format| + if @reservation.update(reservation_params) + format.html { redirect_to reservation_url(@reservation), notice: 'Reservation was successfully updated.' } + format.json { render :show, status: :ok, location: @reservation } + else + format.html { render :edit, status: :unprocessable_entity } + format.json { render json: @reservation.errors, status: :unprocessable_entity } + end + end + end + + # DELETE /reservations/1 or /reservations/1.json + def destroy + @reservation.destroy + + respond_to do |format| + format.html { redirect_to reservations_url, notice: 'Reservation was successfully destroyed.' } + format.json { head :no_content } + end + end + + private + + # Use callbacks to share common setup or constraints between actions. + def set_reservation + @reservation = Reservation.find(params[:id]) + end + + # Only allow a list of trusted parameters through. + def reservation_params + params.require(:reservation).permit(:reservation_date, :due_date, :car_id) + end +end diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb new file mode 100644 index 0000000..09705d1 --- /dev/null +++ b/app/controllers/application_controller.rb @@ -0,0 +1,2 @@ +class ApplicationController < ActionController::Base +end diff --git a/app/controllers/concerns/.keep b/app/controllers/concerns/.keep new file mode 100644 index 0000000..e69de29 diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb new file mode 100644 index 0000000..e1f5419 --- /dev/null +++ b/app/controllers/users_controller.rb @@ -0,0 +1,16 @@ +class UsersController < ApplicationController + def create + @user = User.new(user_params) + if @user.save + render json: { status: :created, user: @user } + else + render json: { errors: @user.errors.full_messages }, status: :unprocessable_entity + end + end + + private + + def user_params + params.require(:user).permit(:name, :email, :password) + end +end diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb new file mode 100644 index 0000000..de6be79 --- /dev/null +++ b/app/helpers/application_helper.rb @@ -0,0 +1,2 @@ +module ApplicationHelper +end diff --git a/app/helpers/appointments_helper.rb b/app/helpers/appointments_helper.rb new file mode 100644 index 0000000..ec5f632 --- /dev/null +++ b/app/helpers/appointments_helper.rb @@ -0,0 +1,2 @@ +module AppointmentsHelper +end diff --git a/app/helpers/cars_helper.rb b/app/helpers/cars_helper.rb new file mode 100644 index 0000000..c410fa3 --- /dev/null +++ b/app/helpers/cars_helper.rb @@ -0,0 +1,2 @@ +module CarsHelper +end diff --git a/app/helpers/reservations_helper.rb b/app/helpers/reservations_helper.rb new file mode 100644 index 0000000..f28b699 --- /dev/null +++ b/app/helpers/reservations_helper.rb @@ -0,0 +1,2 @@ +module ReservationsHelper +end diff --git a/app/helpers/users_helper.rb b/app/helpers/users_helper.rb new file mode 100644 index 0000000..2310a24 --- /dev/null +++ b/app/helpers/users_helper.rb @@ -0,0 +1,2 @@ +module UsersHelper +end diff --git a/app/javascript/application.js b/app/javascript/application.js new file mode 100644 index 0000000..0d7b494 --- /dev/null +++ b/app/javascript/application.js @@ -0,0 +1,3 @@ +// Configure your import map in config/importmap.rb. Read more: https://github.com/rails/importmap-rails +import "@hotwired/turbo-rails" +import "controllers" diff --git a/app/javascript/controllers/application.js b/app/javascript/controllers/application.js new file mode 100644 index 0000000..1213e85 --- /dev/null +++ b/app/javascript/controllers/application.js @@ -0,0 +1,9 @@ +import { Application } from "@hotwired/stimulus" + +const application = Application.start() + +// Configure Stimulus development experience +application.debug = false +window.Stimulus = application + +export { application } diff --git a/app/javascript/controllers/hello_controller.js b/app/javascript/controllers/hello_controller.js new file mode 100644 index 0000000..5975c07 --- /dev/null +++ b/app/javascript/controllers/hello_controller.js @@ -0,0 +1,7 @@ +import { Controller } from "@hotwired/stimulus" + +export default class extends Controller { + connect() { + this.element.textContent = "Hello World!" + } +} diff --git a/app/javascript/controllers/index.js b/app/javascript/controllers/index.js new file mode 100644 index 0000000..54ad4ca --- /dev/null +++ b/app/javascript/controllers/index.js @@ -0,0 +1,11 @@ +// Import and register all your controllers from the importmap under controllers/* + +import { application } from "controllers/application" + +// Eager load all controllers defined in the import map under controllers/**/*_controller +import { eagerLoadControllersFrom } from "@hotwired/stimulus-loading" +eagerLoadControllersFrom("controllers", application) + +// Lazy load controllers as they appear in the DOM (remember not to preload controllers in import map!) +// import { lazyLoadControllersFrom } from "@hotwired/stimulus-loading" +// lazyLoadControllersFrom("controllers", application) diff --git a/app/jobs/application_job.rb b/app/jobs/application_job.rb new file mode 100644 index 0000000..d394c3d --- /dev/null +++ b/app/jobs/application_job.rb @@ -0,0 +1,7 @@ +class ApplicationJob < ActiveJob::Base + # Automatically retry jobs that encountered a deadlock + # retry_on ActiveRecord::Deadlocked + + # Most jobs are safe to ignore if the underlying records are no longer available + # discard_on ActiveJob::DeserializationError +end diff --git a/app/mailers/application_mailer.rb b/app/mailers/application_mailer.rb new file mode 100644 index 0000000..286b223 --- /dev/null +++ b/app/mailers/application_mailer.rb @@ -0,0 +1,4 @@ +class ApplicationMailer < ActionMailer::Base + default from: 'from@example.com' + layout 'mailer' +end diff --git a/app/models/application_record.rb b/app/models/application_record.rb new file mode 100644 index 0000000..b63caeb --- /dev/null +++ b/app/models/application_record.rb @@ -0,0 +1,3 @@ +class ApplicationRecord < ActiveRecord::Base + primary_abstract_class +end diff --git a/app/models/car.rb b/app/models/car.rb new file mode 100644 index 0000000..152fc20 --- /dev/null +++ b/app/models/car.rb @@ -0,0 +1,12 @@ +class Car < ApplicationRecord + belongs_to :user + has_many :reservations, dependent: :destroy, foreign_key: 'car_id', class_name: 'Reservation' + + validates :make, presence: true + validates :model, presence: true + validates :year, presence: true, numericality: { less_than_or_equal_to: Date.today.year } + validates :daily_rate, presence: true, numericality: { greater_than: 0 } + validates :user, presence: true + validates :image, presence: true + validates :description, presence: true +end diff --git a/app/models/concerns/.keep b/app/models/concerns/.keep new file mode 100644 index 0000000..e69de29 diff --git a/app/models/reservation.rb b/app/models/reservation.rb new file mode 100644 index 0000000..8b30d7a --- /dev/null +++ b/app/models/reservation.rb @@ -0,0 +1,27 @@ +class Reservation < ApplicationRecord + belongs_to :user + belongs_to :car + + validates :reservation_date, presence: true + validates :due_date, presence: true + + def car_make + car.make + end + + def car_image + car.image + end + + def car_model + car.model + end + + def car_year + car.year + end + + def car_daily_rate + car.daily_rate + end +end diff --git a/app/models/user.rb b/app/models/user.rb new file mode 100644 index 0000000..20fb307 --- /dev/null +++ b/app/models/user.rb @@ -0,0 +1,8 @@ +class User < ApplicationRecord + has_secure_password + has_many :reservations + has_many :cars + + validates :name, presence: true + validates :email, presence: true, uniqueness: true +end diff --git a/app/views/cars/_car.html.erb b/app/views/cars/_car.html.erb new file mode 100644 index 0000000..2717337 --- /dev/null +++ b/app/views/cars/_car.html.erb @@ -0,0 +1,22 @@ +
+

+ Make: + <%= car.make %> +

+ +

+ Model: + <%= car.model %> +

+ +

+ Year: + <%= car.year %> +

+ +

+ Price: + <%= car.price %> +

+ +
diff --git a/app/views/cars/_car.json.jbuilder b/app/views/cars/_car.json.jbuilder new file mode 100644 index 0000000..7fff5ad --- /dev/null +++ b/app/views/cars/_car.json.jbuilder @@ -0,0 +1,2 @@ +json.extract! car, :id, :make, :model, :year, :price, :created_at, :updated_at +json.url car_url(car, format: :json) diff --git a/app/views/cars/_form.html.erb b/app/views/cars/_form.html.erb new file mode 100644 index 0000000..a6f609d --- /dev/null +++ b/app/views/cars/_form.html.erb @@ -0,0 +1,37 @@ +<%= form_with(model: car) do |form| %> + <% if car.errors.any? %> +
+

<%= pluralize(car.errors.count, "error") %> prohibited this car from being saved:

+ + +
+ <% end %> + +
+ <%= form.label :make, style: "display: block" %> + <%= form.text_field :make %> +
+ +
+ <%= form.label :model, style: "display: block" %> + <%= form.text_field :model %> +
+ +
+ <%= form.label :year, style: "display: block" %> + <%= form.number_field :year %> +
+ +
+ <%= form.label :price, style: "display: block" %> + <%= form.number_field :price %> +
+ +
+ <%= form.submit %> +
+<% end %> diff --git a/app/views/cars/edit.html.erb b/app/views/cars/edit.html.erb new file mode 100644 index 0000000..bcf1cf3 --- /dev/null +++ b/app/views/cars/edit.html.erb @@ -0,0 +1,10 @@ +

Editing car

+ +<%= render "form", car: @car %> + +
+ +
+ <%= link_to "Show this car", @car %> | + <%= link_to "Back to cars", cars_path %> +
diff --git a/app/views/cars/index.html.erb b/app/views/cars/index.html.erb new file mode 100644 index 0000000..9acc865 --- /dev/null +++ b/app/views/cars/index.html.erb @@ -0,0 +1,14 @@ +

<%= notice %>

+ +

Cars

+ +
+ <% @cars.each do |car| %> + <%= render car %> +

+ <%= link_to "Show this car", car %> +

+ <% end %> +
+ +<%= link_to "New car", new_car_path %> diff --git a/app/views/cars/index.json.jbuilder b/app/views/cars/index.json.jbuilder new file mode 100644 index 0000000..6390597 --- /dev/null +++ b/app/views/cars/index.json.jbuilder @@ -0,0 +1 @@ +json.array! @cars, partial: 'cars/car', as: :car diff --git a/app/views/cars/new.html.erb b/app/views/cars/new.html.erb new file mode 100644 index 0000000..8d443cd --- /dev/null +++ b/app/views/cars/new.html.erb @@ -0,0 +1,9 @@ +

New car

+ +<%= render "form", car: @car %> + +
+ +
+ <%= link_to "Back to cars", cars_path %> +
diff --git a/app/views/cars/show.html.erb b/app/views/cars/show.html.erb new file mode 100644 index 0000000..96090bb --- /dev/null +++ b/app/views/cars/show.html.erb @@ -0,0 +1,10 @@ +

<%= notice %>

+ +<%= render @car %> + +
+ <%= link_to "Edit this car", edit_car_path(@car) %> | + <%= link_to "Back to cars", cars_path %> + + <%= button_to "Destroy this car", @car, method: :delete %> +
diff --git a/app/views/cars/show.json.jbuilder b/app/views/cars/show.json.jbuilder new file mode 100644 index 0000000..5395377 --- /dev/null +++ b/app/views/cars/show.json.jbuilder @@ -0,0 +1 @@ +json.partial! 'cars/car', car: @car diff --git a/app/views/layouts/application.html.erb b/app/views/layouts/application.html.erb new file mode 100644 index 0000000..949a921 --- /dev/null +++ b/app/views/layouts/application.html.erb @@ -0,0 +1,16 @@ + + + + AppointmentCapstoneBackend + + <%= csrf_meta_tags %> + <%= csp_meta_tag %> + + <%= stylesheet_link_tag "application", "data-turbo-track": "reload" %> + <%= javascript_importmap_tags %> + + + + <%= yield %> + + diff --git a/app/views/layouts/mailer.html.erb b/app/views/layouts/mailer.html.erb new file mode 100644 index 0000000..cbd34d2 --- /dev/null +++ b/app/views/layouts/mailer.html.erb @@ -0,0 +1,13 @@ + + + + + + + + + <%= yield %> + + diff --git a/app/views/layouts/mailer.text.erb b/app/views/layouts/mailer.text.erb new file mode 100644 index 0000000..37f0bdd --- /dev/null +++ b/app/views/layouts/mailer.text.erb @@ -0,0 +1 @@ +<%= yield %> diff --git a/app/views/reservations/_form.html.erb b/app/views/reservations/_form.html.erb new file mode 100644 index 0000000..0d27e81 --- /dev/null +++ b/app/views/reservations/_form.html.erb @@ -0,0 +1,27 @@ +<%= form_with(model: reservation) do |form| %> + <% if reservation.errors.any? %> +
+

<%= pluralize(reservation.errors.count, "error") %> prohibited this reservation from being saved:

+ + +
+ <% end %> + +
+ <%= form.label :reservation_date, style: "display: block" %> + <%= form.date_field :reservation_date %> +
+ +
+ <%= form.label :appointment_id, style: "display: block" %> + <%= form.text_field :appointment_id %> +
+ +
+ <%= form.submit %> +
+<% end %> diff --git a/app/views/reservations/_reservation.html.erb b/app/views/reservations/_reservation.html.erb new file mode 100644 index 0000000..0a10b2c --- /dev/null +++ b/app/views/reservations/_reservation.html.erb @@ -0,0 +1,12 @@ +
+

+ Reservation date: + <%= reservation.reservation_date %> +

+ +

+ Appointment: + <%= reservation.appointment_id %> +

+ +
diff --git a/app/views/reservations/_reservation.json.jbuilder b/app/views/reservations/_reservation.json.jbuilder new file mode 100644 index 0000000..31944bb --- /dev/null +++ b/app/views/reservations/_reservation.json.jbuilder @@ -0,0 +1,2 @@ +json.extract! reservation, :id, :reservation_date, :appointment_id, :created_at, :updated_at +json.url reservation_url(reservation, format: :json) diff --git a/app/views/reservations/edit.html.erb b/app/views/reservations/edit.html.erb new file mode 100644 index 0000000..527678d --- /dev/null +++ b/app/views/reservations/edit.html.erb @@ -0,0 +1,10 @@ +

Editing reservation

+ +<%= render "form", reservation: @reservation %> + +
+ +
+ <%= link_to "Show this reservation", @reservation %> | + <%= link_to "Back to reservations", reservations_path %> +
diff --git a/app/views/reservations/index.html.erb b/app/views/reservations/index.html.erb new file mode 100644 index 0000000..5a1d363 --- /dev/null +++ b/app/views/reservations/index.html.erb @@ -0,0 +1,14 @@ +

<%= notice %>

+ +

Reservations

+ +
+ <% @reservations.each do |reservation| %> + <%= render reservation %> +

+ <%= link_to "Show this reservation", reservation %> +

+ <% end %> +
+ +<%= link_to "New reservation", new_reservation_path %> diff --git a/app/views/reservations/index.json.jbuilder b/app/views/reservations/index.json.jbuilder new file mode 100644 index 0000000..e0f3915 --- /dev/null +++ b/app/views/reservations/index.json.jbuilder @@ -0,0 +1 @@ +json.array! @reservations, partial: 'reservations/reservation', as: :reservation diff --git a/app/views/reservations/new.html.erb b/app/views/reservations/new.html.erb new file mode 100644 index 0000000..7dbb041 --- /dev/null +++ b/app/views/reservations/new.html.erb @@ -0,0 +1,9 @@ +

New reservation

+ +<%= render "form", reservation: @reservation %> + +
+ +
+ <%= link_to "Back to reservations", reservations_path %> +
diff --git a/app/views/reservations/show.html.erb b/app/views/reservations/show.html.erb new file mode 100644 index 0000000..273cdbb --- /dev/null +++ b/app/views/reservations/show.html.erb @@ -0,0 +1,10 @@ +

<%= notice %>

+ +<%= render @reservation %> + +
+ <%= link_to "Edit this reservation", edit_reservation_path(@reservation) %> | + <%= link_to "Back to reservations", reservations_path %> + + <%= button_to "Destroy this reservation", @reservation, method: :delete %> +
diff --git a/app/views/reservations/show.json.jbuilder b/app/views/reservations/show.json.jbuilder new file mode 100644 index 0000000..32b9d3b --- /dev/null +++ b/app/views/reservations/show.json.jbuilder @@ -0,0 +1 @@ +json.partial! 'reservations/reservation', reservation: @reservation diff --git a/app/views/users/_form.html.erb b/app/views/users/_form.html.erb new file mode 100644 index 0000000..3d71cb1 --- /dev/null +++ b/app/views/users/_form.html.erb @@ -0,0 +1,32 @@ +<%= form_with(model: user) do |form| %> + <% if user.errors.any? %> +
+

<%= pluralize(user.errors.count, "error") %> prohibited this user from being saved:

+ + +
+ <% end %> + +
+ <%= form.label :name, style: "display: block" %> + <%= form.text_field :name %> +
+ +
+ <%= form.label :email, style: "display: block" %> + <%= form.text_field :email %> +
+ +
+ <%= form.label :phone, style: "display: block" %> + <%= form.text_field :phone %> +
+ +
+ <%= form.submit %> +
+<% end %> diff --git a/app/views/users/_user.html.erb b/app/views/users/_user.html.erb new file mode 100644 index 0000000..8b21cf0 --- /dev/null +++ b/app/views/users/_user.html.erb @@ -0,0 +1,17 @@ +
+

+ Name: + <%= user.name %> +

+ +

+ Email: + <%= user.email %> +

+ +

+ Phone: + <%= user.phone %> +

+ +
diff --git a/app/views/users/_user.json.jbuilder b/app/views/users/_user.json.jbuilder new file mode 100644 index 0000000..c75391b --- /dev/null +++ b/app/views/users/_user.json.jbuilder @@ -0,0 +1,2 @@ +json.extract! user, :id, :name, :email, :phone, :created_at, :updated_at +json.url user_url(user, format: :json) diff --git a/app/views/users/edit.html.erb b/app/views/users/edit.html.erb new file mode 100644 index 0000000..9dda632 --- /dev/null +++ b/app/views/users/edit.html.erb @@ -0,0 +1,10 @@ +

Editing user

+ +<%= render "form", user: @user %> + +
+ +
+ <%= link_to "Show this user", @user %> | + <%= link_to "Back to users", users_path %> +
diff --git a/app/views/users/index.html.erb b/app/views/users/index.html.erb new file mode 100644 index 0000000..fe4dd5c --- /dev/null +++ b/app/views/users/index.html.erb @@ -0,0 +1,14 @@ +

<%= notice %>

+ +

Users

+ +
+ <% @users.each do |user| %> + <%= render user %> +

+ <%= link_to "Show this user", user %> +

+ <% end %> +
+ +<%= link_to "New user", new_user_path %> diff --git a/app/views/users/index.json.jbuilder b/app/views/users/index.json.jbuilder new file mode 100644 index 0000000..2faf5af --- /dev/null +++ b/app/views/users/index.json.jbuilder @@ -0,0 +1 @@ +json.array! @users, partial: 'users/user', as: :user diff --git a/app/views/users/new.html.erb b/app/views/users/new.html.erb new file mode 100644 index 0000000..eedbd83 --- /dev/null +++ b/app/views/users/new.html.erb @@ -0,0 +1,9 @@ +

New user

+ +<%= render "form", user: @user %> + +
+ +
+ <%= link_to "Back to users", users_path %> +
diff --git a/app/views/users/show.html.erb b/app/views/users/show.html.erb new file mode 100644 index 0000000..673fae2 --- /dev/null +++ b/app/views/users/show.html.erb @@ -0,0 +1,10 @@ +

<%= notice %>

+ +<%= render @user %> + +
+ <%= link_to "Edit this user", edit_user_path(@user) %> | + <%= link_to "Back to users", users_path %> + + <%= button_to "Destroy this user", @user, method: :delete %> +
diff --git a/app/views/users/show.json.jbuilder b/app/views/users/show.json.jbuilder new file mode 100644 index 0000000..2a33f71 --- /dev/null +++ b/app/views/users/show.json.jbuilder @@ -0,0 +1 @@ +json.partial! 'users/user', user: @user diff --git a/bin/bundle b/bin/bundle new file mode 100644 index 0000000..ee73929 --- /dev/null +++ b/bin/bundle @@ -0,0 +1,109 @@ +#!/usr/bin/env ruby +# frozen_string_literal: true + +# +# This file was generated by Bundler. +# +# The application 'bundle' is installed as part of a gem, and +# this file is here to facilitate running it. +# + +require "rubygems" + +m = Module.new do + module_function + + def invoked_as_script? + File.expand_path($0) == File.expand_path(__FILE__) + end + + def env_var_version + ENV["BUNDLER_VERSION"] + end + + def cli_arg_version + return unless invoked_as_script? # don't want to hijack other binstubs + return unless "update".start_with?(ARGV.first || " ") # must be running `bundle update` + bundler_version = nil + update_index = nil + ARGV.each_with_index do |a, i| + if update_index && update_index.succ == i && a =~ Gem::Version::ANCHORED_VERSION_PATTERN + bundler_version = a + end + next unless a =~ /\A--bundler(?:[= ](#{Gem::Version::VERSION_PATTERN}))?\z/ + bundler_version = $1 + update_index = i + end + bundler_version + end + + def gemfile + gemfile = ENV["BUNDLE_GEMFILE"] + return gemfile if gemfile && !gemfile.empty? + + File.expand_path("../Gemfile", __dir__) + end + + def lockfile + lockfile = + case File.basename(gemfile) + when "gems.rb" then gemfile.sub(/\.rb$/, gemfile) + else "#{gemfile}.lock" + end + File.expand_path(lockfile) + end + + def lockfile_version + return unless File.file?(lockfile) + lockfile_contents = File.read(lockfile) + return unless lockfile_contents =~ /\n\nBUNDLED WITH\n\s{2,}(#{Gem::Version::VERSION_PATTERN})\n/ + Regexp.last_match(1) + end + + def bundler_requirement + @bundler_requirement ||= + env_var_version || + cli_arg_version || + bundler_requirement_for(lockfile_version) + end + + def bundler_requirement_for(version) + return "#{Gem::Requirement.default}.a" unless version + + bundler_gem_version = Gem::Version.new(version) + + bundler_gem_version.approximate_recommendation + end + + def load_bundler! + ENV["BUNDLE_GEMFILE"] ||= gemfile + + activate_bundler + end + + def activate_bundler + gem_error = activation_error_handling do + gem "bundler", bundler_requirement + end + return if gem_error.nil? + require_error = activation_error_handling do + require "bundler/version" + end + return if require_error.nil? && Gem::Requirement.new(bundler_requirement).satisfied_by?(Gem::Version.new(Bundler::VERSION)) + warn "Activating bundler (#{bundler_requirement}) failed:\n#{gem_error.message}\n\nTo install the version of bundler this project requires, run `gem install bundler -v '#{bundler_requirement}'`" + exit 42 + end + + def activation_error_handling + yield + nil + rescue StandardError, LoadError => e + e + end +end + +m.load_bundler! + +if m.invoked_as_script? + load Gem.bin_path("bundler", "bundle") +end diff --git a/bin/importmap b/bin/importmap new file mode 100644 index 0000000..36502ab --- /dev/null +++ b/bin/importmap @@ -0,0 +1,4 @@ +#!/usr/bin/env ruby + +require_relative "../config/application" +require "importmap/commands" diff --git a/bin/rails b/bin/rails new file mode 100644 index 0000000..efc0377 --- /dev/null +++ b/bin/rails @@ -0,0 +1,4 @@ +#!/usr/bin/env ruby +APP_PATH = File.expand_path("../config/application", __dir__) +require_relative "../config/boot" +require "rails/commands" diff --git a/bin/rake b/bin/rake new file mode 100644 index 0000000..4fbf10b --- /dev/null +++ b/bin/rake @@ -0,0 +1,4 @@ +#!/usr/bin/env ruby +require_relative "../config/boot" +require "rake" +Rake.application.run diff --git a/bin/setup b/bin/setup new file mode 100644 index 0000000..ec47b79 --- /dev/null +++ b/bin/setup @@ -0,0 +1,33 @@ +#!/usr/bin/env ruby +require "fileutils" + +# path to your application root. +APP_ROOT = File.expand_path("..", __dir__) + +def system!(*args) + system(*args) || abort("\n== Command #{args} failed ==") +end + +FileUtils.chdir APP_ROOT do + # This script is a way to set up or update your development environment automatically. + # This script is idempotent, so that you can run it at any time and get an expectable outcome. + # Add necessary setup steps to this file. + + puts "== Installing dependencies ==" + system! "gem install bundler --conservative" + system("bundle check") || system!("bundle install") + + # puts "\n== Copying sample files ==" + # unless File.exist?("config/database.yml") + # FileUtils.cp "config/database.yml.sample", "config/database.yml" + # end + + puts "\n== Preparing database ==" + system! "bin/rails db:prepare" + + puts "\n== Removing old logs and tempfiles ==" + system! "bin/rails log:clear tmp:clear" + + puts "\n== Restarting application server ==" + system! "bin/rails restart" +end diff --git a/config.ru b/config.ru new file mode 100644 index 0000000..ad1fbf2 --- /dev/null +++ b/config.ru @@ -0,0 +1,6 @@ +# This file is used by Rack-based servers to start the application. + +require_relative 'config/environment' + +run Rails.application +Rails.application.load_server diff --git a/config/application.rb b/config/application.rb new file mode 100644 index 0000000..f75ef1f --- /dev/null +++ b/config/application.rb @@ -0,0 +1,33 @@ +require_relative "boot" + +require "rails/all" + +# Require the gems listed in Gemfile, including any gems +# you've limited to :test, :development, or :production. +Bundler.require(*Rails.groups) + +module AppointmentCapstoneBackend + class Application < Rails::Application + # Initialize configuration defaults for originally generated Rails version. + config.load_defaults 7.0 + + config.middleware.insert_before 0, Rack::Cors do + allow do + origins '*' + resource '*', + headers: :any, + methods: [:get, :post, :put, :patch, :delete, :options, :head], + expose: ['access-token', 'expiry', 'token-type', 'uid', 'client'], + max_age: 600 + end + end + + # Configuration for the application, engines, and railties goes here. + # + # These settings can be overridden in specific environments using the files + # in config/environments, which are processed later. + # + # config.time_zone = "Central Time (US & Canada)" + # config.eager_load_paths << Rails.root.join("extras") + end +end diff --git a/config/boot.rb b/config/boot.rb new file mode 100644 index 0000000..988a5dd --- /dev/null +++ b/config/boot.rb @@ -0,0 +1,4 @@ +ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../Gemfile", __dir__) + +require "bundler/setup" # Set up gems listed in the Gemfile. +require "bootsnap/setup" # Speed up boot time by caching expensive operations. diff --git a/config/cable.yml b/config/cable.yml new file mode 100644 index 0000000..35be99a --- /dev/null +++ b/config/cable.yml @@ -0,0 +1,10 @@ +development: + adapter: async + +test: + adapter: test + +production: + adapter: redis + url: <%= ENV.fetch("REDIS_URL") { "redis://localhost:6379/1" } %> + channel_prefix: Appointment_Capstone_Backend_production diff --git a/config/credentials.yml.enc b/config/credentials.yml.enc new file mode 100644 index 0000000..78b4f23 --- /dev/null +++ b/config/credentials.yml.enc @@ -0,0 +1 @@ +1xr7xCnyDtXDeU8lzfkBeuow7o+06Uwvwfa0i06k4g2B0E9Nm+8A5LZjoBdicSmuyrV9ooi5mgE/d+wAmvZ58DTdKmA7FUS8JCGoeuJt88Xltl0THdqzGS8EDHVo/yGFMcAJ73N/bEwVOaDiVcBFqtRX/utghOrgzm40EpP7iBcCkenGPApTotk9LpHMGMLF4Tz6uiIX6LHModzEIRwfkK+ncgYECtwqenR/Mp5FojkXE8FG0aI7052sGJIx6s/sKMqZjOHvgOCXG0g59+0TxQOYNcm4Ixx/ClvwE8kpa2yXczQmtSlZH+2QQkGeQz/VYMrs77BPkFuyvPSMzyOYrJy1IpOHjrzS+4iqDcXFT+PANHXO6LK4QmQL8DwBeVK01Zi/GoItTie06fzmFjFg07apieW66vLCD/0y--7ql0ads3tDuJoIEy--J0Afi6Yk+tR3oFciJEMoqA== \ No newline at end of file diff --git a/config/environment.rb b/config/environment.rb new file mode 100644 index 0000000..cac5315 --- /dev/null +++ b/config/environment.rb @@ -0,0 +1,5 @@ +# Load the Rails application. +require_relative "application" + +# Initialize the Rails application. +Rails.application.initialize! diff --git a/config/environments/development.rb b/config/environments/development.rb new file mode 100644 index 0000000..8500f45 --- /dev/null +++ b/config/environments/development.rb @@ -0,0 +1,70 @@ +require "active_support/core_ext/integer/time" + +Rails.application.configure do + # Settings specified here will take precedence over those in config/application.rb. + + # In the development environment your application's code is reloaded any time + # it changes. This slows down response time but is perfect for development + # since you don't have to restart the web server when you make code changes. + config.cache_classes = false + + # Do not eager load code on boot. + config.eager_load = false + + # Show full error reports. + config.consider_all_requests_local = true + + # Enable server timing + config.server_timing = true + + # Enable/disable caching. By default caching is disabled. + # Run rails dev:cache to toggle caching. + if Rails.root.join("tmp/caching-dev.txt").exist? + config.action_controller.perform_caching = true + config.action_controller.enable_fragment_cache_logging = true + + config.cache_store = :memory_store + config.public_file_server.headers = { + "Cache-Control" => "public, max-age=#{2.days.to_i}" + } + else + config.action_controller.perform_caching = false + + config.cache_store = :null_store + end + + # Store uploaded files on the local file system (see config/storage.yml for options). + config.active_storage.service = :local + + # Don't care if the mailer can't send. + config.action_mailer.raise_delivery_errors = false + + config.action_mailer.perform_caching = false + + # Print deprecation notices to the Rails logger. + config.active_support.deprecation = :log + + # Raise exceptions for disallowed deprecations. + config.active_support.disallowed_deprecation = :raise + + # Tell Active Support which deprecation messages to disallow. + config.active_support.disallowed_deprecation_warnings = [] + + # Raise an error on page load if there are pending migrations. + config.active_record.migration_error = :page_load + + # Highlight code that triggered database queries in logs. + config.active_record.verbose_query_logs = true + + # Suppress logger output for asset requests. + config.assets.quiet = true + + # Raises error for missing translations. + # config.i18n.raise_on_missing_translations = true + + # Annotate rendered view with file names. + # config.action_view.annotate_rendered_view_with_filenames = true + + # Uncomment if you wish to allow Action Cable access from any origin. + # config.action_cable.disable_request_forgery_protection = true +end diff --git a/config/environments/production.rb b/config/environments/production.rb new file mode 100644 index 0000000..caa56c8 --- /dev/null +++ b/config/environments/production.rb @@ -0,0 +1,93 @@ +require "active_support/core_ext/integer/time" + +Rails.application.configure do + # Settings specified here will take precedence over those in config/application.rb. + + # Code is not reloaded between requests. + config.cache_classes = true + + # Eager load code on boot. This eager loads most of Rails and + # your application in memory, allowing both threaded web servers + # and those relying on copy on write to perform better. + # Rake tasks automatically ignore this option for performance. + config.eager_load = true + + # Full error reports are disabled and caching is turned on. + config.consider_all_requests_local = false + config.action_controller.perform_caching = true + + # Ensures that a master key has been made available in either ENV["RAILS_MASTER_KEY"] + # or in config/master.key. This key is used to decrypt credentials (and other encrypted files). + # config.require_master_key = true + + # Disable serving static files from the `/public` folder by default since + # Apache or NGINX already handles this. + config.public_file_server.enabled = ENV["RAILS_SERVE_STATIC_FILES"].present? + + # Compress CSS using a preprocessor. + # config.assets.css_compressor = :sass + + # Do not fallback to assets pipeline if a precompiled asset is missed. + config.assets.compile = false + + # Enable serving of images, stylesheets, and JavaScripts from an asset server. + # config.asset_host = "http://assets.example.com" + + # Specifies the header that your server uses for sending files. + # config.action_dispatch.x_sendfile_header = "X-Sendfile" # for Apache + # config.action_dispatch.x_sendfile_header = "X-Accel-Redirect" # for NGINX + + # Store uploaded files on the local file system (see config/storage.yml for options). + config.active_storage.service = :local + + # Mount Action Cable outside main process or domain. + # config.action_cable.mount_path = nil + # config.action_cable.url = "wss://example.com/cable" + # config.action_cable.allowed_request_origins = [ "http://example.com", /http:\/\/example.*/ ] + + # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies. + # config.force_ssl = true + + # Include generic and useful information about system operation, but avoid logging too much + # information to avoid inadvertent exposure of personally identifiable information (PII). + config.log_level = :info + + # Prepend all log lines with the following tags. + config.log_tags = [ :request_id ] + + # Use a different cache store in production. + # config.cache_store = :mem_cache_store + + # Use a real queuing backend for Active Job (and separate queues per environment). + # config.active_job.queue_adapter = :resque + # config.active_job.queue_name_prefix = "Appointment_Capstone_Backend_production" + + config.action_mailer.perform_caching = false + + # Ignore bad email addresses and do not raise email delivery errors. + # Set this to true and configure the email server for immediate delivery to raise delivery errors. + # config.action_mailer.raise_delivery_errors = false + + # Enable locale fallbacks for I18n (makes lookups for any locale fall back to + # the I18n.default_locale when a translation cannot be found). + config.i18n.fallbacks = true + + # Don't log any deprecations. + config.active_support.report_deprecations = false + + # Use default logging formatter so that PID and timestamp are not suppressed. + config.log_formatter = ::Logger::Formatter.new + + # Use a different logger for distributed setups. + # require "syslog/logger" + # config.logger = ActiveSupport::TaggedLogging.new(Syslog::Logger.new "app-name") + + if ENV["RAILS_LOG_TO_STDOUT"].present? + logger = ActiveSupport::Logger.new(STDOUT) + logger.formatter = config.log_formatter + config.logger = ActiveSupport::TaggedLogging.new(logger) + end + + # Do not dump schema after migrations. + config.active_record.dump_schema_after_migration = false +end diff --git a/config/environments/test.rb b/config/environments/test.rb new file mode 100644 index 0000000..6ea4d1e --- /dev/null +++ b/config/environments/test.rb @@ -0,0 +1,60 @@ +require "active_support/core_ext/integer/time" + +# The test environment is used exclusively to run your application's +# test suite. You never need to work with it otherwise. Remember that +# your test database is "scratch space" for the test suite and is wiped +# and recreated between test runs. Don't rely on the data there! + +Rails.application.configure do + # Settings specified here will take precedence over those in config/application.rb. + + # Turn false under Spring and add config.action_view.cache_template_loading = true. + config.cache_classes = true + + # Eager loading loads your whole application. When running a single test locally, + # this probably isn't necessary. It's a good idea to do in a continuous integration + # system, or in some way before deploying your code. + config.eager_load = ENV["CI"].present? + + # Configure public file server for tests with Cache-Control for performance. + config.public_file_server.enabled = true + config.public_file_server.headers = { + "Cache-Control" => "public, max-age=#{1.hour.to_i}" + } + + # Show full error reports and disable caching. + config.consider_all_requests_local = true + config.action_controller.perform_caching = false + config.cache_store = :null_store + + # Raise exceptions instead of rendering exception templates. + config.action_dispatch.show_exceptions = false + + # Disable request forgery protection in test environment. + config.action_controller.allow_forgery_protection = false + + # Store uploaded files on the local file system in a temporary directory. + config.active_storage.service = :test + + config.action_mailer.perform_caching = false + + # Tell Action Mailer not to deliver emails to the real world. + # The :test delivery method accumulates sent emails in the + # ActionMailer::Base.deliveries array. + config.action_mailer.delivery_method = :test + + # Print deprecation notices to the stderr. + config.active_support.deprecation = :stderr + + # Raise exceptions for disallowed deprecations. + config.active_support.disallowed_deprecation = :raise + + # Tell Active Support which deprecation messages to disallow. + config.active_support.disallowed_deprecation_warnings = [] + + # Raises error for missing translations. + # config.i18n.raise_on_missing_translations = true + + # Annotate rendered view with file names. + # config.action_view.annotate_rendered_view_with_filenames = true +end diff --git a/config/importmap.rb b/config/importmap.rb new file mode 100644 index 0000000..8dce42d --- /dev/null +++ b/config/importmap.rb @@ -0,0 +1,7 @@ +# Pin npm packages by running ./bin/importmap + +pin "application", preload: true +pin "@hotwired/turbo-rails", to: "turbo.min.js", preload: true +pin "@hotwired/stimulus", to: "stimulus.min.js", preload: true +pin "@hotwired/stimulus-loading", to: "stimulus-loading.js", preload: true +pin_all_from "app/javascript/controllers", under: "controllers" diff --git a/config/initializers/assets.rb b/config/initializers/assets.rb new file mode 100644 index 0000000..2eeef96 --- /dev/null +++ b/config/initializers/assets.rb @@ -0,0 +1,12 @@ +# Be sure to restart your server when you modify this file. + +# Version of your assets, change this if you want to expire all your assets. +Rails.application.config.assets.version = "1.0" + +# Add additional assets to the asset load path. +# Rails.application.config.assets.paths << Emoji.images_path + +# Precompile additional assets. +# application.js, application.css, and all non-JS/CSS in the app/assets +# folder are already added. +# Rails.application.config.assets.precompile += %w( admin.js admin.css ) diff --git a/config/initializers/content_security_policy.rb b/config/initializers/content_security_policy.rb new file mode 100644 index 0000000..54f47cf --- /dev/null +++ b/config/initializers/content_security_policy.rb @@ -0,0 +1,25 @@ +# Be sure to restart your server when you modify this file. + +# Define an application-wide content security policy. +# See the Securing Rails Applications Guide for more information: +# https://guides.rubyonrails.org/security.html#content-security-policy-header + +# Rails.application.configure do +# config.content_security_policy do |policy| +# policy.default_src :self, :https +# policy.font_src :self, :https, :data +# policy.img_src :self, :https, :data +# policy.object_src :none +# policy.script_src :self, :https +# policy.style_src :self, :https +# # Specify URI for violation reports +# # policy.report_uri "/csp-violation-report-endpoint" +# end +# +# # Generate session nonces for permitted importmap and inline scripts +# config.content_security_policy_nonce_generator = ->(request) { request.session.id.to_s } +# config.content_security_policy_nonce_directives = %w(script-src) +# +# # Report violations without enforcing the policy. +# # config.content_security_policy_report_only = true +# end diff --git a/config/initializers/filter_parameter_logging.rb b/config/initializers/filter_parameter_logging.rb new file mode 100644 index 0000000..adc6568 --- /dev/null +++ b/config/initializers/filter_parameter_logging.rb @@ -0,0 +1,8 @@ +# Be sure to restart your server when you modify this file. + +# Configure parameters to be filtered from the log file. Use this to limit dissemination of +# sensitive information. See the ActiveSupport::ParameterFilter documentation for supported +# notations and behaviors. +Rails.application.config.filter_parameters += [ + :passw, :secret, :token, :_key, :crypt, :salt, :certificate, :otp, :ssn +] diff --git a/config/initializers/inflections.rb b/config/initializers/inflections.rb new file mode 100644 index 0000000..3860f65 --- /dev/null +++ b/config/initializers/inflections.rb @@ -0,0 +1,16 @@ +# Be sure to restart your server when you modify this file. + +# Add new inflection rules using the following format. Inflections +# are locale specific, and you may define rules for as many different +# locales as you wish. All of these examples are active by default: +# ActiveSupport::Inflector.inflections(:en) do |inflect| +# inflect.plural /^(ox)$/i, "\\1en" +# inflect.singular /^(ox)en/i, "\\1" +# inflect.irregular "person", "people" +# inflect.uncountable %w( fish sheep ) +# end + +# These inflection rules are supported but not enabled by default: +# ActiveSupport::Inflector.inflections(:en) do |inflect| +# inflect.acronym "RESTful" +# end diff --git a/config/initializers/permissions_policy.rb b/config/initializers/permissions_policy.rb new file mode 100644 index 0000000..00f64d7 --- /dev/null +++ b/config/initializers/permissions_policy.rb @@ -0,0 +1,11 @@ +# Define an application-wide HTTP permissions policy. For further +# information see https://developers.google.com/web/updates/2018/06/feature-policy +# +# Rails.application.config.permissions_policy do |f| +# f.camera :none +# f.gyroscope :none +# f.microphone :none +# f.usb :none +# f.fullscreen :self +# f.payment :self, "https://secure.example.com" +# end diff --git a/config/locales/en.yml b/config/locales/en.yml new file mode 100644 index 0000000..8ca56fc --- /dev/null +++ b/config/locales/en.yml @@ -0,0 +1,33 @@ +# Files in the config/locales directory are used for internationalization +# and are automatically loaded by Rails. If you want to use locales other +# than English, add the necessary files in this directory. +# +# To use the locales, use `I18n.t`: +# +# I18n.t "hello" +# +# In views, this is aliased to just `t`: +# +# <%= t("hello") %> +# +# To use a different locale, set it with `I18n.locale`: +# +# I18n.locale = :es +# +# This would use the information in config/locales/es.yml. +# +# The following keys must be escaped otherwise they will not be retrieved by +# the default I18n backend: +# +# true, false, on, off, yes, no +# +# Instead, surround them with single quotes. +# +# en: +# "true": "foo" +# +# To learn more, please read the Rails Internationalization guide +# available at https://guides.rubyonrails.org/i18n.html. + +en: + hello: "Hello world" diff --git a/config/puma.rb b/config/puma.rb new file mode 100644 index 0000000..824cc9a --- /dev/null +++ b/config/puma.rb @@ -0,0 +1,43 @@ +# Puma can serve each request in a thread from an internal thread pool. +# The `threads` method setting takes two numbers: a minimum and maximum. +# Any libraries that use thread pools should be configured to match +# the maximum value specified for Puma. Default is set to 5 threads for minimum +# and maximum; this matches the default thread size of Active Record. +# +max_threads_count = ENV.fetch("RAILS_MAX_THREADS") { 5 } +min_threads_count = ENV.fetch("RAILS_MIN_THREADS") { max_threads_count } +threads min_threads_count, max_threads_count + +# Specifies the `worker_timeout` threshold that Puma will use to wait before +# terminating a worker in development environments. +# +worker_timeout 3600 if ENV.fetch("RAILS_ENV", "development") == "development" + +# Specifies the `port` that Puma will listen on to receive requests; default is 3000. +# +port ENV.fetch("PORT") { 4000 } + +# Specifies the `environment` that Puma will run in. +# +environment ENV.fetch("RAILS_ENV") { "development" } + +# Specifies the `pidfile` that Puma will use. +pidfile ENV.fetch("PIDFILE") { "tmp/pids/server.pid" } + +# Specifies the number of `workers` to boot in clustered mode. +# Workers are forked web server processes. If using threads and workers together +# the concurrency of the application would be max `threads` * `workers`. +# Workers do not work on JRuby or Windows (both of which do not support +# processes). +# +# workers ENV.fetch("WEB_CONCURRENCY") { 0 } + +# Use the `preload_app!` method when specifying a `workers` number. +# This directive tells Puma to first boot the application and load code +# before forking the application. This takes advantage of Copy On Write +# process behavior so workers use less memory. +# +# preload_app! + +# Allow puma to be restarted by `bin/rails restart` command. +plugin :tmp_restart diff --git a/config/routes.rb b/config/routes.rb new file mode 100644 index 0000000..b7d43b6 --- /dev/null +++ b/config/routes.rb @@ -0,0 +1,13 @@ +Rails.application.routes.draw do + namespace :api do + namespace :v1 do + post 'auth/login', to: 'authentication#login' + post 'auth/register', to: 'authentication#register' + delete 'auth/logout', to: 'authentication#logout' + + resources :users + resources :reservations + resources :cars + end + end +end diff --git a/config/storage.yml b/config/storage.yml new file mode 100644 index 0000000..4942ab6 --- /dev/null +++ b/config/storage.yml @@ -0,0 +1,34 @@ +test: + service: Disk + root: <%= Rails.root.join("tmp/storage") %> + +local: + service: Disk + root: <%= Rails.root.join("storage") %> + +# Use bin/rails credentials:edit to set the AWS secrets (as aws:access_key_id|secret_access_key) +# amazon: +# service: S3 +# access_key_id: <%= Rails.application.credentials.dig(:aws, :access_key_id) %> +# secret_access_key: <%= Rails.application.credentials.dig(:aws, :secret_access_key) %> +# region: us-east-1 +# bucket: your_own_bucket-<%= Rails.env %> + +# Remember not to checkin your GCS keyfile to a repository +# google: +# service: GCS +# project: your_project +# credentials: <%= Rails.root.join("path/to/gcs.keyfile") %> +# bucket: your_own_bucket-<%= Rails.env %> + +# Use bin/rails credentials:edit to set the Azure Storage secret (as azure_storage:storage_access_key) +# microsoft: +# service: AzureStorage +# storage_account_name: your_account_name +# storage_access_key: <%= Rails.application.credentials.dig(:azure_storage, :storage_access_key) %> +# container: your_container_name-<%= Rails.env %> + +# mirror: +# service: Mirror +# primary: local +# mirrors: [ amazon, google, microsoft ] diff --git a/db/migrate/20230403002443_create_users.rb b/db/migrate/20230403002443_create_users.rb new file mode 100644 index 0000000..33b1925 --- /dev/null +++ b/db/migrate/20230403002443_create_users.rb @@ -0,0 +1,11 @@ +class CreateUsers < ActiveRecord::Migration[7.0] + def change + create_table :users do |t| + t.string :name + t.string :email + t.string :password_digest + + t.timestamps + end + end +end diff --git a/db/migrate/20230403002724_create_cars.rb b/db/migrate/20230403002724_create_cars.rb new file mode 100644 index 0000000..c8afa43 --- /dev/null +++ b/db/migrate/20230403002724_create_cars.rb @@ -0,0 +1,13 @@ +class CreateCars < ActiveRecord::Migration[7.0] + def change + create_table :cars do |t| + t.string :make + t.string :model + t.string :image + t.integer :year + t.integer :price + + t.timestamps + end + end +end diff --git a/db/migrate/20230403063146_create_reservations.rb b/db/migrate/20230403063146_create_reservations.rb new file mode 100644 index 0000000..8a049cc --- /dev/null +++ b/db/migrate/20230403063146_create_reservations.rb @@ -0,0 +1,10 @@ +class CreateReservations < ActiveRecord::Migration[7.0] + def change + create_table :reservations do |t| + t.date :date + t.time :time + + t.timestamps + end + end +end diff --git a/db/migrate/20230403220606_add_user_ref_to_cars.rb b/db/migrate/20230403220606_add_user_ref_to_cars.rb new file mode 100644 index 0000000..4cd46da --- /dev/null +++ b/db/migrate/20230403220606_add_user_ref_to_cars.rb @@ -0,0 +1,5 @@ +class AddUserRefToCars < ActiveRecord::Migration[7.0] + def change + add_reference :cars, :user, null: false, foreign_key: { to_table: 'users' } + end +end diff --git a/db/migrate/20230403220926_add_user_ref_to_reservations.rb b/db/migrate/20230403220926_add_user_ref_to_reservations.rb new file mode 100644 index 0000000..035ac19 --- /dev/null +++ b/db/migrate/20230403220926_add_user_ref_to_reservations.rb @@ -0,0 +1,5 @@ +class AddUserRefToReservations < ActiveRecord::Migration[7.0] + def change + add_reference :reservations, :user, null: false, foreign_key: { to_table: 'users' } + end +end diff --git a/db/migrate/20230403221124_add_car_ref_to_reservations.rb b/db/migrate/20230403221124_add_car_ref_to_reservations.rb new file mode 100644 index 0000000..973405a --- /dev/null +++ b/db/migrate/20230403221124_add_car_ref_to_reservations.rb @@ -0,0 +1,5 @@ +class AddCarRefToReservations < ActiveRecord::Migration[7.0] + def change + add_reference :reservations, :car, null: false, foreign_key: { to_table: 'cars' } + end +end diff --git a/db/migrate/20230406103115_rename_date_column_in_reservation.rb b/db/migrate/20230406103115_rename_date_column_in_reservation.rb new file mode 100644 index 0000000..6519bb8 --- /dev/null +++ b/db/migrate/20230406103115_rename_date_column_in_reservation.rb @@ -0,0 +1,5 @@ +class RenameDateColumnInReservation < ActiveRecord::Migration[7.0] + def change + rename_column :reservations, :date, :reservation_date + end +end diff --git a/db/migrate/20230406103829_remove_time_from_reservations.rb b/db/migrate/20230406103829_remove_time_from_reservations.rb new file mode 100644 index 0000000..5781e2f --- /dev/null +++ b/db/migrate/20230406103829_remove_time_from_reservations.rb @@ -0,0 +1,5 @@ +class RemoveTimeFromReservations < ActiveRecord::Migration[7.0] + def change + remove_column :reservations, :time, :time + end +end diff --git a/db/migrate/20230406104000_add_due_date_to_reservations.rb b/db/migrate/20230406104000_add_due_date_to_reservations.rb new file mode 100644 index 0000000..041abe8 --- /dev/null +++ b/db/migrate/20230406104000_add_due_date_to_reservations.rb @@ -0,0 +1,5 @@ +class AddDueDateToReservations < ActiveRecord::Migration[7.0] + def change + add_column :reservations, :due_date, :date + end +end diff --git a/db/migrate/20230410075041_add_role_to_user.rb b/db/migrate/20230410075041_add_role_to_user.rb new file mode 100644 index 0000000..af918a2 --- /dev/null +++ b/db/migrate/20230410075041_add_role_to_user.rb @@ -0,0 +1,5 @@ +class AddRoleToUser < ActiveRecord::Migration[7.0] + def change + add_column :users, :role, :string, null: false, default: 'user' + end +end diff --git a/db/migrate/20230412024730_add_description_to_cars.rb b/db/migrate/20230412024730_add_description_to_cars.rb new file mode 100644 index 0000000..71ed36a --- /dev/null +++ b/db/migrate/20230412024730_add_description_to_cars.rb @@ -0,0 +1,5 @@ +class AddDescriptionToCars < ActiveRecord::Migration[7.0] + def change + add_column :cars, :description, :text + end +end diff --git a/db/migrate/20230412024834_remove_price_from_cars.rb b/db/migrate/20230412024834_remove_price_from_cars.rb new file mode 100644 index 0000000..69fe8d1 --- /dev/null +++ b/db/migrate/20230412024834_remove_price_from_cars.rb @@ -0,0 +1,5 @@ +class RemovePriceFromCars < ActiveRecord::Migration[7.0] + def change + remove_column :cars, :price + end +end diff --git a/db/migrate/20230412024928_add_daily_rate_to_cars.rb b/db/migrate/20230412024928_add_daily_rate_to_cars.rb new file mode 100644 index 0000000..ddd61f2 --- /dev/null +++ b/db/migrate/20230412024928_add_daily_rate_to_cars.rb @@ -0,0 +1,5 @@ +class AddDailyRateToCars < ActiveRecord::Migration[7.0] + def change + add_column :cars, :daily_rate, :integer + end +end diff --git a/db/schema.rb b/db/schema.rb new file mode 100644 index 0000000..2bbf7b4 --- /dev/null +++ b/db/schema.rb @@ -0,0 +1,53 @@ +# This file is auto-generated from the current state of the database. Instead +# of editing this file, please use the migrations feature of Active Record to +# incrementally modify your database, and then regenerate this schema definition. +# +# This file is the source Rails uses to define your schema when running `bin/rails +# db:schema:load`. When creating a new database, `bin/rails db:schema:load` tends to +# be faster and is potentially less error prone than running all of your +# migrations from scratch. Old migrations may fail to apply correctly if those +# migrations use external dependencies or application code. +# +# It's strongly recommended that you check this file into your version control system. + +ActiveRecord::Schema[7.0].define(version: 2023_04_12_024928) do + # These are extensions that must be enabled in order to support this database + enable_extension "plpgsql" + + create_table "cars", force: :cascade do |t| + t.string "make" + t.string "model" + t.string "image" + t.integer "year" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + t.bigint "user_id", null: false + t.text "description" + t.integer "daily_rate" + t.index ["user_id"], name: "index_cars_on_user_id" + end + + create_table "reservations", force: :cascade do |t| + t.date "reservation_date" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + t.bigint "user_id", null: false + t.bigint "car_id", null: false + t.date "due_date" + t.index ["car_id"], name: "index_reservations_on_car_id" + t.index ["user_id"], name: "index_reservations_on_user_id" + end + + create_table "users", force: :cascade do |t| + t.string "name" + t.string "email" + t.string "password_digest" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + t.string "role", default: "user", null: false + end + + add_foreign_key "cars", "users" + add_foreign_key "reservations", "cars" + add_foreign_key "reservations", "users" +end diff --git a/db/seeds.rb b/db/seeds.rb new file mode 100644 index 0000000..3d33248 --- /dev/null +++ b/db/seeds.rb @@ -0,0 +1,14 @@ +# This file should contain all the record creation needed to seed the database with its default values. +# The data can then be loaded with the bin/rails db:seed command (or created alongside the database with db:setup). +# +# Examples: +# +# movies = Movie.create([{ name: "Star Wars" }, { name: "Lord of the Rings" }]) +# Character.create(name: "Luke", movie: movies.first) + +car1 = Car.create(make: "Toyota", model: "Corolla", year: 2018, user_id: 1, image: "https://st2.depositphotos.com/3037725/45775/i/600/depositphotos_457759302-stock-photo-tula-russia-february-2021-toyota.jpg", + daily_rate: 2500, description: "The Toyota Corolla is a reliable and stylish compact sedan that is perfect for daily commutes or longer road trips. This 2018 model features a sleek exterior design that is both sporty and refined, and its interior is spacious and comfortable. With a daily rental rate of 2500, you can enjoy all the benefits of this well-rounded vehicle without breaking the bank.") +car2 = Car.create(make: "Honda", model: "Civic", year: 2019, user_id: 1, image: "https://images.hgmsites.net/lrg/2020-honda-civic-sport-manual-angular-front-exterior-view_100751892_l.jpg", + daily_rate: 2000, description: "The Honda Civic is a popular and versatile compact car that offers excellent performance, fuel efficiency, and safety features. This 2019 model boasts a striking exterior design that is sure to turn heads, and its comfortable and well-equipped interior makes it a joy to drive. At a daily rental rate of 2000, you can experience the quality and reliability of this top-rated vehicle without having to make a long-term commitment.") +Reservation.create(reservation_date: "2023-04-10", user_id: 1, car_id: car1.id, due_date: "2023-04-12") +Reservation.create(reservation_date: "2023-04-14", user_id: 1, car_id: car2.id, due_date: "2023-04-16") \ No newline at end of file diff --git a/lib/assets/.keep b/lib/assets/.keep new file mode 100644 index 0000000..e69de29 diff --git a/lib/tasks/.keep b/lib/tasks/.keep new file mode 100644 index 0000000..e69de29 diff --git a/log/.keep b/log/.keep new file mode 100644 index 0000000..e69de29 diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..0b33ccd --- /dev/null +++ b/package-lock.json @@ -0,0 +1,13 @@ +{ + "name": "appointment-capstone-backend", + "version": "1.0.0", + "lockfileVersion": 2, + "requires": true, + "packages": { + "": { + "name": "appointment-capstone-backend", + "version": "1.0.0", + "license": "ISC" + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..55b8faf --- /dev/null +++ b/package.json @@ -0,0 +1,23 @@ +{ + "name": "appointment-capstone-backend", + "version": "1.0.0", + "description": "", + "main": "index.js", + "directories": { + "lib": "lib", + "test": "test" + }, + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "repository": { + "type": "git", + "url": "git+https://github.com/vanelnw/Appointment-Capstone-Backend.git" + }, + "author": "", + "license": "ISC", + "bugs": { + "url": "https://github.com/vanelnw/Appointment-Capstone-Backend/issues" + }, + "homepage": "https://github.com/vanelnw/Appointment-Capstone-Backend#readme" +} diff --git a/public/404.html b/public/404.html new file mode 100644 index 0000000..2be3af2 --- /dev/null +++ b/public/404.html @@ -0,0 +1,67 @@ + + + + The page you were looking for doesn't exist (404) + + + + + + +
+
+

The page you were looking for doesn't exist.

+

You may have mistyped the address or the page may have moved.

+
+

If you are the application owner check the logs for more information.

+
+ + diff --git a/public/422.html b/public/422.html new file mode 100644 index 0000000..c08eac0 --- /dev/null +++ b/public/422.html @@ -0,0 +1,67 @@ + + + + The change you wanted was rejected (422) + + + + + + +
+
+

The change you wanted was rejected.

+

Maybe you tried to change something you didn't have access to.

+
+

If you are the application owner check the logs for more information.

+
+ + diff --git a/public/500.html b/public/500.html new file mode 100644 index 0000000..78a030a --- /dev/null +++ b/public/500.html @@ -0,0 +1,66 @@ + + + + We're sorry, but something went wrong (500) + + + + + + +
+
+

We're sorry, but something went wrong.

+
+

If you are the application owner check the logs for more information.

+
+ + diff --git a/public/apple-touch-icon-precomposed.png b/public/apple-touch-icon-precomposed.png new file mode 100644 index 0000000..e69de29 diff --git a/public/apple-touch-icon.png b/public/apple-touch-icon.png new file mode 100644 index 0000000..e69de29 diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000..e69de29 diff --git a/public/robots.txt b/public/robots.txt new file mode 100644 index 0000000..c19f78a --- /dev/null +++ b/public/robots.txt @@ -0,0 +1 @@ +# See https://www.robotstxt.org/robotstxt.html for documentation on how to use the robots.txt file diff --git a/spec/models/cars_spec.rb b/spec/models/cars_spec.rb new file mode 100644 index 0000000..d862b7a --- /dev/null +++ b/spec/models/cars_spec.rb @@ -0,0 +1,45 @@ +require 'rails_helper' + +RSpec.describe Car, type: :model do + let(:user) { User.create(name: 'Test User', email: 'test@example.com', password: 'password') } + subject do + Car.new(make: 'Ford', model: 'Fiesta', year: 2021, daily_rate: 50, user_id: user.id, image: 'car.jpg', + description: 'A fun car to drive') + end + + describe 'validations' do + it 'is valid with valid attributes' do + expect(subject).to be_valid + end + + it 'is not valid without a make' do + subject.make = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a model' do + subject.model = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a year' do + subject.year = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a daily rate' do + subject.daily_rate = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a image' do + subject.image = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a user' do + subject.user_id = nil + expect(subject).not_to be_valid + end + end +end diff --git a/spec/models/reservations_spec.rb b/spec/models/reservations_spec.rb new file mode 100644 index 0000000..3b0dc57 --- /dev/null +++ b/spec/models/reservations_spec.rb @@ -0,0 +1,36 @@ +require 'rails_helper' + +RSpec.describe Reservation, type: :model do + let(:user) { User.create(name: 'Test User', email: 'test@example.com', password: 'password') } + let(:car) do + Car.create(make: 'Ford', model: 'Fiesta', year: 2021, daily_rate: 50, user_id: user.id, image: 'car.jpg', + description: 'A fun car to drive') + end + subject { Reservation.new(reservation_date: '2023-04-10', due_date: '2023-04-15', user_id: user.id, car_id: car.id) } + + describe 'validations' do + it 'is valid with valid attributes' do + expect(subject).to be_valid + end + + it 'is not valid without a reservation date' do + subject.reservation_date = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a due date' do + subject.due_date = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a car' do + subject.car_id = nil + expect(subject).not_to be_valid + end + + it 'is not valid without a user' do + subject.user_id = nil + expect(subject).not_to be_valid + end + end +end diff --git a/spec/models/users_spec.rb b/spec/models/users_spec.rb new file mode 100644 index 0000000..320b3c7 --- /dev/null +++ b/spec/models/users_spec.rb @@ -0,0 +1,32 @@ +require 'rails_helper' + +RSpec.describe User, type: :model do + subject { User.new(name: 'Test User', email: 'test@example.com', password: 'password') } + + describe 'validations' do + it 'is valid with valid attributes' do + expect(subject).to be_valid + end + + it 'is not valid without a name' do + subject.name = nil + expect(subject).not_to be_valid + end + + it 'is not valid without an email' do + subject.email = nil + expect(subject).not_to be_valid + end + + it 'is not valid with a duplicate email' do + User.create(name: 'Another User', email: 'test@example.com', password: 'password123') + expect(subject).not_to be_valid + end + end + + describe 'password encryption' do + it 'encrypts the password' do + expect(subject.password_digest).not_to eq('password') + end + end +end diff --git a/spec/rails_helper.rb b/spec/rails_helper.rb new file mode 100644 index 0000000..ce37f7e --- /dev/null +++ b/spec/rails_helper.rb @@ -0,0 +1,63 @@ +# This file is copied to spec/ when you run 'rails generate rspec:install' +require 'spec_helper' +ENV['RAILS_ENV'] ||= 'test' +require_relative '../config/environment' +# Prevent database truncation if the environment is production +abort('The Rails environment is running in production mode!') if Rails.env.production? +require 'rspec/rails' +# Add additional requires below this line. Rails is not loaded until this point! + +# Requires supporting ruby files with custom matchers and macros, etc, in +# spec/support/ and its subdirectories. Files matching `spec/**/*_spec.rb` are +# run as spec files by default. This means that files in spec/support that end +# in _spec.rb will both be required and run as specs, causing the specs to be +# run twice. It is recommended that you do not name files matching this glob to +# end with _spec.rb. You can configure this pattern with the --pattern +# option on the command line or in ~/.rspec, .rspec or `.rspec-local`. +# +# The following line is provided for convenience purposes. It has the downside +# of increasing the boot-up time by auto-requiring all files in the support +# directory. Alternatively, in the individual `*_spec.rb` files, manually +# require only the support files necessary. +# +# Dir[Rails.root.join('spec', 'support', '**', '*.rb')].sort.each { |f| require f } + +# Checks for pending migrations and applies them before tests are run. +# If you are not using ActiveRecord, you can remove these lines. +begin + ActiveRecord::Migration.maintain_test_schema! +rescue ActiveRecord::PendingMigrationError => e + abort e.to_s.strip +end +RSpec.configure do |config| + # Remove this line if you're not using ActiveRecord or ActiveRecord fixtures + config.fixture_path = "#{Rails.root}/spec/fixtures" + + # If you're not using ActiveRecord, or you'd prefer not to run each of your + # examples within a transaction, remove the following line or assign false + # instead of true. + config.use_transactional_fixtures = true + + # You can uncomment this line to turn off ActiveRecord support entirely. + # config.use_active_record = false + + # RSpec Rails can automatically mix in different behaviours to your tests + # based on their file location, for example enabling you to call `get` and + # `post` in specs under `spec/controllers`. + # + # You can disable this behaviour by removing the line below, and instead + # explicitly tag your specs with their type, e.g.: + # + # RSpec.describe UsersController, type: :controller do + # # ... + # end + # + # The different available types are documented in the features, such as in + # https://relishapp.com/rspec/rspec-rails/docs + config.infer_spec_type_from_file_location! + + # Filter lines from Rails gems in backtraces. + config.filter_rails_from_backtrace! + # arbitrary gems may also be filtered via: + # config.filter_gems_from_backtrace("gem name") +end diff --git a/spec/requests/authentication_spec.rb b/spec/requests/authentication_spec.rb new file mode 100644 index 0000000..bdd6297 --- /dev/null +++ b/spec/requests/authentication_spec.rb @@ -0,0 +1,33 @@ +require 'rails_helper' + +RSpec.describe Api::V1::AuthenticationController, type: :request do + describe 'POST authentication' do + let(:signup_params) { { name: 'Test User', email: 'test@example.com', password: 'password' } } + let(:login_params) { { email: 'test@example.com', password: signup_params[:password] } } + + before do + post '/api/v1/auth/register', params: signup_params.to_json, headers: { 'Content-Type' => 'application/json' } + end + + it 'creates a user' do + json_response = JSON.parse(response.body) + expect(json_response['user']['name']).to eq('Test User') + expect(json_response['user']['email']).to eq('test@example.com') + end + + it 'returns a success message' do + json_response = JSON.parse(response.body) + expect(json_response['message']).to eq('Registered successfully!') + end + + it 'returns a status code 201' do + expect(response).to have_http_status(:created) + end + + it 'logins the user' do + post '/api/v1/auth/login', params: login_params.to_json, headers: { 'Content-Type' => 'application/json' } + json_response = JSON.parse(response.body) + expect(json_response['message']).to eq('Logged in successfully!') + end + end +end diff --git a/spec/requests/cars_spec.rb b/spec/requests/cars_spec.rb new file mode 100644 index 0000000..7a987b7 --- /dev/null +++ b/spec/requests/cars_spec.rb @@ -0,0 +1,104 @@ +require 'rails_helper' + +RSpec.describe Api::V1::CarsController, type: :request do + describe 'GET /api/v1/cars' do + let(:user) { User.create(name: 'user1', email: 'user1@example.com', password_digest: 'password') } + let(:payload) { { user_id: user.id } } + let(:token) { JWT.encode(payload, Rails.application.secret_key_base) } + let(:headers) { { 'Authorization' => "Bearer #{token}" } } + let(:car1) do + Car.create(make: 'Ford', model: 'Mustang', year: 2020, user_id: user.id, image: 'ford.jpg', daily_rate: 2000, + description: 'Car for testing') + end + let(:car2) do + Car.create(make: 'Chevrolet', model: 'Camaro', year: 2021, user_id: user.id, image: 'chevrolet.jpg', + daily_rate: 2500, description: 'Car for testing') + end + + before do + car1 + car2 + get '/api/v1/cars', headers: + end + + it 'returns a success response' do + expect(response).to have_http_status(:ok) + end + + it 'returns all cars' do + json_response = JSON.parse(response.body) + expect(json_response.size).to eq(2) + end + end + + describe 'GET /api/v1/cars/:id' do + it 'renders a JSON response with the car' do + user = User.create(name: 'user1', email: 'user1@example.com', password_digest: 'password') + payload = { user_id: user.id } + token = JWT.encode(payload, Rails.application.secret_key_base) + headers = { 'Authorization' => "Bearer #{token}" } + car = Car.create(make: 'Ford', model: 'Mustang', year: 2020, user_id: user.id, image: 'ford.jpg', + daily_rate: 2000, description: 'Car for testing') + + get("/api/v1/cars/#{car.id}", headers:) + + expect(response).to have_http_status(:ok) + expect(response.content_type).to eq('application/json; charset=utf-8') + expect(response.body).to include(car.to_json) + end + end + + describe 'POST /api/v1/cars/' do + let(:user) { User.create(name: 'user1', email: 'user1@example.com', password_digest: 'password') } + let(:valid_attributes) do + { make: 'Ford', model: 'Mustang', year: 2020, user_id: user.id, image: 'ford.jpg', daily_rate: 2000, + description: 'Car for testing' } + end + + context 'when the request is valid' do + before do + token = JWT.encode({ user_id: user.id }, Rails.application.secret_key_base) + post '/api/v1/cars', params: valid_attributes.to_json, + headers: { 'Authorization' => "Bearer #{token}", 'Content-Type' => 'application/json' } + end + + it 'creates a car' do + json_response = JSON.parse(response.body) + expect(json_response['car']['make']).to eq('Ford') + expect(json_response['car']['model']).to eq('Mustang') + expect(json_response['car']['year']).to eq(2020) + expect(json_response['car']['user_id']).to eq(user.id) + expect(json_response['car']['image']).to eq('ford.jpg') + expect(json_response['car']['daily_rate']).to eq(2000) + expect(json_response['car']['description']).to eq('Car for testing') + end + + it 'returns a success message' do + json_response = JSON.parse(response.body) + expect(json_response['message']).to eq('Car added successfully!') + end + + it 'returns a status code 201' do + expect(response).to have_http_status(:created) + end + end + end + + describe 'DELETE /api/v1/cars/:id' do + let(:user) { User.create(name: 'user1', email: 'user1@example.com', password_digest: 'password') } + let(:payload) { { user_id: user.id } } + let(:token) { JWT.encode(payload, Rails.application.secret_key_base) } + let(:headers) { { 'Authorization' => "Bearer #{token}" } } + + let!(:car) do + Car.create(make: 'Ford', model: 'Mustang', year: 2020, user_id: user.id, image: 'ford.jpg', daily_rate: 2000, + description: 'Car for testing') + end + + it 'deletes the car' do + expect do + delete "/api/v1/cars/#{car.id}", headers: + end.to change(Car, :count).by(-1) + end + end +end diff --git a/spec/requests/reservations_spec.rb b/spec/requests/reservations_spec.rb new file mode 100644 index 0000000..f64e6fc --- /dev/null +++ b/spec/requests/reservations_spec.rb @@ -0,0 +1,63 @@ +require 'rails_helper' + +RSpec.describe Api::V1::ReservationsController, type: :request do + describe 'GET /api/v1/reservations' do + let(:user) { User.create(name: 'user1', email: 'user1@example.com', password_digest: 'password') } + let(:payload) { { user_id: user.id } } + let(:token) { JWT.encode(payload, Rails.application.secret_key_base) } + let(:headers) { { 'Authorization' => "Bearer #{token}" } } + let(:car) do + Car.create(make: 'Ford', model: 'Mustang', year: 2020, user_id: user.id, image: 'ford.jpg', daily_rate: 2000, + description: 'Car for testing') + end + let(:reservation) do + Reservation.create(reservation_date: '2023-04-10', due_date: '2023-04-15', user_id: user.id, car_id: car.id) + end + + before { get '/api/v1/reservations', headers: } + + it 'returns a success response' do + expect(response).to have_http_status(:ok) + end + + it 'returns all reservations' do + json_response = JSON.parse(response.body) + expect(json_response.size).to eq(1) + end + end + + describe 'POST /api/v1/reservations' do + let(:user) { User.create(name: 'user1', email: 'user1@example.com', password_digest: 'password') } + let(:car) do + Car.create(make: 'Ford', model: 'Mustang', year: 2020, user_id: user.id, image: 'ford.jpg', daily_rate: 2000, + description: 'Car for testing') + end + let(:valid_attributes) { { reservation_date: Date.today + 1, due_date: Date.today + 7, car_id: car.id } } + + context 'when the request is valid' do + before do + token = JWT.encode({ user_id: user.id }, Rails.application.secret_key_base) + post '/api/v1/reservations', params: valid_attributes.to_json, + headers: { 'Authorization' => "Bearer #{token}", + 'Content-Type' => 'application/json' } + end + + it 'creates a reservation' do + json_response = JSON.parse(response.body) + expect(json_response['reservation']['reservation_date']).to eq((Date.today + 1).strftime('%Y-%m-%d')) + expect(json_response['reservation']['due_date']).to eq((Date.today + 7).strftime('%Y-%m-%d')) + expect(json_response['reservation']['user_id']).to eq(user.id) + expect(json_response['reservation']['car_id']).to eq(car.id) + end + + it 'returns a success message' do + json_response = JSON.parse(response.body) + expect(json_response['message']).to eq('Car reserved successfully!') + end + + it 'returns a status code 201' do + expect(response).to have_http_status(:created) + end + end + end +end diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb new file mode 100644 index 0000000..d0cef33 --- /dev/null +++ b/spec/spec_helper.rb @@ -0,0 +1,92 @@ +# This file was generated by the `rails generate rspec:install` command. Conventionally, all +# specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`. +# The generated `.rspec` file contains `--require spec_helper` which will cause +# this file to always be loaded, without a need to explicitly require it in any +# files. +# +# Given that it is always loaded, you are encouraged to keep this file as +# light-weight as possible. Requiring heavyweight dependencies from this file +# will add to the boot time of your test suite on EVERY test run, even for an +# individual file that may not need all of that loaded. Instead, consider making +# a separate helper file that requires the additional dependencies and performs +# the additional setup, and require it from the spec files that actually need +# it. +# +# See https://rubydoc.info/gems/rspec-core/RSpec/Core/Configuration +RSpec.configure do |config| + # rspec-expectations config goes here. You can use an alternate + # assertion/expectation library such as wrong or the stdlib/minitest + # assertions if you prefer. + config.expect_with :rspec do |expectations| + # This option will default to `true` in RSpec 4. It makes the `description` + # and `failure_message` of custom matchers include text for helper methods + # defined using `chain`, e.g.: + # be_bigger_than(2).and_smaller_than(4).description + # # => "be bigger than 2 and smaller than 4" + # ...rather than: + # # => "be bigger than 2" + expectations.include_chain_clauses_in_custom_matcher_descriptions = true + end + + # rspec-mocks config goes here. You can use an alternate test double + # library (such as bogus or mocha) by changing the `mock_with` option here. + config.mock_with :rspec do |mocks| + # Prevents you from mocking or stubbing a method that does not exist on + # a real object. This is generally recommended, and will default to + # `true` in RSpec 4. + mocks.verify_partial_doubles = true + end + + # This option will default to `:apply_to_host_groups` in RSpec 4 (and will + # have no way to turn it off -- the option exists only for backwards + # compatibility in RSpec 3). It causes shared context metadata to be + # inherited by the metadata hash of host groups and examples, rather than + # triggering implicit auto-inclusion in groups with matching metadata. + config.shared_context_metadata_behavior = :apply_to_host_groups + + # The settings below are suggested to provide a good initial experience + # with RSpec, but feel free to customize to your heart's content. + # # This allows you to limit a spec run to individual examples or groups + # # you care about by tagging them with `:focus` metadata. When nothing + # # is tagged with `:focus`, all examples get run. RSpec also provides + # # aliases for `it`, `describe`, and `context` that include `:focus` + # # metadata: `fit`, `fdescribe` and `fcontext`, respectively. + # config.filter_run_when_matching :focus + # + # # Allows RSpec to persist some state between runs in order to support + # # the `--only-failures` and `--next-failure` CLI options. We recommend + # # you configure your source control system to ignore this file. + # config.example_status_persistence_file_path = "spec/examples.txt" + # + # # Limits the available syntax to the non-monkey patched syntax that is + # # recommended. For more details, see: + # # https://relishapp.com/rspec/rspec-core/docs/configuration/zero-monkey-patching-mode + # config.disable_monkey_patching! + # + # # Many RSpec users commonly either run the entire suite or an individual + # # file, and it's useful to allow more verbose output when running an + # # individual spec file. + # if config.files_to_run.one? + # # Use the documentation formatter for detailed output, + # # unless a formatter has already been configured + # # (e.g. via a command-line flag). + # config.default_formatter = "doc" + # end + # + # # Print the 10 slowest examples and example groups at the + # # end of the spec run, to help surface which specs are running + # # particularly slow. + # config.profile_examples = 10 + # + # # Run specs in random order to surface order dependencies. If you find an + # # order dependency and want to debug it, you can fix the order by providing + # # the seed, which is printed after each run. + # # --seed 1234 + # config.order = :random + # + # # Seed global randomization in this process using the `--seed` CLI option. + # # Setting this allows you to use `--seed` to deterministically reproduce + # # test failures related to randomization by passing the same `--seed` value + # # as the one that triggered the failure. + # Kernel.srand config.seed +end diff --git a/storage/.keep b/storage/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/application_system_test_case.rb b/test/application_system_test_case.rb new file mode 100644 index 0000000..23701b4 --- /dev/null +++ b/test/application_system_test_case.rb @@ -0,0 +1,5 @@ +require 'test_helper' + +class ApplicationSystemTestCase < ActionDispatch::SystemTestCase + driven_by :selenium, using: :chrome, screen_size: [1400, 1400] +end diff --git a/test/channels/application_cable/connection_test.rb b/test/channels/application_cable/connection_test.rb new file mode 100644 index 0000000..d05dbd2 --- /dev/null +++ b/test/channels/application_cable/connection_test.rb @@ -0,0 +1,11 @@ +require 'test_helper' + +class ApplicationCable::ConnectionTest < ActionCable::Connection::TestCase + # test "connects with cookies" do + # cookies.signed[:user_id] = 42 + # + # connect + # + # assert_equal connection.user_id, "42" + # end +end diff --git a/test/controllers/.keep b/test/controllers/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/controllers/appointments_controller_test.rb b/test/controllers/appointments_controller_test.rb new file mode 100644 index 0000000..305fd79 --- /dev/null +++ b/test/controllers/appointments_controller_test.rb @@ -0,0 +1,52 @@ +require 'test_helper' + +class AppointmentsControllerTest < ActionDispatch::IntegrationTest + setup do + @appointment = appointments(:one) + end + + test 'should get index' do + get appointments_url + assert_response :success + end + + test 'should get new' do + get new_appointment_url + assert_response :success + end + + test 'should create appointment' do + assert_difference('Appointment.count') do + post appointments_url, + params: { appointment: { appointment_date: @appointment.appointment_date, car_id: @appointment.car_id, + user_id: @appointment.user_id } } + end + + assert_redirected_to appointment_url(Appointment.last) + end + + test 'should show appointment' do + get appointment_url(@appointment) + assert_response :success + end + + test 'should get edit' do + get edit_appointment_url(@appointment) + assert_response :success + end + + test 'should update appointment' do + patch appointment_url(@appointment), + params: { appointment: { appointment_date: @appointment.appointment_date, car_id: @appointment.car_id, + user_id: @appointment.user_id } } + assert_redirected_to appointment_url(@appointment) + end + + test 'should destroy appointment' do + assert_difference('Appointment.count', -1) do + delete appointment_url(@appointment) + end + + assert_redirected_to appointments_url + end +end diff --git a/test/controllers/cars_controller_test.rb b/test/controllers/cars_controller_test.rb new file mode 100644 index 0000000..c518977 --- /dev/null +++ b/test/controllers/cars_controller_test.rb @@ -0,0 +1,48 @@ +require 'test_helper' + +class CarsControllerTest < ActionDispatch::IntegrationTest + setup do + @car = cars(:one) + end + + test 'should get index' do + get cars_url + assert_response :success + end + + test 'should get new' do + get new_car_url + assert_response :success + end + + test 'should create car' do + assert_difference('Car.count') do + post cars_url, params: { car: { make: @car.make, model: @car.model, price: @car.price, year: @car.year } } + end + + assert_redirected_to car_url(Car.last) + end + + test 'should show car' do + get car_url(@car) + assert_response :success + end + + test 'should get edit' do + get edit_car_url(@car) + assert_response :success + end + + test 'should update car' do + patch car_url(@car), params: { car: { make: @car.make, model: @car.model, price: @car.price, year: @car.year } } + assert_redirected_to car_url(@car) + end + + test 'should destroy car' do + assert_difference('Car.count', -1) do + delete car_url(@car) + end + + assert_redirected_to cars_url + end +end diff --git a/test/controllers/reservations_controller_test.rb b/test/controllers/reservations_controller_test.rb new file mode 100644 index 0000000..7e4a9a6 --- /dev/null +++ b/test/controllers/reservations_controller_test.rb @@ -0,0 +1,52 @@ +require 'test_helper' + +class ReservationsControllerTest < ActionDispatch::IntegrationTest + setup do + @reservation = reservations(:one) + end + + test 'should get index' do + get reservations_url + assert_response :success + end + + test 'should get new' do + get new_reservation_url + assert_response :success + end + + test 'should create reservation' do + assert_difference('Reservation.count') do + post reservations_url, + params: { reservation: { appointment_id: @reservation.appointment_id, + reservation_date: @reservation.reservation_date } } + end + + assert_redirected_to reservation_url(Reservation.last) + end + + test 'should show reservation' do + get reservation_url(@reservation) + assert_response :success + end + + test 'should get edit' do + get edit_reservation_url(@reservation) + assert_response :success + end + + test 'should update reservation' do + patch reservation_url(@reservation), + params: { reservation: { appointment_id: @reservation.appointment_id, + reservation_date: @reservation.reservation_date } } + assert_redirected_to reservation_url(@reservation) + end + + test 'should destroy reservation' do + assert_difference('Reservation.count', -1) do + delete reservation_url(@reservation) + end + + assert_redirected_to reservations_url + end +end diff --git a/test/controllers/users_controller_test.rb b/test/controllers/users_controller_test.rb new file mode 100644 index 0000000..c83c872 --- /dev/null +++ b/test/controllers/users_controller_test.rb @@ -0,0 +1,48 @@ +require 'test_helper' + +class UsersControllerTest < ActionDispatch::IntegrationTest + setup do + @user = users(:one) + end + + test 'should get index' do + get users_url + assert_response :success + end + + test 'should get new' do + get new_user_url + assert_response :success + end + + test 'should create user' do + assert_difference('User.count') do + post users_url, params: { user: { email: @user.email, name: @user.name, phone: @user.phone } } + end + + assert_redirected_to user_url(User.last) + end + + test 'should show user' do + get user_url(@user) + assert_response :success + end + + test 'should get edit' do + get edit_user_url(@user) + assert_response :success + end + + test 'should update user' do + patch user_url(@user), params: { user: { email: @user.email, name: @user.name, phone: @user.phone } } + assert_redirected_to user_url(@user) + end + + test 'should destroy user' do + assert_difference('User.count', -1) do + delete user_url(@user) + end + + assert_redirected_to users_url + end +end diff --git a/test/fixtures/cars.yml b/test/fixtures/cars.yml new file mode 100644 index 0000000..5088eec --- /dev/null +++ b/test/fixtures/cars.yml @@ -0,0 +1,13 @@ +# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html + +one: + make: MyString + model: MyString + year: 1 + price: 1 + +two: + make: MyString + model: MyString + year: 1 + price: 1 diff --git a/test/fixtures/files/.keep b/test/fixtures/files/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/reservations.yml b/test/fixtures/reservations.yml new file mode 100644 index 0000000..c28f598 --- /dev/null +++ b/test/fixtures/reservations.yml @@ -0,0 +1,9 @@ +# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html + +one: + reservation_date: 2023-04-02 + appointment: one + +two: + reservation_date: 2023-04-02 + appointment: two diff --git a/test/fixtures/users.yml b/test/fixtures/users.yml new file mode 100644 index 0000000..6cfbfa4 --- /dev/null +++ b/test/fixtures/users.yml @@ -0,0 +1,11 @@ +# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html + +one: + name: MyString + email: MyString + phone: MyString + +two: + name: MyString + email: MyString + phone: MyString diff --git a/test/helpers/.keep b/test/helpers/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/integration/.keep b/test/integration/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/mailers/.keep b/test/mailers/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/models/.keep b/test/models/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/models/car_test.rb b/test/models/car_test.rb new file mode 100644 index 0000000..39bdaec --- /dev/null +++ b/test/models/car_test.rb @@ -0,0 +1,7 @@ +require 'test_helper' + +class CarTest < ActiveSupport::TestCase + # test "the truth" do + # assert true + # end +end diff --git a/test/models/reservation_test.rb b/test/models/reservation_test.rb new file mode 100644 index 0000000..391559f --- /dev/null +++ b/test/models/reservation_test.rb @@ -0,0 +1,7 @@ +require 'test_helper' + +class ReservationTest < ActiveSupport::TestCase + # test "the truth" do + # assert true + # end +end diff --git a/test/models/user_test.rb b/test/models/user_test.rb new file mode 100644 index 0000000..82f61e0 --- /dev/null +++ b/test/models/user_test.rb @@ -0,0 +1,7 @@ +require 'test_helper' + +class UserTest < ActiveSupport::TestCase + # test "the truth" do + # assert true + # end +end diff --git a/test/system/.keep b/test/system/.keep new file mode 100644 index 0000000..e69de29 diff --git a/test/system/cars_test.rb b/test/system/cars_test.rb new file mode 100644 index 0000000..e5840a1 --- /dev/null +++ b/test/system/cars_test.rb @@ -0,0 +1,47 @@ +require 'application_system_test_case' + +class CarsTest < ApplicationSystemTestCase + setup do + @car = cars(:one) + end + + test 'visiting the index' do + visit cars_url + assert_selector 'h1', text: 'Cars' + end + + test 'should create car' do + visit cars_url + click_on 'New car' + + fill_in 'Make', with: @car.make + fill_in 'Model', with: @car.model + fill_in 'Price', with: @car.price + fill_in 'Year', with: @car.year + click_on 'Create Car' + + assert_text 'Car was successfully created' + click_on 'Back' + end + + test 'should update Car' do + visit car_url(@car) + click_on 'Edit this car', match: :first + + fill_in 'Make', with: @car.make + fill_in 'Model', with: @car.model + fill_in 'Price', with: @car.price + fill_in 'Year', with: @car.year + click_on 'Update Car' + + assert_text 'Car was successfully updated' + click_on 'Back' + end + + test 'should destroy Car' do + visit car_url(@car) + click_on 'Destroy this car', match: :first + + assert_text 'Car was successfully destroyed' + end +end diff --git a/test/system/reservations_test.rb b/test/system/reservations_test.rb new file mode 100644 index 0000000..1c054d2 --- /dev/null +++ b/test/system/reservations_test.rb @@ -0,0 +1,43 @@ +require 'application_system_test_case' + +class ReservationsTest < ApplicationSystemTestCase + setup do + @reservation = reservations(:one) + end + + test 'visiting the index' do + visit reservations_url + assert_selector 'h1', text: 'Reservations' + end + + test 'should create reservation' do + visit reservations_url + click_on 'New reservation' + + fill_in 'Appointment', with: @reservation.appointment_id + fill_in 'Reservation date', with: @reservation.reservation_date + click_on 'Create Reservation' + + assert_text 'Reservation was successfully created' + click_on 'Back' + end + + test 'should update Reservation' do + visit reservation_url(@reservation) + click_on 'Edit this reservation', match: :first + + fill_in 'Appointment', with: @reservation.appointment_id + fill_in 'Reservation date', with: @reservation.reservation_date + click_on 'Update Reservation' + + assert_text 'Reservation was successfully updated' + click_on 'Back' + end + + test 'should destroy Reservation' do + visit reservation_url(@reservation) + click_on 'Destroy this reservation', match: :first + + assert_text 'Reservation was successfully destroyed' + end +end diff --git a/test/system/users_test.rb b/test/system/users_test.rb new file mode 100644 index 0000000..110ffe7 --- /dev/null +++ b/test/system/users_test.rb @@ -0,0 +1,45 @@ +require 'application_system_test_case' + +class UsersTest < ApplicationSystemTestCase + setup do + @user = users(:one) + end + + test 'visiting the index' do + visit users_url + assert_selector 'h1', text: 'Users' + end + + test 'should create user' do + visit users_url + click_on 'New user' + + fill_in 'Email', with: @user.email + fill_in 'Name', with: @user.name + fill_in 'Phone', with: @user.phone + click_on 'Create User' + + assert_text 'User was successfully created' + click_on 'Back' + end + + test 'should update User' do + visit user_url(@user) + click_on 'Edit this user', match: :first + + fill_in 'Email', with: @user.email + fill_in 'Name', with: @user.name + fill_in 'Phone', with: @user.phone + click_on 'Update User' + + assert_text 'User was successfully updated' + click_on 'Back' + end + + test 'should destroy User' do + visit user_url(@user) + click_on 'Destroy this user', match: :first + + assert_text 'User was successfully destroyed' + end +end diff --git a/test/test_helper.rb b/test/test_helper.rb new file mode 100644 index 0000000..d5300f8 --- /dev/null +++ b/test/test_helper.rb @@ -0,0 +1,13 @@ +ENV['RAILS_ENV'] ||= 'test' +require_relative '../config/environment' +require 'rails/test_help' + +class ActiveSupport::TestCase + # Run tests in parallel with specified workers + parallelize(workers: :number_of_processors) + + # Setup all fixtures in test/fixtures/*.yml for all tests in alphabetical order. + fixtures :all + + # Add more helper methods to be used by all tests here... +end diff --git a/tmp/.keep b/tmp/.keep new file mode 100644 index 0000000..e69de29 diff --git a/tmp/pids/.keep b/tmp/pids/.keep new file mode 100644 index 0000000..e69de29 diff --git a/tmp/storage/.keep b/tmp/storage/.keep new file mode 100644 index 0000000..e69de29 diff --git a/vendor/.keep b/vendor/.keep new file mode 100644 index 0000000..e69de29 diff --git a/vendor/javascript/.keep b/vendor/javascript/.keep new file mode 100644 index 0000000..e69de29 diff --git a/yarn.lock b/yarn.lock new file mode 100644 index 0000000..fb57ccd --- /dev/null +++ b/yarn.lock @@ -0,0 +1,4 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + +