+ <% sub_order.order_items.each do |item| %>
+ <% if item.product.user_id == @user.id %>
+
Creature: <%= item.product.name %>
+
Quantity: <%= item.quantity %>
+
Price: $<%= item.product.price * item.quantity %>
+
+ <% end %>
+ <% end %>
+
+
+ <% end %>
+
+
+
+ <% end %>
+
+
diff --git a/bin/bundle b/bin/bundle
new file mode 100755
index 0000000000..66e9889e8b
--- /dev/null
+++ b/bin/bundle
@@ -0,0 +1,3 @@
+#!/usr/bin/env ruby
+ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../Gemfile', __FILE__)
+load Gem.bin_path('bundler', 'bundle')
diff --git a/bin/rails b/bin/rails
new file mode 100755
index 0000000000..5badb2fde0
--- /dev/null
+++ b/bin/rails
@@ -0,0 +1,9 @@
+#!/usr/bin/env ruby
+begin
+ load File.expand_path('../spring', __FILE__)
+rescue LoadError => e
+ raise unless e.message.include?('spring')
+end
+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 100755
index 0000000000..d87d5f5781
--- /dev/null
+++ b/bin/rake
@@ -0,0 +1,9 @@
+#!/usr/bin/env ruby
+begin
+ load File.expand_path('../spring', __FILE__)
+rescue LoadError => e
+ raise unless e.message.include?('spring')
+end
+require_relative '../config/boot'
+require 'rake'
+Rake.application.run
diff --git a/bin/setup b/bin/setup
new file mode 100755
index 0000000000..78c4e861dc
--- /dev/null
+++ b/bin/setup
@@ -0,0 +1,38 @@
+#!/usr/bin/env ruby
+require 'pathname'
+require 'fileutils'
+include FileUtils
+
+# path to your application root.
+APP_ROOT = Pathname.new File.expand_path('../../', __FILE__)
+
+def system!(*args)
+ system(*args) || abort("\n== Command #{args} failed ==")
+end
+
+chdir APP_ROOT do
+ # This script is a starting point to setup your application.
+ # Add necessary setup steps to this file.
+
+ puts '== Installing dependencies =='
+ system! 'gem install bundler --conservative'
+ system('bundle check') || system!('bundle install')
+
+ # Install JavaScript dependencies if using Yarn
+ # system('bin/yarn')
+
+
+ # puts "\n== Copying sample files =="
+ # unless File.exist?('config/database.yml')
+ # cp 'config/database.yml.sample', 'config/database.yml'
+ # end
+
+ puts "\n== Preparing database =="
+ system! 'bin/rails db:setup'
+
+ 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/bin/spring b/bin/spring
new file mode 100755
index 0000000000..fb2ec2ebb4
--- /dev/null
+++ b/bin/spring
@@ -0,0 +1,17 @@
+#!/usr/bin/env ruby
+
+# This file loads spring without using Bundler, in order to be fast.
+# It gets overwritten when you run the `spring binstub` command.
+
+unless defined?(Spring)
+ require 'rubygems'
+ require 'bundler'
+
+ lockfile = Bundler::LockfileParser.new(Bundler.default_lockfile.read)
+ spring = lockfile.specs.detect { |spec| spec.name == "spring" }
+ if spring
+ Gem.use_paths Gem.dir, Bundler.bundle_path.to_s, *Gem.path
+ gem 'spring', spring.version
+ require 'spring/binstub'
+ end
+end
diff --git a/bin/update b/bin/update
new file mode 100755
index 0000000000..a8e4462f20
--- /dev/null
+++ b/bin/update
@@ -0,0 +1,29 @@
+#!/usr/bin/env ruby
+require 'pathname'
+require 'fileutils'
+include FileUtils
+
+# path to your application root.
+APP_ROOT = Pathname.new File.expand_path('../../', __FILE__)
+
+def system!(*args)
+ system(*args) || abort("\n== Command #{args} failed ==")
+end
+
+chdir APP_ROOT do
+ # This script is a way to update your development environment automatically.
+ # Add necessary update steps to this file.
+
+ puts '== Installing dependencies =='
+ system! 'gem install bundler --conservative'
+ system('bundle check') || system!('bundle install')
+
+ puts "\n== Updating database =="
+ system! 'bin/rails db:migrate'
+
+ 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/bin/yarn b/bin/yarn
new file mode 100755
index 0000000000..c2bacef836
--- /dev/null
+++ b/bin/yarn
@@ -0,0 +1,11 @@
+#!/usr/bin/env ruby
+VENDOR_PATH = File.expand_path('..', __dir__)
+Dir.chdir(VENDOR_PATH) do
+ begin
+ exec "yarnpkg #{ARGV.join(" ")}"
+ rescue Errno::ENOENT
+ $stderr.puts "Yarn executable was not detected in the system."
+ $stderr.puts "Download Yarn at https://yarnpkg.com/en/docs/install"
+ exit 1
+ end
+end
diff --git a/config.ru b/config.ru
new file mode 100644
index 0000000000..f7ba0b527b
--- /dev/null
+++ b/config.ru
@@ -0,0 +1,5 @@
+# This file is used by Rack-based servers to start the application.
+
+require_relative 'config/environment'
+
+run Rails.application
diff --git a/config/application.rb b/config/application.rb
new file mode 100644
index 0000000000..5f6ca0f9b2
--- /dev/null
+++ b/config/application.rb
@@ -0,0 +1,25 @@
+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 Betsy
+ class Application < Rails::Application
+ config.generators do |g|
+ # Force new test files to be generated in the minitest-spec style
+ g.test_framework :minitest, spec: true
+
+ # Always use .js files, never .coffee
+ g.javascript_engine :js
+ end
+ # Initialize configuration defaults for originally generated Rails version.
+ config.load_defaults 5.1
+
+ # Settings in config/environments/* take precedence over those specified here.
+ # Application configuration should go into files in config/initializers
+ # -- all .rb files in that directory are automatically loaded.
+ end
+end
diff --git a/config/boot.rb b/config/boot.rb
new file mode 100644
index 0000000000..30f5120df6
--- /dev/null
+++ b/config/boot.rb
@@ -0,0 +1,3 @@
+ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../Gemfile', __dir__)
+
+require 'bundler/setup' # Set up gems listed in the Gemfile.
diff --git a/config/cable.yml b/config/cable.yml
new file mode 100644
index 0000000000..3cba994bb2
--- /dev/null
+++ b/config/cable.yml
@@ -0,0 +1,10 @@
+development:
+ adapter: async
+
+test:
+ adapter: async
+
+production:
+ adapter: redis
+ url: redis://localhost:6379/1
+ channel_prefix: betsy_production
diff --git a/config/database.yml b/config/database.yml
new file mode 100644
index 0000000000..6903bb6083
--- /dev/null
+++ b/config/database.yml
@@ -0,0 +1,85 @@
+# PostgreSQL. Versions 9.1 and up are supported.
+#
+# Install the pg driver:
+# gem install pg
+# On OS X with Homebrew:
+# gem install pg -- --with-pg-config=/usr/local/bin/pg_config
+# On OS X with MacPorts:
+# gem install pg -- --with-pg-config=/opt/local/lib/postgresql84/bin/pg_config
+# On Windows:
+# gem install pg
+# Choose the win32 build.
+# Install PostgreSQL and put its /bin directory on your path.
+#
+# Configure Using Gemfile
+# gem 'pg'
+#
+default: &default
+ adapter: postgresql
+ encoding: unicode
+ # For details on connection pooling, see Rails configuration guide
+ # http://guides.rubyonrails.org/configuring.html#database-pooling
+ pool: <%= ENV.fetch("RAILS_MAX_THREADS") { 5 } %>
+
+development:
+ <<: *default
+ database: betsy_development
+
+ # The specified database role being used to connect to postgres.
+ # To create additional roles in postgres see `$ createuser --help`.
+ # When left blank, postgres will use the default role. This is
+ # the same name as the operating system user that initialized the database.
+ #username: betsy
+
+ # The password associated with the postgres role (username).
+ #password:
+
+ # Connect on a TCP socket. Omitted by default since the client uses a
+ # domain socket that doesn't need configuration. Windows does not have
+ # domain sockets, so uncomment these lines.
+ #host: localhost
+
+ # The TCP port the server listens on. Defaults to 5432.
+ # If your server runs on a different port number, change accordingly.
+ #port: 5432
+
+ # Schema search path. The server defaults to $user,public
+ #schema_search_path: myapp,sharedapp,public
+
+ # Minimum log levels, in increasing order:
+ # debug5, debug4, debug3, debug2, debug1,
+ # log, notice, warning, error, fatal, and panic
+ # Defaults to warning.
+ #min_messages: notice
+
+# Warning: The database defined as "test" will be erased and
+# re-generated from your development database when you run "rake".
+# Do not set this db to the same as development or production.
+test:
+ <<: *default
+ database: betsy_test
+
+# As with config/secrets.yml, you never want to store sensitive information,
+# like your database password, in your source code. If your source code is
+# ever seen by anyone, they now have access to your database.
+#
+# Instead, provide the password as a unix environment variable when you boot
+# the app. Read http://guides.rubyonrails.org/configuring.html#configuring-a-database
+# for a full rundown on how to provide these environment variables in a
+# production deployment.
+#
+# On Heroku and other platform providers, you may have a full connection URL
+# available as an environment variable. For example:
+#
+# DATABASE_URL="postgres://myuser:mypass@localhost/somedatabase"
+#
+# You can use this database configuration with:
+#
+# production:
+# url: <%= ENV['DATABASE_URL'] %>
+#
+production:
+ <<: *default
+ database: betsy_production
+ username: betsy
+ password: <%= ENV['BETSY_DATABASE_PASSWORD'] %>
diff --git a/config/environment.rb b/config/environment.rb
new file mode 100644
index 0000000000..426333bb46
--- /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 0000000000..5187e22186
--- /dev/null
+++ b/config/environments/development.rb
@@ -0,0 +1,54 @@
+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 on
+ # every request. 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/disable caching. By default caching is disabled.
+ if Rails.root.join('tmp/caching-dev.txt').exist?
+ config.action_controller.perform_caching = true
+
+ config.cache_store = :memory_store
+ config.public_file_server.headers = {
+ 'Cache-Control' => "public, max-age=#{2.days.seconds.to_i}"
+ }
+ else
+ config.action_controller.perform_caching = false
+
+ config.cache_store = :null_store
+ end
+
+ # 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 an error on page load if there are pending migrations.
+ config.active_record.migration_error = :page_load
+
+ # Debug mode disables concatenation and preprocessing of assets.
+ # This option may cause significant delays in view rendering with a large
+ # number of complex assets.
+ config.assets.debug = true
+
+ # Suppress logger output for asset requests.
+ config.assets.quiet = true
+
+ # Raises error for missing translations
+ # config.action_view.raise_on_missing_translations = true
+
+ # Use an evented file watcher to asynchronously detect changes in source code,
+ # routes, locales, etc. This feature depends on the listen gem.
+ config.file_watcher = ActiveSupport::EventedFileUpdateChecker
+end
diff --git a/config/environments/production.rb b/config/environments/production.rb
new file mode 100644
index 0000000000..9284f84839
--- /dev/null
+++ b/config/environments/production.rb
@@ -0,0 +1,91 @@
+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
+
+ # Attempt to read encrypted secrets from `config/secrets.yml.enc`.
+ # Requires an encryption key in `ENV["RAILS_MASTER_KEY"]` or
+ # `config/secrets.yml.key`.
+ config.read_encrypted_secrets = 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 JavaScripts and CSS.
+ config.assets.js_compressor = :uglifier
+ # config.assets.css_compressor = :sass
+
+ # Do not fallback to assets pipeline if a precompiled asset is missed.
+ config.assets.compile = false
+
+ # `config.assets.precompile` and `config.assets.version` have moved to config/initializers/assets.rb
+
+ # Enable serving of images, stylesheets, and JavaScripts from an asset server.
+ # config.action_controller.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
+
+ # 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
+
+ # Use the lowest log level to ensure availability of diagnostic information
+ # when problems arise.
+ config.log_level = :debug
+
+ # 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 = "betsy_#{Rails.env}"
+ 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
+
+ # Send deprecation notices to registered listeners.
+ config.active_support.deprecation = :notify
+
+ # 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 0000000000..8e5cbde533
--- /dev/null
+++ b/config/environments/test.rb
@@ -0,0 +1,42 @@
+Rails.application.configure do
+ # Settings specified here will take precedence over those in config/application.rb.
+
+ # 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!
+ config.cache_classes = true
+
+ # Do not eager load code on boot. This avoids loading your whole application
+ # just for the purpose of running a single test. If you are using a tool that
+ # preloads Rails for running tests, you may have to set it to true.
+ config.eager_load = false
+
+ # 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.seconds.to_i}"
+ }
+
+ # Show full error reports and disable caching.
+ config.consider_all_requests_local = true
+ config.action_controller.perform_caching = false
+
+ # 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
+ 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
+
+ # Raises error for missing translations
+ # config.action_view.raise_on_missing_translations = true
+end
diff --git a/config/initializers/application_controller_renderer.rb b/config/initializers/application_controller_renderer.rb
new file mode 100644
index 0000000000..89d2efab2b
--- /dev/null
+++ b/config/initializers/application_controller_renderer.rb
@@ -0,0 +1,8 @@
+# Be sure to restart your server when you modify this file.
+
+# ActiveSupport::Reloader.to_prepare do
+# ApplicationController.renderer.defaults.merge!(
+# http_host: 'example.org',
+# https: false
+# )
+# end
diff --git a/config/initializers/assets.rb b/config/initializers/assets.rb
new file mode 100644
index 0000000000..4b828e80cb
--- /dev/null
+++ b/config/initializers/assets.rb
@@ -0,0 +1,14 @@
+# 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
+# Add Yarn node_modules folder to the asset load path.
+Rails.application.config.assets.paths << Rails.root.join('node_modules')
+
+# 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/backtrace_silencers.rb b/config/initializers/backtrace_silencers.rb
new file mode 100644
index 0000000000..59385cdf37
--- /dev/null
+++ b/config/initializers/backtrace_silencers.rb
@@ -0,0 +1,7 @@
+# Be sure to restart your server when you modify this file.
+
+# You can add backtrace silencers for libraries that you're using but don't wish to see in your backtraces.
+# Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ }
+
+# You can also remove all the silencers if you're trying to debug a problem that might stem from framework code.
+# Rails.backtrace_cleaner.remove_silencers!
diff --git a/config/initializers/cookies_serializer.rb b/config/initializers/cookies_serializer.rb
new file mode 100644
index 0000000000..5a6a32d371
--- /dev/null
+++ b/config/initializers/cookies_serializer.rb
@@ -0,0 +1,5 @@
+# Be sure to restart your server when you modify this file.
+
+# Specify a serializer for the signed and encrypted cookie jars.
+# Valid options are :json, :marshal, and :hybrid.
+Rails.application.config.action_dispatch.cookies_serializer = :json
diff --git a/config/initializers/filter_parameter_logging.rb b/config/initializers/filter_parameter_logging.rb
new file mode 100644
index 0000000000..4a994e1e7b
--- /dev/null
+++ b/config/initializers/filter_parameter_logging.rb
@@ -0,0 +1,4 @@
+# Be sure to restart your server when you modify this file.
+
+# Configure sensitive parameters which will be filtered from the log file.
+Rails.application.config.filter_parameters += [:password]
diff --git a/config/initializers/inflections.rb b/config/initializers/inflections.rb
new file mode 100644
index 0000000000..ac033bf9dc
--- /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/mime_types.rb b/config/initializers/mime_types.rb
new file mode 100644
index 0000000000..dc1899682b
--- /dev/null
+++ b/config/initializers/mime_types.rb
@@ -0,0 +1,4 @@
+# Be sure to restart your server when you modify this file.
+
+# Add new mime types for use in respond_to blocks:
+# Mime::Type.register "text/richtext", :rtf
diff --git a/config/initializers/omniauth.rb b/config/initializers/omniauth.rb
new file mode 100644
index 0000000000..fd4416122a
--- /dev/null
+++ b/config/initializers/omniauth.rb
@@ -0,0 +1,3 @@
+Rails.application.config.middleware.use OmniAuth::Builder do
+ provider :github, ENV["GITHUB_CLIENT_ID"], ENV["GITHUB_CLIENT_SECRET"], scope: "user:email"
+end
diff --git a/config/initializers/wrap_parameters.rb b/config/initializers/wrap_parameters.rb
new file mode 100644
index 0000000000..bbfc3961bf
--- /dev/null
+++ b/config/initializers/wrap_parameters.rb
@@ -0,0 +1,14 @@
+# Be sure to restart your server when you modify this file.
+
+# This file contains settings for ActionController::ParamsWrapper which
+# is enabled by default.
+
+# Enable parameter wrapping for JSON. You can disable this by setting :format to an empty array.
+ActiveSupport.on_load(:action_controller) do
+ wrap_parameters format: [:json]
+end
+
+# To enable root element in JSON for ActiveRecord objects.
+# ActiveSupport.on_load(:active_record) do
+# self.include_root_in_json = true
+# end
diff --git a/config/locales/en.yml b/config/locales/en.yml
new file mode 100644
index 0000000000..decc5a8573
--- /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 http://guides.rubyonrails.org/i18n.html.
+
+en:
+ hello: "Hello world"
diff --git a/config/puma.rb b/config/puma.rb
new file mode 100644
index 0000000000..1e19380dcb
--- /dev/null
+++ b/config/puma.rb
@@ -0,0 +1,56 @@
+# 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.
+#
+threads_count = ENV.fetch("RAILS_MAX_THREADS") { 5 }
+threads threads_count, threads_count
+
+# Specifies the `port` that Puma will listen on to receive requests; default is 3000.
+#
+port ENV.fetch("PORT") { 3000 }
+
+# Specifies the `environment` that Puma will run in.
+#
+environment ENV.fetch("RAILS_ENV") { "development" }
+
+# Specifies the number of `workers` to boot in clustered mode.
+# Workers are forked webserver 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") { 2 }
+
+# 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. If you use this option
+# you need to make sure to reconnect any threads in the `on_worker_boot`
+# block.
+#
+# preload_app!
+
+# If you are preloading your application and using Active Record, it's
+# recommended that you close any connections to the database before workers
+# are forked to prevent connection leakage.
+#
+# before_fork do
+# ActiveRecord::Base.connection_pool.disconnect! if defined?(ActiveRecord)
+# end
+
+# The code in the `on_worker_boot` will be called if you are using
+# clustered mode by specifying a number of `workers`. After each worker
+# process is booted, this block will be run. If you are using the `preload_app!`
+# option, you will want to use this block to reconnect to any threads
+# or connections that may have been created at application boot, as Ruby
+# cannot share connections between processes.
+#
+# on_worker_boot do
+# ActiveRecord::Base.establish_connection if defined?(ActiveRecord)
+# end
+#
+
+# Allow puma to be restarted by `rails restart` command.
+plugin :tmp_restart
diff --git a/config/routes.rb b/config/routes.rb
new file mode 100644
index 0000000000..f5b29c6cdc
--- /dev/null
+++ b/config/routes.rb
@@ -0,0 +1,22 @@
+Rails.application.routes.draw do
+ # For details on the DSL available within this file, see http://guides.rubyonrails.org/routing.html
+ root 'products#index', as: 'root'
+
+
+ resources :users
+ resources :orders, only: [:index, :show, :update, :new]
+ resources :categories, only: [:index, :new, :create, :show]
+ resources :order_items, only: [:create, :update, :destroy]
+ resources :products, only: [:show, :index]
+
+ resources :users do
+ resources :products, only: [:new, :create, :edit, :update]
+ end
+
+ post "/product/:product_id/reviews", to: "reviews#create", as: "product_reviews"
+ get "/products/:product_id/reviews/new", to: "reviews#new", as: "new_product_review"
+
+ get "/auth/github", as: 'github_login'
+ get "/auth/:provider/callback", to: "sessions#create", as: "auth_callback"
+ delete '/logout', to: 'sessions#destroy', as: 'logout'
+end
diff --git a/config/secrets.yml b/config/secrets.yml
new file mode 100644
index 0000000000..fbb53c9653
--- /dev/null
+++ b/config/secrets.yml
@@ -0,0 +1,32 @@
+# Be sure to restart your server when you modify this file.
+
+# Your secret key is used for verifying the integrity of signed cookies.
+# If you change this key, all old signed cookies will become invalid!
+
+# Make sure the secret is at least 30 characters and all random,
+# no regular words or you'll be exposed to dictionary attacks.
+# You can use `rails secret` to generate a secure secret key.
+
+# Make sure the secrets in this file are kept private
+# if you're sharing your code publicly.
+
+# Shared secrets are available across all environments.
+
+# shared:
+# api_key: a1B2c3D4e5F6
+
+# Environmental secrets are only available for that specific environment.
+
+development:
+ secret_key_base: f5f317d86d7f21fcda8530d9bbc1579f39265ecd3c5fa61c7ec0bc5b06fcdab157a69d3e1ea5417423e36fea247a85d20887a14cbefd8dc80feaa43e768086e3
+
+test:
+ secret_key_base: 52be4c70a272fd3228fa947bd8b6a72188f3285af927ddfc36553fec6105bb93dcb6b5e01c6bc20d86be0699b8fad226a4b5eef674878a75145a59d58244bc30
+
+# Do not keep production secrets in the unencrypted secrets file.
+# Instead, either read values from the environment.
+# Or, use `bin/rails secrets:setup` to configure encrypted secrets
+# and move the `production:` environment over there.
+
+production:
+ secret_key_base: <%= ENV["SECRET_KEY_BASE"] %>
diff --git a/config/spring.rb b/config/spring.rb
new file mode 100644
index 0000000000..c9119b40c0
--- /dev/null
+++ b/config/spring.rb
@@ -0,0 +1,6 @@
+%w(
+ .ruby-version
+ .rbenv-vars
+ tmp/restart.txt
+ tmp/caching-dev.txt
+).each { |path| Spring.watch(path) }
diff --git a/db/migrate/20180419170141_create_products.rb b/db/migrate/20180419170141_create_products.rb
new file mode 100644
index 0000000000..49c4c470cb
--- /dev/null
+++ b/db/migrate/20180419170141_create_products.rb
@@ -0,0 +1,15 @@
+class CreateProducts < ActiveRecord::Migration[5.1]
+ def change
+ create_table :products do |t|
+ t.string :name
+ t.integer :stock
+ t.integer :price
+ t.string :description
+ t.boolean :status
+ t.string :category
+ t.string :image_url
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20180419170403_create_reviews.rb b/db/migrate/20180419170403_create_reviews.rb
new file mode 100644
index 0000000000..f369d1c988
--- /dev/null
+++ b/db/migrate/20180419170403_create_reviews.rb
@@ -0,0 +1,9 @@
+class CreateReviews < ActiveRecord::Migration[5.1]
+ def change
+ create_table :reviews do |t|
+ t.integer :rating
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20180419170702_create_orders.rb b/db/migrate/20180419170702_create_orders.rb
new file mode 100644
index 0000000000..82264eacb1
--- /dev/null
+++ b/db/migrate/20180419170702_create_orders.rb
@@ -0,0 +1,16 @@
+class CreateOrders < ActiveRecord::Migration[5.1]
+ def change
+ create_table :orders do |t|
+ t.string :status
+ t.integer :cc_num
+ t.integer :cvv
+ t.string :email
+ t.string :street_address
+ t.string :state
+ t.string :city
+ t.integer :zip
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20180419170846_create_users.rb b/db/migrate/20180419170846_create_users.rb
new file mode 100644
index 0000000000..59010f7a8a
--- /dev/null
+++ b/db/migrate/20180419170846_create_users.rb
@@ -0,0 +1,12 @@
+class CreateUsers < ActiveRecord::Migration[5.1]
+ def change
+ create_table :users do |t|
+ t.string :username
+ t.string :email
+ t.integer :uid
+ t.string :provider
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20180419170950_create_categories.rb b/db/migrate/20180419170950_create_categories.rb
new file mode 100644
index 0000000000..5bef4913b8
--- /dev/null
+++ b/db/migrate/20180419170950_create_categories.rb
@@ -0,0 +1,9 @@
+class CreateCategories < ActiveRecord::Migration[5.1]
+ def change
+ create_table :categories do |t|
+ t.string :name
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20180419201142_create_join_table_products_orders.rb b/db/migrate/20180419201142_create_join_table_products_orders.rb
new file mode 100644
index 0000000000..4fc5e631d5
--- /dev/null
+++ b/db/migrate/20180419201142_create_join_table_products_orders.rb
@@ -0,0 +1,8 @@
+class CreateJoinTableProductsOrders < ActiveRecord::Migration[5.1]
+ def change
+ create_join_table :products, :orders do |t|
+ # t.index [:product_id, :order_id]
+ # t.index [:order_id, :product_id]
+ end
+ end
+end
diff --git a/db/migrate/20180419201409_create_join_table_products_categories.rb b/db/migrate/20180419201409_create_join_table_products_categories.rb
new file mode 100644
index 0000000000..4dc1e42284
--- /dev/null
+++ b/db/migrate/20180419201409_create_join_table_products_categories.rb
@@ -0,0 +1,8 @@
+class CreateJoinTableProductsCategories < ActiveRecord::Migration[5.1]
+ def change
+ create_join_table :products, :categories do |t|
+ # t.index [:product_id, :category_id]
+ # t.index [:category_id, :product_id]
+ end
+ end
+end
diff --git a/db/migrate/20180419201856_create_products_categories.rb b/db/migrate/20180419201856_create_products_categories.rb
new file mode 100644
index 0000000000..8c779dd726
--- /dev/null
+++ b/db/migrate/20180419201856_create_products_categories.rb
@@ -0,0 +1,9 @@
+class CreateProductsCategories < ActiveRecord::Migration[5.1]
+ def change
+ create_table :products_categories do |t|
+ t.integer :quantity
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20180419205915_add_name_on_card_to_order.rb b/db/migrate/20180419205915_add_name_on_card_to_order.rb
new file mode 100644
index 0000000000..d310d3b34c
--- /dev/null
+++ b/db/migrate/20180419205915_add_name_on_card_to_order.rb
@@ -0,0 +1,5 @@
+class AddNameOnCardToOrder < ActiveRecord::Migration[5.1]
+ def change
+ add_column :orders, :name_on_card, :string
+ end
+end
diff --git a/db/migrate/20180419210438_add_quantity_to_orders_products.rb b/db/migrate/20180419210438_add_quantity_to_orders_products.rb
new file mode 100644
index 0000000000..10544e2616
--- /dev/null
+++ b/db/migrate/20180419210438_add_quantity_to_orders_products.rb
@@ -0,0 +1,5 @@
+class AddQuantityToOrdersProducts < ActiveRecord::Migration[5.1]
+ def change
+ add_column :orders_products, :quantity, :integer
+ end
+end
diff --git a/db/migrate/20180419210701_add_text_review_to_reviews.rb b/db/migrate/20180419210701_add_text_review_to_reviews.rb
new file mode 100644
index 0000000000..88d1279d42
--- /dev/null
+++ b/db/migrate/20180419210701_add_text_review_to_reviews.rb
@@ -0,0 +1,5 @@
+class AddTextReviewToReviews < ActiveRecord::Migration[5.1]
+ def change
+ add_column :reviews, :text_review, :string
+ end
+end
diff --git a/db/migrate/20180419211006_remove_category_from_products.rb b/db/migrate/20180419211006_remove_category_from_products.rb
new file mode 100644
index 0000000000..f36bc48965
--- /dev/null
+++ b/db/migrate/20180419211006_remove_category_from_products.rb
@@ -0,0 +1,5 @@
+class RemoveCategoryFromProducts < ActiveRecord::Migration[5.1]
+ def change
+ remove_column :products, :category, :string
+ end
+end
diff --git a/db/migrate/20180419221427_add_user_id_column_to_products_table.rb b/db/migrate/20180419221427_add_user_id_column_to_products_table.rb
new file mode 100644
index 0000000000..e99089e342
--- /dev/null
+++ b/db/migrate/20180419221427_add_user_id_column_to_products_table.rb
@@ -0,0 +1,5 @@
+class AddUserIdColumnToProductsTable < ActiveRecord::Migration[5.1]
+ def change
+ add_column :products, :user_id, :integer
+ end
+end
diff --git a/db/migrate/20180421034944_change_table_name_orders_products_to_order_items.rb b/db/migrate/20180421034944_change_table_name_orders_products_to_order_items.rb
new file mode 100644
index 0000000000..d0bd74d376
--- /dev/null
+++ b/db/migrate/20180421034944_change_table_name_orders_products_to_order_items.rb
@@ -0,0 +1,5 @@
+class ChangeTableNameOrdersProductsToOrderItems < ActiveRecord::Migration[5.1]
+ def change
+ rename_table :orders_products, :OrderItems
+ end
+end
diff --git a/db/migrate/20180421035821_change_table_name_order_items_to_order_items.rb b/db/migrate/20180421035821_change_table_name_order_items_to_order_items.rb
new file mode 100644
index 0000000000..4c825edfff
--- /dev/null
+++ b/db/migrate/20180421035821_change_table_name_order_items_to_order_items.rb
@@ -0,0 +1,5 @@
+class ChangeTableNameOrderItemsToOrderItems < ActiveRecord::Migration[5.1]
+ def change
+ rename_table :OrderItems, :order_items
+ end
+end
diff --git a/db/migrate/20180421040238_drop_categories_product.rb b/db/migrate/20180421040238_drop_categories_product.rb
new file mode 100644
index 0000000000..b9ad5125a0
--- /dev/null
+++ b/db/migrate/20180421040238_drop_categories_product.rb
@@ -0,0 +1,9 @@
+class DropCategoriesProduct < ActiveRecord::Migration[5.1]
+ def change
+ drop_table :categories_products do |t|
+ t.bigint "product_id", null: false
+ t.bigint "category_id", null: false
+ t.timestamps null: false
+ end
+ end
+end
diff --git a/db/migrate/20180421040743_add_productid_to_review_table.rb b/db/migrate/20180421040743_add_productid_to_review_table.rb
new file mode 100644
index 0000000000..74cca18c1c
--- /dev/null
+++ b/db/migrate/20180421040743_add_productid_to_review_table.rb
@@ -0,0 +1,5 @@
+class AddProductidToReviewTable < ActiveRecord::Migration[5.1]
+ def change
+ add_column :reviews, :user_id, :bigint
+ end
+end
diff --git a/db/migrate/20180421041521_change_ccnum_column_to_strng.rb b/db/migrate/20180421041521_change_ccnum_column_to_strng.rb
new file mode 100644
index 0000000000..b8abe58291
--- /dev/null
+++ b/db/migrate/20180421041521_change_ccnum_column_to_strng.rb
@@ -0,0 +1,5 @@
+class ChangeCcnumColumnToStrng < ActiveRecord::Migration[5.1]
+ def change
+ change_column :orders, :cc_num, :string
+ end
+end
diff --git a/db/migrate/20180421054721_add_product_id_to_reviews_table_try_two.rb b/db/migrate/20180421054721_add_product_id_to_reviews_table_try_two.rb
new file mode 100644
index 0000000000..8404eab6ad
--- /dev/null
+++ b/db/migrate/20180421054721_add_product_id_to_reviews_table_try_two.rb
@@ -0,0 +1,7 @@
+class AddProductIdToReviewsTableTryTwo < ActiveRecord::Migration[5.1]
+ def change
+ def change
+ add_column :reviews, :product_id, :bigint
+ end
+ end
+end
diff --git a/db/migrate/20180421055333_add_product_id_to_reviews_table_try_three.rb b/db/migrate/20180421055333_add_product_id_to_reviews_table_try_three.rb
new file mode 100644
index 0000000000..efc65d84ae
--- /dev/null
+++ b/db/migrate/20180421055333_add_product_id_to_reviews_table_try_three.rb
@@ -0,0 +1,7 @@
+class AddProductIdToReviewsTableTryThree < ActiveRecord::Migration[5.1]
+ def change
+ def change
+ add_column :reviews, :user_id, :integer
+ end
+ end
+end
diff --git a/db/migrate/20180421055653_add_product_id_to_reviews_table_try_four.rb b/db/migrate/20180421055653_add_product_id_to_reviews_table_try_four.rb
new file mode 100644
index 0000000000..0ebd9dd2fc
--- /dev/null
+++ b/db/migrate/20180421055653_add_product_id_to_reviews_table_try_four.rb
@@ -0,0 +1,5 @@
+class AddProductIdToReviewsTableTryFour < ActiveRecord::Migration[5.1]
+ def change
+ add_column :reviews, :product_id, :integer
+ end
+end
diff --git a/db/migrate/20180425070534_add_order_items_id.rb b/db/migrate/20180425070534_add_order_items_id.rb
new file mode 100644
index 0000000000..85f5b49bbe
--- /dev/null
+++ b/db/migrate/20180425070534_add_order_items_id.rb
@@ -0,0 +1,5 @@
+class AddOrderItemsId < ActiveRecord::Migration[5.1]
+ def change
+ add_column :order_items, :id, :primary_key
+ end
+end
diff --git a/db/schema.rb b/db/schema.rb
new file mode 100644
index 0000000000..73a50633fc
--- /dev/null
+++ b/db/schema.rb
@@ -0,0 +1,80 @@
+# 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.
+#
+# Note that this schema.rb definition is the authoritative source for your
+# database schema. If you need to create the application database on another
+# system, you should be using db:schema:load, not running all the migrations
+# from scratch. The latter is a flawed and unsustainable approach (the more migrations
+# you'll amass, the slower it'll run and the greater likelihood for issues).
+#
+# It's strongly recommended that you check this file into your version control system.
+
+ActiveRecord::Schema.define(version: 20180425070534) do
+
+ # These are extensions that must be enabled in order to support this database
+ enable_extension "plpgsql"
+
+ create_table "categories", force: :cascade do |t|
+ t.string "name"
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ end
+
+ create_table "categories_products", force: :cascade do |t|
+ t.bigint "category_id"
+ t.bigint "product_id"
+ t.integer "quantity"
+ end
+
+ create_table "order_items", force: :cascade do |t|
+ t.bigint "product_id", null: false
+ t.bigint "order_id", null: false
+ t.integer "quantity"
+ end
+
+ create_table "orders", force: :cascade do |t|
+ t.string "status"
+ t.string "cc_num"
+ t.integer "cvv"
+ t.string "email"
+ t.string "street_address"
+ t.string "state"
+ t.string "city"
+ t.integer "zip"
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ t.string "name_on_card"
+ end
+
+ create_table "products", force: :cascade do |t|
+ t.string "name"
+ t.integer "stock"
+ t.integer "price"
+ t.string "description"
+ t.boolean "status"
+ t.string "image_url"
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ t.integer "user_id"
+ end
+
+ create_table "reviews", force: :cascade do |t|
+ t.integer "rating"
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ t.string "text_review"
+ t.bigint "user_id"
+ t.integer "product_id"
+ end
+
+ create_table "users", force: :cascade do |t|
+ t.string "username"
+ t.string "email"
+ t.integer "uid"
+ t.string "provider"
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ end
+
+end
diff --git a/db/seed_data/categories.csv b/db/seed_data/categories.csv
new file mode 100644
index 0000000000..645390ea15
--- /dev/null
+++ b/db/seed_data/categories.csv
@@ -0,0 +1,4 @@
+id,name
+1,exotic
+2,fluffy
+3,cute
diff --git a/db/seed_data/orders.csv b/db/seed_data/orders.csv
new file mode 100644
index 0000000000..38f4b6ae26
--- /dev/null
+++ b/db/seed_data/orders.csv
@@ -0,0 +1,11 @@
+id,status
+1,pending
+2,paid
+3,complete
+4,cancelled
+5,paid
+6,pending
+7,complete
+8,pending
+9,pending
+10,paid
diff --git a/db/seed_data/products.csv b/db/seed_data/products.csv
new file mode 100644
index 0000000000..1ee4b32032
--- /dev/null
+++ b/db/seed_data/products.csv
@@ -0,0 +1,20 @@
+id,user_id,image_url,name,price,stock,description
+1,2,https://placekitten.com/g/200/300,Fluffy Kitten,400,2,Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s.
+2,2,https://free-images.com/sm/ff13/brownthroated_twotoed_sloth.jpg,Three-Toed Sloth,2800,6,Chill, mellow, and mindful being. Great mediation partner.
+3,1,https://free-images.com/sm/ac35/koala_bear_tree_sitting_0.jpg,Koala Bear,2000,4,Clocking in at an average of 18-22 hours of sleep per day, the Koala is the perfect sleep companion.
+4,3,https://free-images.com/sm/54a4/giant_panda_2.jpg,Giant Panda,13000,1,The Giant Panda usually spends their time sleeping, and foraging for food. The Giant Panda would be a cool hiking mate.
+5,3,https://free-images.com/sm/8eee/armadillo_portrait_standing_looking.jpg,Armadillo,799,2,Night Owl? Armadillos are most active in the evening, and sleep for 18-19 hours a day. Great late night buddy.
+6,4,https://free-images.com/sm/9e2f/animal_bat_brown_close.jpg,Brown Bat,1350,8,Not only do Brown Bats sleep 20 hours a day, but they also hibernate half of the year! What a life, huh? If you need some deep rest, the Brown Bat is the pet for you.
+7,5,https://free-images.com/sm/73c6/eye_dragon_dragon_s.jpg,Lovable Dragon,25000,2,Dragons are among the most popular and enduring of the world's mythological creatures. Dragon tales are known in many cultures, from the Americas to Europe to India to China. Though they populate our books, films, and television shows, they have a long and rich history in many forms.
+8,5,https://free-images.com/sm/ddb7/triceratops_dinosaur_reptile_1640111.jpg,Cute Triceratops,60000,1,Triceratops is a genus of herbivorous ceratopsid dinosaur that first appeared during the late Maastrichtian stage of the late Cretaceous period, about 68 million years ago in what is now North America.
+9,6,https://free-images.com/sm/1a04/maerten_de_vos_unicorn.jpg,Unicorn,80000,1,While many mythic creatures are man-eating monsters or evil spirits, others, like unicorns, are powerful and peaceful.
+10,7,https://free-images.com/sm/360e/narwhal_psf_png.jpg,Nice Narwhal,50000,3,Often dubbed the unicorns of the sea, narwhals are strange and beautiful creatures with long tusks protruding from their heads. Members of the population of more than 80,000 can weigh up to 4,200 pounds and grow as long as 17 feet in length.
+11,9,https://free-images.com/sm/7a67/stone_marble_rock.jpg,pet rock,50,1000,Don’t have the time in your busy schedule for caring for a living pet, but still desire loving companion? Rock your world with a pet rock! This pet will never cause you to get a noise complaint from your finicky neighbors! This pet will love you and express that love quietly. Very quietly. Your cute little rock will also never run away! Pet rock!
+12,10,https://free-images.com/sm/7e74/elephant_baby_sl_c5.jpg,baby elephant,100,10,Sponsor this baby elephant that lives in a protected animal sanctuary because their mother has been taken away due to the illegal ivory trade. THIS IS TRAGIC, RIGHT?!?!? Take a stand against this and also support a cute little baby elephant that drinks milk from a bottle and will grow up to roam the wild rather than living in a tiny stupid zoo! :heart:
+13,2,https://free-images.com/sm/3b44/hamster_pet_animal_small.jpg,hamster,10,20,Hamsters are weird, but that’s okay. They are still great pets! Hamsters love to tunnel and they also like those little woodchip things that sometimes spill and get all over your house, which is really annoying. That’s also okay because it’s worth it for the joy of cohabitating with this little fluffy rodent love balls. Comes with 3 miniature carrots for nibbling.
+14,4,https://free-images.com/sm/1fd3/axolotl_ambystoma_mexicanum_aquarium_1.jpg,Axolotl,25,40,These creatures are crazy! They have the greatest tissue regeneration abilities known to science. They can regrow entire limbs and even regrow damaged brain tissue! They have evolved to never complete their metamorphosis, so they remain in a juvenile state throughout their lives, even reproducing while in this state. This is crazy! They are super weird. They also love to eat worms and they are fun to watch swim around! Remember to get a few little dark spaces for the tank so they can hide in the shadows. They love shadows.
+15,5,https://free-images.com/sm/e5fc/cloud_sky_blue_clouds_5.jpg,cloud,5,1000,Get a pet cloud! Provided by mother nature.
+16,2,https://free-images.com/sm/b1bf/hedgehog_spring_animal_548335.jpg,Hedgehog,45,2,Soft at heart
+17,3,https://free-images.com/sm/e034/crocodile_tooth_reptile_dangerous_3.jpg,Crocodile,200,1,Rhinestone-encrusted collar included
+18,4,https://free-images.com/sm/889f/golden_doodle_dog_forest.jpg,Golden-doodle,800,1,Extreme fluff situations
+19,5,https://free-images.com/sm/2f30/wombat_photo.jpg,Wombat,100,3,Short-legged marsupial native to Australia
diff --git a/db/seed_data/reviews.csv b/db/seed_data/reviews.csv
new file mode 100644
index 0000000000..82b37e1e07
--- /dev/null
+++ b/db/seed_data/reviews.csv
@@ -0,0 +1,11 @@
+id,rating,user_id,product_id
+1,4,1,4
+2,5,3,5
+3,1,5,6
+4,3,3,7
+5,4,2,8
+6,4,10,9
+7,5,4,10
+8,2,3,8
+9,1,5,4
+10,5,2,2
diff --git a/db/seed_data/users.csv b/db/seed_data/users.csv
new file mode 100644
index 0000000000..c7da63385d
--- /dev/null
+++ b/db/seed_data/users.csv
@@ -0,0 +1,11 @@
+id,uid,provider
+1
+2,123456789,github
+3,123456777,github
+4
+5,111222555,github
+6
+7
+8,222222555,github
+9,333222555,github
+10
diff --git a/db/seeds.rb b/db/seeds.rb
new file mode 100644
index 0000000000..a327476c2f
--- /dev/null
+++ b/db/seeds.rb
@@ -0,0 +1,144 @@
+require 'faker'
+require 'csv'
+
+PRODUCT_FILE = Rails.root.join('db', 'seed_data', 'products.csv')
+puts "Loading raw product data from #{PRODUCT_FILE}"
+
+product_failures = []
+CSV.foreach(PRODUCT_FILE, :headers => true) do |row|
+ product = Product.new
+ product.id = row['id']
+ product.user_id = row['user_id']
+ product.image_url = row['image_url']
+ product.name = row['name']
+ product.price = row['price']
+ product.stock = row['stock']
+ product.description = row['description']
+ product.status = true
+ successful = product.save
+ if !successful
+ product_failures << product
+ puts "Failed to save product: #{product.inspect}"
+ else
+ puts "Created product: #{product.inspect}"
+ end
+end
+
+puts "Added #{Product.count} product records"
+puts "#{product_failures.length} products failed to save"
+
+
+
+USER_FILE = Rails.root.join('db', 'seed_data', 'users.csv')
+puts "Loading raw user data from #{USER_FILE}"
+
+user_failures = []
+CSV.foreach(USER_FILE, :headers => true) do |row|
+ user = User.new
+ user.id = row['id']
+ user.username = Faker::Cat.name
+ user.email = Faker::Internet.email
+ user.uid = row['uid']
+ user.provider = row['provider']
+ successful = user.save
+ if !successful
+ user_failures << user
+ puts "Failed to save user: #{user.inspect}"
+ else
+ puts "Created user: #{user.inspect}"
+ end
+end
+
+puts "Added #{User.count} user records"
+puts "#{user_failures.length} users failed to save"
+
+puts "Manually resetting PK sequence on each table"
+ActiveRecord::Base.connection.tables.each do |t|
+ ActiveRecord::Base.connection.reset_pk_sequence!(t)
+end
+
+
+ORDERS_FILE = Rails.root.join('db', 'seed_data', 'orders.csv')
+puts "Loading raw order data from #{ORDERS_FILE}"
+
+order_failures = []
+CSV.foreach(ORDERS_FILE, :headers => true) do |row|
+ order = Order.new
+ order.id = row['id']
+ order.status = row['status']
+ order.name_on_card = Faker::Name.name
+ order.cc_num = Faker::Number.number(4) # Integer of 16 digits can not be held here while the column is just an Interger.. it must be a big integer, or a string
+ order.cvv= Faker::Number.number(3)
+ order.email = Faker::Internet.email
+ order.street_address = Faker::Address.street_address
+ order.state = Faker::Address.state_abbr
+ order.city = Faker::Address.city
+ order.zip = Faker::Address.zip_code
+ successful = order.save
+ if !successful
+ order_failures << order
+ puts "Failed to save order: #{order.inspect}"
+ else
+ puts "Created order: #{order.inspect}"
+ end
+end
+
+puts "Added #{Order.count} order records"
+puts "#{order_failures.length} orders failed to save"
+
+CATEGORIES_FILE = Rails.root.join('db', 'seed_data', 'categories.csv')
+puts "Loading raw categories data from #{CATEGORIES_FILE}"
+
+category_failures = []
+CSV.foreach(CATEGORIES_FILE, :headers => true) do |row|
+ category = Category.new
+ category.id = row['id']
+ category.name = row['name']
+ successful = category.save
+ if !successful
+ category_failures << category
+ puts "Failed to save categories: #{category.inspect}"
+ else
+ puts "Created category: #{category.inspect}"
+ end
+end
+
+puts "Added #{Category.count} category records"
+puts "#{category_failures.length} categories failed to save"
+
+## Commented out since reviews won't work until DB has been updated
+
+REVIEWS_FILE = Rails.root.join('db', 'seed_data', 'reviews.csv')
+puts "Loading raw reviews data from #{REVIEWS_FILE}"
+
+review_failures = []
+CSV.foreach(REVIEWS_FILE, :headers => true) do |row|
+ review = Review.new
+ review.id = row['id']
+ review.user_id = row['user_id']
+ review.product_id = row['product_id']
+ review.rating = row['rating']
+ review.text_review = Faker::Lorem.sentence
+ successful = review.save
+ if !successful
+ review_failures << review
+ puts "Failed to save review: #{review.inspect}"
+ else
+ puts "Created review: #{review.inspect}"
+ end
+end
+
+puts "Added #{Review.count} review records"
+puts "#{review_failures.length} reviews failed to save"
+
+
+# Since we set the primary key (the ID) manually on each of the
+# tables, we've got to tell postgres to reload the latest ID
+# values. Otherwise when we create a new record it will try
+# to start at ID 1, which will be a conflict.
+puts "Manually resetting PK sequence on each table"
+ActiveRecord::Base.connection.tables.each do |t|
+ ActiveRecord::Base.connection.reset_pk_sequence!(t)
+end
+
+puts "done"
diff --git a/lib/assets/.keep b/lib/assets/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/lib/tasks/.keep b/lib/tasks/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/log/.keep b/log/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/package.json b/package.json
new file mode 100644
index 0000000000..f874acf437
--- /dev/null
+++ b/package.json
@@ -0,0 +1,5 @@
+{
+ "name": "betsy",
+ "private": true,
+ "dependencies": {}
+}
diff --git a/public/404.html b/public/404.html
new file mode 100644
index 0000000000..2be3af26fc
--- /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 0000000000..c08eac0d1d
--- /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 0000000000..78a030af22
--- /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 0000000000..e69de29bb2
diff --git a/public/apple-touch-icon.png b/public/apple-touch-icon.png
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/public/favicon.ico b/public/favicon.ico
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/public/robots.txt b/public/robots.txt
new file mode 100644
index 0000000000..37b576a4a0
--- /dev/null
+++ b/public/robots.txt
@@ -0,0 +1 @@
+# See http://www.robotstxt.org/robotstxt.html for documentation on how to use the robots.txt file
diff --git a/test/application_system_test_case.rb b/test/application_system_test_case.rb
new file mode 100644
index 0000000000..d19212abd5
--- /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/controllers/.keep b/test/controllers/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/controllers/categories_controller_test.rb b/test/controllers/categories_controller_test.rb
new file mode 100644
index 0000000000..c3862235f5
--- /dev/null
+++ b/test/controllers/categories_controller_test.rb
@@ -0,0 +1,87 @@
+require "test_helper"
+
+describe CategoriesController do
+ describe "index" do
+ it "succeeds when there are categories" do
+ Category.count.must_equal 3
+ get categories_path
+ must_respond_with :success
+ end
+
+ it "succeeds when there are no categories" do
+ Category.all.each do |item|
+ item.destroy
+ end
+
+ Category.count.must_equal 0
+ get categories_path
+ must_respond_with :success
+ end
+ end
+
+ describe "new" do
+ it "succeeds" do
+ get new_category_path
+ must_respond_with :success
+ end
+ end
+
+ describe "create" do
+ it "creates a product with valid data" do
+ Category.count.must_equal 3
+
+ user = users(:ada)
+ login(user, :github)
+
+ category_params = {
+ name: "orange"
+ }
+
+ Category.new(category_params).must_be :valid?
+
+ post categories_path, params: {category: category_params}
+ must_respond_with :redirect
+ must_redirect_to user_path(user.id)
+ Category.count.must_equal 4
+ end
+
+
+ it "does not create a product with invalid data" do
+ Category.count.must_equal 3
+ user = users(:ada)
+ login(user, :github)
+ category_params = {
+ name: ""
+ }
+
+ Category.new(category_params).wont_be :valid?
+
+ post categories_path, params: {category: category_params}
+ must_respond_with :redirect
+ must_redirect_to user_path(user.id)
+ Category.count.must_equal 3
+ end
+ end
+
+ describe "show" do
+ it "shows a valid category" do
+ product = products(:dragon)
+ product.categories.length.must_equal 0
+
+ category = categories(:three)
+ product.categories << category
+
+ category.products.length.must_equal 1
+
+ get category_path(category)
+ must_respond_with :success
+ end
+
+ it "won's show invalid category" do
+ get category_path("carrot")
+ must_respond_with :redirect
+ must_redirect_to root_path
+ end
+ end
+
+end
diff --git a/test/controllers/order_items_controller_test.rb b/test/controllers/order_items_controller_test.rb
new file mode 100644
index 0000000000..39bea1b430
--- /dev/null
+++ b/test/controllers/order_items_controller_test.rb
@@ -0,0 +1,62 @@
+require "test_helper"
+describe OrderItemsController do
+
+
+ describe "create" do
+ it "creates an order item with valid data for a real product and open order" do
+ product = products(:cat)
+ order = orders(:one)
+
+ proc {
+ post order_items_path, params: { order_item: { product_id: product.id, order_id: order.id, quantity: 1 } }
+ }.must_change 'OrderItem.count', 1
+
+ must_respond_with :redirect
+ must_redirect_to new_order_path
+ end
+
+ it "renders bad_request and does not update the DB for bogus data" do
+
+ proc {
+ post order_items_path, params: { order_item: { product_id: nil } }
+ }.must_change 'OrderItem.count', 0
+
+ must_respond_with :bad_request
+ end
+ end
+
+ describe "update" do
+ it "changes the quantity of the item in pending order" do
+ order = orders(:one)
+ product = products(:dog)
+
+ item = OrderItem.create(product_id: product.id, order_id: order.id, quantity: 1)
+
+
+ proc {
+ put order_item_path(item.id), params: { quantity: { quantity: 3 } }
+ item = OrderItem.find(item.id)
+ }.must_change 'item.quantity', 2
+
+
+ must_respond_with :redirect
+ must_redirect_to new_order_path
+ end
+ end
+
+ describe "destroy" do
+ it "succeeds for an extant order item ID" do
+ item = order_items(:one)
+ proc {
+ delete order_item_path(item.id)}.must_change 'OrderItem.count', -1
+ end
+
+ it "renders 404 not_found and does not update the DB for a bogus order item ID" do
+ proc {
+ delete order_item_path('abc')}.must_change 'OrderItem.count', 0
+
+ must_respond_with :bad_request
+ end
+
+ end
+end
diff --git a/test/controllers/orders_controller_test.rb b/test/controllers/orders_controller_test.rb
new file mode 100644
index 0000000000..1f46b95afc
--- /dev/null
+++ b/test/controllers/orders_controller_test.rb
@@ -0,0 +1,120 @@
+require "test_helper"
+require 'pry'
+
+describe OrdersController do
+
+ describe "index" do
+
+ it "should get index" do
+ get orders_path
+ must_respond_with :success
+ end
+
+ it "succeeds with no orders" do
+
+ orders = Order.all
+ orders.destroy_all
+
+ get orders_path
+
+ must_respond_with :success
+ end
+ end
+
+
+ describe "new" do
+ it "succeeds" do
+ get new_order_path
+ must_respond_with :success
+ end
+ end
+
+
+
+ describe "update" do
+ it "updates an order with valid data" do
+ new_order = orders(:one)
+
+ order_info = {
+ name_on_card: "Bunny",
+ cc_num: "1234123412341234",
+ cvv: 121,
+ email: "hello@hi.org",
+ street_address: "111 Candy Cane Lane",
+ state: "NP",
+ city: "Santa",
+ zip: 54321,
+ status: "paid"
+ }
+
+ proc {
+ put order_path(new_order.id), params: { order: order_info }
+ }.must_change 'Order.where(status: "paid").count', 1
+
+ placed_order = Order.find_by(cc_num: 1234123412341234)
+
+ placed_order.status.must_equal "paid"
+ must_respond_with :redirect
+ must_redirect_to order_path(placed_order.id)
+ end
+
+ it "renders bad_request and does not update the DB for bogus data" do
+ order = orders(:one)
+
+ bad_order_hash = {
+ name_on_card: nil,
+ }
+
+ proc {
+ put order_path(order.id), params: { order: bad_order_hash }
+ }.must_change 'Order.where(status: "paid").count', 0
+
+ must_respond_with :bad_request
+ end
+
+ it "reduces the quantity of each product on the order" do
+ order = Order.create(status: "pending")
+
+ product = products(:cat)
+
+ item = OrderItem.new(product_id: product.id, order_id: order.id, quantity: 2)
+ item.save
+
+ order_info = {
+ name_on_card: "Bunny",
+ cc_num: "1234123412341234",
+ cvv: 121,
+ email: "hello@hi.org",
+ street_address: "111 Candy Cane Lane",
+ state: "NP",
+ city: "Santa",
+ zip: 54321,
+ status: "paid"
+ }
+
+ proc {
+ put order_path(order.id), params: { order: order_info }
+ product = Product.find(item.product_id)
+ }.must_change 'product.stock', -2
+
+
+ end
+ end
+
+
+ describe "show" do
+ it "succeeds for an extant order ID" do
+ order = orders(:one)
+
+ get order_path(order.id)
+ must_respond_with :success
+
+ end
+
+ it "renders 404 not_found for a bogus order ID" do
+ get order_path("abc")
+ must_respond_with :not_found
+ end
+ end
+
+end
diff --git a/test/controllers/product.rb b/test/controllers/product.rb
new file mode 100644
index 0000000000..71a1daa404
--- /dev/null
+++ b/test/controllers/product.rb
@@ -0,0 +1,38 @@
+class Product < ApplicationRecord
+ has_and_belongs_to_many :categories
+ has_many :order_items
+ has_many :orders, through: :order_items
+ has_many :reviews
+
+ # validates :user_id, presence: true
+
+ validates :name, presence: true
+ validates :price, presence: true, numericality: {greater_than: 0}
+ validates :stock, presence: true, numericality: {greater_than_or_equal_to: 0, only_integer: true}
+
+ def average_rating
+ return reviews.average(:rating).to_i
+ end
+
+ def self.by_category(category)
+ assigned = Product.where.not(category: [])
+ selected = assigned.select {|product| product.categories.include?(id)}
+
+ return selected
+ end
+
+ def self.by_merchant(id)
+ return Product.where(user_id: id)
+ end
+
+
+ def change_stock(quantity)
+ total = self.stock
+ number = total - quantity
+ self.stock = number
+ self.save
+
+ end
+
+
+end
diff --git a/test/controllers/products_controller_test.rb b/test/controllers/products_controller_test.rb
new file mode 100644
index 0000000000..f5c9b6d0ec
--- /dev/null
+++ b/test/controllers/products_controller_test.rb
@@ -0,0 +1,135 @@
+require "test_helper"
+require "pry"
+
+describe ProductsController do
+ describe "index" do
+ it "succeeds when there are works" do
+ Product.count.must_equal 3
+ get products_path
+ must_respond_with :success
+ end
+
+ it "succeeds when there are no works" do
+ Product.all.each do |item|
+ item.destroy
+ end
+
+ Product.count.must_equal 0
+ get products_path
+ must_respond_with :success
+ end
+ end
+
+ describe "show" do
+ it "succeeds" do
+ get product_path(products(:cat))
+ must_respond_with :success
+ end
+
+ # it "redirects if not in db" do
+ # get product_path(nil)
+ # must_redirect_to products_path
+ # end
+ end
+
+ describe "new" do
+ it "succeeds" do
+ user = users(:ada)
+ get new_user_product_path(user.id)
+ must_respond_with :success
+ end
+ end
+
+ describe "create" do
+ it "creates a product with valid data" do
+ Product.count.must_equal 3
+ user = users(:ada)
+
+ post user_products_path(user.id), params: {
+ product: {
+ name: "Bunny",
+ stock: 14,
+ price: 5,
+ description: "Too many, please help",
+ status: true,
+ user_id: user.id,
+ image_url: "www.test-URL.com",
+ categories: [categories(:one)]
+ }
+ }
+ Product.count.must_equal 4
+ must_respond_with :redirect
+ must_redirect_to products_path
+ end
+
+ it "won't create invalid product" do
+ proc {
+ user = users(:ada)
+
+ post user_products_path(user.id), params: {
+ product: {
+ stock: 14,
+ price: 5,
+ description: "Too many, please help",
+ status: true,
+ user_id: user.id,
+ image_url: "www.test-URL.com"
+ }
+ }
+ }.must_change "Product.count", 0
+
+ Product.count.must_equal 3
+ end
+ end
+
+ describe "edit" do
+ it "succeeds for an extant product ID" do
+ get edit_user_product_path(users(:ada).id, products(:dragon).id,)
+ must_respond_with :success
+ end
+
+ # it "renders 404 not_found for a bogus work ID" do
+ # get edit_product_path("carrot")
+ # must_respond_with :not_found
+ # end
+
+ end
+
+ describe "update" do
+ it "succeeds for valid data and an extant work ID" do
+ updated_name = "Donkey"
+
+ patch user_product_path(users(:ada).id, products(:dragon).id), params: {
+ product: {
+ name: "Donkey"
+ }
+ }
+
+ updated_product = Product.find(products(:dragon).id)
+ updated_product.name.must_equal updated_name
+ must_respond_with :redirect
+ end
+
+ # it "renders bad_request for bogus data" do
+ # product = products(:dragon)
+ # put product_path(product.id), params: {
+ # product: {
+ # name: ""
+ # }
+ # }
+ #
+ # product.name.must_equal "Norweigan Ridgeback"
+ # end
+
+
+ # it "renders 404 not_found for a bogus product ID" do
+ # put product_path(products(:dragon).id), params: {
+ # product: {
+ # id: "deeface"
+ # }
+ # }
+ # must_respond_with :not_found
+ # end
+ end
+
+end
diff --git a/test/controllers/reviews_controller_test.rb b/test/controllers/reviews_controller_test.rb
new file mode 100644
index 0000000000..cbb826a5ff
--- /dev/null
+++ b/test/controllers/reviews_controller_test.rb
@@ -0,0 +1,58 @@
+require "test_helper"
+
+describe ReviewsController do
+ describe "new" do
+ it "returns success when making a new review" do
+ get new_product_review_path(Product.first.id)
+
+ must_respond_with :success
+ end
+
+ it "returns not found when making a review to product doesn't exists" do
+ get new_product_review_path(Product.last.id + 1)
+ must_respond_with :not_found
+ end
+ end
+
+ describe "create" do
+ it "redirects to product's page when saving a review" do
+ Review.count.must_equal 2
+ proc {
+ post product_reviews_path(products(:dog).id), params: {
+ review: {
+ text_review: "This is another review that will be created",
+ rating: 2,
+ product_id: products(:dog).id
+ }
+ }
+ }.must_change "Review.count", 1
+
+ must_respond_with :redirect
+ must_redirect_to product_path(products(:dog).id)
+ end
+
+ it "sends bad_request when the review data is bogus" do
+ # Arrange
+ invalid_review_data = {
+ review: {
+ #invalid rating, rating should be between 0 and 5
+ rating: 15,
+ product_id: Product.first.id
+ }
+ }
+ # Double check the data is truly invalid
+ Review.new(invalid_review_data[:review]).wont_be :valid?
+
+ start_review_count = Review.count
+
+ # Act
+ post product_reviews_path(invalid_review_data[:review][:product_id]), params: invalid_review_data
+
+ # Assert
+ must_respond_with :bad_request
+ Review.count.must_equal start_review_count
+ end
+
+ end
+
+end
diff --git a/test/controllers/sessions_controller_test.rb b/test/controllers/sessions_controller_test.rb
new file mode 100644
index 0000000000..6109751716
--- /dev/null
+++ b/test/controllers/sessions_controller_test.rb
@@ -0,0 +1,28 @@
+require "test_helper"
+require 'pry'
+describe SessionsController do
+
+ it "logs in an existing user and redirect to root_path" do
+ user = users(:ada)
+ start_count = User.count
+ login(user, :github)
+ User.count.must_equal start_count
+ must_redirect_to root_path
+ session[:user_id].must_equal user.id
+ end
+
+ it "must have a username" do
+ user = users(:ada)
+ login(user, :github)
+ user.username.must_equal "countess_ada"
+ end
+
+ it "clears the session and redirects back to the root path when a merchant logs out" do
+ user = users(:ada)
+ login(user, :github)
+ delete logout_path
+ session[:user_id].must_equal nil
+ must_redirect_to root_path
+ end
+
+end
diff --git a/test/controllers/users_controller_test.rb b/test/controllers/users_controller_test.rb
new file mode 100644
index 0000000000..4ca0949d13
--- /dev/null
+++ b/test/controllers/users_controller_test.rb
@@ -0,0 +1,51 @@
+require "test_helper"
+
+describe UsersController do
+ describe "index" do
+
+ it "should get index" do
+ get users_path
+ must_respond_with :success
+ end
+
+ it "succeeds with no users" do
+ users(:ada).destroy
+ users(:grace).destroy
+ users(:gretchen).destroy
+
+ users.count.must_equal 0
+
+ end
+ end
+
+ describe "show" do
+ it "should get show" do
+ id = users(:ada).id
+ get users_path(id)
+
+ must_respond_with :success
+ end
+ end
+
+ describe "update" do
+ it "changes user information" do
+ updated_username = "CandyCanes111"
+ put user_path(users(:grace).id), params: { user: { username: updated_username} }
+
+ updated_user = User.find(users(:grace).id)
+ updated_user.username.must_equal "CandyCanes111"
+ end
+
+ it "will not change username to one that already exists" do
+ updated_username = users(:grace).username
+ initial_username = users(:ada).username
+
+ put user_path(users(:ada).id), params: { user: { username: updated_username} }
+
+ updated_user = User.find(users(:ada).id)
+ updated_user.username.must_equal initial_username
+
+ end
+ end
+
+end
diff --git a/test/fixtures/.keep b/test/fixtures/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/fixtures/categories.yml b/test/fixtures/categories.yml
new file mode 100644
index 0000000000..0d5d28db3f
--- /dev/null
+++ b/test/fixtures/categories.yml
@@ -0,0 +1,8 @@
+# Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+one:
+ name: cats
+two:
+ name: dogs
+three:
+ name: exotic
diff --git a/test/fixtures/files/.keep b/test/fixtures/files/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/fixtures/order_items.yml b/test/fixtures/order_items.yml
new file mode 100644
index 0000000000..1fce0fff92
--- /dev/null
+++ b/test/fixtures/order_items.yml
@@ -0,0 +1,11 @@
+# Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+one:
+ quantity: 1
+ order: one
+ product: cat
+
+two:
+ quantity: 1
+ order: two
+ product: dog
diff --git a/test/fixtures/orders.yml b/test/fixtures/orders.yml
new file mode 100644
index 0000000000..079cd8d93e
--- /dev/null
+++ b/test/fixtures/orders.yml
@@ -0,0 +1,23 @@
+# Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+one:
+ status: pending
+ name_on_card: Ada Lovelace
+ cc_num: 1234
+ cvv: 123
+ email: ada@adadevelopersacademy.org
+ street_address: 123 Pine Street
+ state: WA
+ city: Seattle
+ zip: 98104
+
+two:
+ status: paid
+ name_on_card: Gretchen Greps
+ cc_num: 4321
+ cvv: 321
+ email: gretchen@adadevelopersacademy.org
+ street_address: 321 Pike street
+ state: WA
+ city: Seattle
+ zip: 98122
diff --git a/test/fixtures/products.yml b/test/fixtures/products.yml
new file mode 100644
index 0000000000..2ddada46cc
--- /dev/null
+++ b/test/fixtures/products.yml
@@ -0,0 +1,26 @@
+# Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+cat:
+ name: British Short-Hair
+ stock: 4
+ price: 500
+ description: Excellent company on a rainy day
+ status: true
+ user_id: ada.id
+ image_url: https://placebear.com/g/500/400
+dog:
+ name: Cavalier King Charles Spaniel
+ stock: 2
+ price: 2750
+ description: Each and every one is a unique princess
+ status: false
+ user_id: grace.id
+ image_url: https://placebear.com/g/500/400
+dragon:
+ name: Norweigan Ridgeback
+ stock: 1
+ price: 25000
+ description: Very Rare; in excellent health
+ status: true
+ user_id: gretchen.id
+ image_url: https://placebear.com/g/500/400
diff --git a/test/fixtures/reviews.yml b/test/fixtures/reviews.yml
new file mode 100644
index 0000000000..1c2321ebdd
--- /dev/null
+++ b/test/fixtures/reviews.yml
@@ -0,0 +1,12 @@
+# Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+one:
+ rating: 1
+ product: dragon
+ user: ada
+ text_review: This dragon is a cute dragon
+two:
+ rating: 1
+ product: dragon
+ user: gretchen
+ text_review: This is the loudest pet I have ever owned. It burned my house down.
diff --git a/test/fixtures/users.yml b/test/fixtures/users.yml
new file mode 100644
index 0000000000..9ca2cfb381
--- /dev/null
+++ b/test/fixtures/users.yml
@@ -0,0 +1,22 @@
+# Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
+
+ada:
+ provider: github
+ uid: 12345
+ email: ada@adadevelopersacademy.org
+ username: countess_ada
+grace:
+ provider: github
+ uid: 13371337
+ email: grace@hooper.net
+ username: graceful_hoops
+gretchen:
+ provider: github
+ uid: 9999089
+ email: gretchen@adadevelopersacademy.org
+ username: gigi
+lily:
+ provider: github
+ uid: 999978
+ email: lily@adadevelopersacademy.org
+ username: lillerz
diff --git a/test/helpers/.keep b/test/helpers/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/integration/.keep b/test/integration/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/mailers/.keep b/test/mailers/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/models/.keep b/test/models/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/models/category_test.rb b/test/models/category_test.rb
new file mode 100644
index 0000000000..781320ad8e
--- /dev/null
+++ b/test/models/category_test.rb
@@ -0,0 +1,9 @@
+require "test_helper"
+
+describe Category do
+ let(:category) { Category.new }
+
+ it "must be valid" do
+ value(category).must_be :valid?
+ end
+end
diff --git a/test/models/order_item_test.rb b/test/models/order_item_test.rb
new file mode 100644
index 0000000000..9b8a879914
--- /dev/null
+++ b/test/models/order_item_test.rb
@@ -0,0 +1,29 @@
+require "test_helper"
+
+describe OrderItem do
+ let(:order_item) { OrderItem.new(product_id: products(:cat).id, order_id: orders(:one).id, quantity: 1) }
+
+ it "must be valid" do
+ value(order_item).must_be :valid?
+ end
+
+ describe "get subtotal" do
+ it "returns correct subtotal for quantity" do
+ order = orders(:one)
+ product = products(:cat)
+ order_item.quantity = 3
+ sum = product.price * 3
+
+ order_item.get_subtotal.must_equal sum
+ end
+ end
+
+ describe "already_exists" do
+ it "returns true if product is already in pending order" do
+ order = orders(:one)
+ second = OrderItem.new(product_id: products(:cat).id, order_id: orders(:one).id, quantity: 2)
+
+ second.already_exists?.must_equal true
+ end
+ end
+end
diff --git a/test/models/order_test.rb b/test/models/order_test.rb
new file mode 100644
index 0000000000..44020f1076
--- /dev/null
+++ b/test/models/order_test.rb
@@ -0,0 +1,41 @@
+require "test_helper"
+
+describe Order do
+
+ describe "get total" do
+ it "returns order total" do
+ order = orders(:one)
+ existing_item = order.order_items.first
+ existing_item.destroy
+
+ product1 = products(:cat)
+ product2 = products(:dog)
+ item1 = OrderItem.create(product_id: product1.id, order_id: order.id, quantity: 1)
+ item2 = OrderItem.create(product_id: product2.id, order_id: order.id, quantity: 1)
+
+ sum = product1.price + product2.price
+
+ order.get_total.must_equal sum
+ end
+ end
+
+ describe "merchant products" do
+ it "returns a list of the merchant's products" do
+ merchant = users(:ada)
+ product1 = products(:cat)
+ product1.update(user_id: merchant.id)
+ product2 = products(:dog)
+ product2.update(user_id: merchant.id)
+
+ merchant_product = merchant.products.length
+ merchant_product.must_equal 2
+ end
+
+ it "doesn't return any product if merchant has zero" do
+ merchant = users(:ada)
+
+ Order.merchant_products(merchant.id).must_be_empty
+ end
+ end
+
+end
diff --git a/test/models/product_test.rb b/test/models/product_test.rb
new file mode 100644
index 0000000000..5996bcdc30
--- /dev/null
+++ b/test/models/product_test.rb
@@ -0,0 +1,173 @@
+require "test_helper"
+
+describe Product do
+ describe "validations" do
+ before do
+ @product = products(:dog)
+ end
+
+ describe "price" do
+ it "has a price" do
+ @product.price.must_equal 2750.00
+ end
+
+
+ it "must have a valid price" do
+ @product.price = 0
+ @product.valid?.must_equal false
+ end
+ end
+
+ describe "name" do
+ it "must have a name" do
+ @product.name.must_equal "Cavalier King Charles Spaniel"
+ end
+
+ it "must have a valid name" do
+ @product.name = ""
+ @product.valid?.must_equal false
+ end
+ end
+
+ describe "stock" do
+ it "must be a number" do
+ @product.stock = "c"
+ @product.valid?.must_equal false
+ end
+
+ it "can be equal to 0" do
+ @product.stock = 0
+ @product.valid?.must_equal true
+ end
+
+ it "must not be less than 0" do
+ @product.stock = -2
+ @product.valid?.must_equal false
+ end
+ end
+ end
+
+ describe "relations" do
+ before do
+ @product = products(:dragon)
+ end
+
+ describe "reviews" do
+ it "responds to reviews with no reviews" do
+ a = products(:dog)
+ a.reviews.count.must_equal 0
+ end
+
+ it "responds to reviews" do
+ @product.reviews.count.must_equal 2
+ end
+
+ it "can register new reviews" do
+ Review.create({
+ text_review: "This is another review that will be created",
+ rating: 2,
+ product_id: @product.id,
+ user_id: users(:ada)
+ })
+
+ @product.reviews.count.must_equal 3
+ end
+ end
+
+ describe "categories" do
+ it "responds to categories with no categories added" do
+ @product.categories.length.must_equal 0
+ end
+
+ it "responds to categories with added categories" do
+ @product.categories << categories(:three)
+ @product.categories.length.must_equal 1
+ end
+ end
+ end
+
+ describe "methods" do
+ describe "by_category" do
+ it "returns an empty array if category has no items" do
+ Product.by_category(categories(:one)).length.must_equal 0
+ Product.by_category(categories(:one)).must_equal []
+ end
+
+ it "returns an item assigned to a category" do
+ product = products(:cat)
+ product.categories << categories(:one)
+ product.categories << categories(:three)
+
+ product.categories.must_include categories(:one)
+
+ filtered = Product.by_category(categories(:one))
+
+ filtered.length.must_equal 1
+ filtered.first.name.must_equal product.name
+ end
+
+ it "returns multiple items assigned to a category" do
+ Product.by_category(categories(:three)).must_equal []
+
+ product_1 = products(:cat)
+ product_2 = products(:dragon)
+ product_1.categories << categories(:three)
+ product_2.categories << categories(:three)
+
+ filtered = Product.by_category(categories(:three))
+
+ filtered.length.must_equal 2
+ end
+ end
+
+ describe "by_merchant" do #is this method redundant?
+ before do
+ @user = users(:lily)
+ end
+ it "returns an empty array if merchant has no items" do
+ @user.products.must_equal []
+ Product.by_merchant(@user.id).must_equal []
+ end
+
+ it "returns an item assigned to a user" do
+ data = {
+ name: "Bunny",
+ stock: 14,
+ price: 5,
+ description: "Too many, please help",
+ status: true,
+ user_id: @user.id,
+ image_url: "www.test-URL.com"
+ }
+
+ Product.create(data)
+ Product.by_merchant(@user.id).length.must_equal 1
+ end
+
+ it "returns multiple items assigned to a category" do
+
+ Product.create({
+ name: "Jaguar",
+ stock: 2,
+ price: 500,
+ description: "Excellent",
+ status: true,
+ user_id: @user.id,
+ image_url: "https://placebear.com/g/500/400"
+ })
+
+ Product.create({
+ name: "Bunny",
+ stock: 14,
+ price: 5,
+ description: "Too many, please help",
+ status: true,
+ user_id: @user.id,
+ image_url: "www.test-URL.com"
+ })
+
+ Product.by_merchant(@user.id).length.must_equal 2
+ end
+ end
+ end
+end
diff --git a/test/models/review_test.rb b/test/models/review_test.rb
new file mode 100644
index 0000000000..df782c302c
--- /dev/null
+++ b/test/models/review_test.rb
@@ -0,0 +1,42 @@
+require "test_helper"
+
+describe Review do
+ let(:review) { reviews(:one) }
+
+ describe "#valid?" do
+
+ it "should return false without a rating" do
+ review.rating = nil
+ review.valid?.must_equal false
+ end
+
+ it "should return false if rating is not between 1 and 5" do
+ review.rating = 6
+ review.valid?.must_equal false
+ end
+
+ it "should return false if rating is a letter" do
+ review.rating = "b"
+ review.valid?.must_equal false
+ end
+
+ it "should no more than 500 character" do
+
+ review.text_review = "Aenean commodo ligula eget dolor. Aenean massa. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Donec quam felis, ultricies nec, pellentesque eu, pretium quis, sem. Nulla consequat massa quis enim. Donec pede justo, fringilla vel, aliquet nec, vulputate eget, arcu. In enim justo, rhoncus ut, imperdiet a, venenatis vitae, justo. Nullam dictum felis eu pede mollis pretium. Integer tincidunt. Cras dapibusAenean commodo ligula eget dolor. Aenean massa. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Donec quam felis, ultricies nec, pellentesque eu, pretium quis, sem. Nulla consequat massa quis enim. Donec pede justo, fringilla vel, aliquet nec, vulputate eget, arcu. In enim justo, rhoncus ut, imperdiet a, venenatis vitae, justo. Nullam dictum felis eu pede mollis pretium. Integer tincidunt. Cras dapibus"
+ review.valid?.must_equal false
+
+ end
+ end
+
+ describe "#product" do
+ it "should return the associated product" do
+ review.product.must_equal products(:dragon)
+ end
+
+ it "should return the correct number of reviews" do
+ products(:dragon).reviews.count.must_equal 2
+ end
+ end
+
+
+end
diff --git a/test/models/user_test.rb b/test/models/user_test.rb
new file mode 100644
index 0000000000..d378d8530c
--- /dev/null
+++ b/test/models/user_test.rb
@@ -0,0 +1,9 @@
+require "test_helper"
+
+describe User do
+
+
+ it "must be valid" do
+ value(users(:ada)).must_be :valid?
+ end
+end
diff --git a/test/system/.keep b/test/system/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/test_helper.rb b/test/test_helper.rb
new file mode 100644
index 0000000000..49f55b39e2
--- /dev/null
+++ b/test/test_helper.rb
@@ -0,0 +1,55 @@
+require 'simplecov'
+SimpleCov.start
+ENV["RAILS_ENV"] = "test"
+require File.expand_path("../../config/environment", __FILE__)
+require "rails/test_help"
+require "minitest/rails"
+require "minitest/reporters" # for Colorized output
+
+
+# For colorful output!
+Minitest::Reporters.use!(
+ Minitest::Reporters::SpecReporter.new,
+ ENV,
+ Minitest.backtrace_filter
+)
+
+
+# To add Capybara feature tests add `gem "minitest-rails-capybara"`
+# to the test group in the Gemfile and uncomment the following:
+# require "minitest/rails/capybara"
+
+# Uncomment for awesome colorful output
+# require "minitest/pride"
+
+class ActiveSupport::TestCase
+ # Setup all fixtures in test/fixtures/*.yml for all tests in alphabetical order.
+ fixtures :all
+
+ def setup
+ OmniAuth.config.test_mode = true
+ end
+
+ def login(user, provider)
+ OmniAuth.config.mock_auth[provider] = OmniAuth::AuthHash.new(mock_auth_hash(user))
+ get auth_callback_path(provider)
+
+ end
+ def logout(user, provider)
+ OmniAuth.config.mock_auth[provider] = OmniAuth::AuthHash.new(mock_auth_hash(user))
+ delete auth_callback_path(provider)
+
+ end
+
+ def mock_auth_hash(user)
+ return {
+ provider: user.provider,
+ uid: user.uid,
+ info: {
+ email: user.email,
+ nickname: user.username
+ }
+ }
+ end
+end
+ # Add more helper methods to be used by all tests here...
diff --git a/tmp/.keep b/tmp/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/vendor/.keep b/vendor/.keep
new file mode 100644
index 0000000000..e69de29bb2