diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000000..18b43c9cd2
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,27 @@
+# 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 uploaded files in development
+/storage/*
+!/storage/.keep
+
+/node_modules
+/yarn-error.log
+
+/public/assets
+.byebug_history
+
+# Ignore master key for decrypting credentials and more.
+/config/master.key
diff --git a/.ruby-version b/.ruby-version
new file mode 100644
index 0000000000..160fe391c8
--- /dev/null
+++ b/.ruby-version
@@ -0,0 +1 @@
+2.5.5
\ No newline at end of file
diff --git a/Gemfile b/Gemfile
new file mode 100644
index 0000000000..536ec6fc3e
--- /dev/null
+++ b/Gemfile
@@ -0,0 +1,86 @@
+source 'https://rubygems.org'
+git_source(:github) { |repo| "https://github.com/#{repo}.git" }
+
+ruby '2.5.5'
+
+# Bundle edge Rails instead: gem 'rails', github: 'rails/rails'
+gem 'rails', '~> 5.2.3'
+# Use postgresql as the database for Active Record
+gem 'pg', '>= 0.18', '< 2.0'
+# Use Puma as the app server
+gem 'puma', '~> 3.11'
+# Use SCSS for stylesheets
+gem 'sass-rails', '~> 5.0'
+# Use Uglifier as compressor for JavaScript assets
+gem 'uglifier', '>= 1.3.0'
+# See https://github.com/rails/execjs#readme for more supported runtimes
+# gem 'mini_racer', platforms: :ruby
+
+# Use CoffeeScript for .coffee assets and views
+# gem 'coffee-rails', '~> 4.2'
+# Turbolinks makes navigating your web application faster. Read more: https://github.com/turbolinks/turbolinks
+gem 'turbolinks', '~> 5'
+# Build JSON APIs with ease. Read more: https://github.com/rails/jbuilder
+gem 'jbuilder', '~> 2.5'
+# Use Redis adapter to run Action Cable in production
+# gem 'redis', '~> 4.0'
+# Use ActiveModel has_secure_password
+# gem 'bcrypt', '~> 3.1.7'
+
+# Use ActiveStorage variant
+# gem 'mini_magick', '~> 4.8'
+
+# Use Capistrano for deployment
+# gem 'capistrano-rails', group: :development
+
+# Reduces boot times through caching; required in config/boot.rb
+gem 'bootsnap', '>= 1.1.0', require: false
+
+group :development, :test do
+ # Call 'byebug' anywhere in the code to stop execution and get a debugger console
+ gem 'byebug', platforms: [:mri, :mingw, :x64_mingw]
+end
+
+group :development do
+ # Access an interactive console on exception pages or by calling 'console' anywhere in the code.
+ gem 'web-console', '>= 3.3.0'
+ gem 'listen', '>= 3.0.5', '< 3.2'
+ # Spring speeds up development by keeping your application running in the background. Read more: https://github.com/rails/spring
+ gem 'spring'
+ gem 'spring-watcher-listen', '~> 2.0.0'
+end
+
+group :test do
+ # Adds support for Capybara system testing and selenium driver
+ gem 'capybara', '>= 2.15'
+ gem 'selenium-webdriver'
+ # Easy installation and use of chromedriver to run system tests with Chrome
+ gem 'chromedriver-helper'
+end
+
+# Windows does not include zoneinfo files, so bundle the tzinfo-data gem
+gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby]
+
+gem 'jquery-rails'
+gem 'jquery-turbolinks'
+gem 'bootstrap', '~> 4.1.3'
+group :development, :test do
+ gem 'pry-rails'
+end
+
+group :development do
+ gem 'debase', '>= 0.2.4.1'
+ gem 'ruby-debug-ide', '>= 0.7.0'
+end
+
+group :development do
+ gem 'better_errors'
+ gem 'binding_of_caller'
+ gem 'guard'
+ gem 'guard-minitest'
+end
+
+group :test do
+ gem 'minitest-rails'
+ gem 'minitest-reporters'
+end
diff --git a/Gemfile.lock b/Gemfile.lock
new file mode 100644
index 0000000000..031d0f2eff
--- /dev/null
+++ b/Gemfile.lock
@@ -0,0 +1,282 @@
+GEM
+ remote: https://rubygems.org/
+ specs:
+ actioncable (5.2.3)
+ actionpack (= 5.2.3)
+ nio4r (~> 2.0)
+ websocket-driver (>= 0.6.1)
+ actionmailer (5.2.3)
+ actionpack (= 5.2.3)
+ actionview (= 5.2.3)
+ activejob (= 5.2.3)
+ mail (~> 2.5, >= 2.5.4)
+ rails-dom-testing (~> 2.0)
+ actionpack (5.2.3)
+ actionview (= 5.2.3)
+ activesupport (= 5.2.3)
+ rack (~> 2.0)
+ rack-test (>= 0.6.3)
+ rails-dom-testing (~> 2.0)
+ rails-html-sanitizer (~> 1.0, >= 1.0.2)
+ actionview (5.2.3)
+ activesupport (= 5.2.3)
+ builder (~> 3.1)
+ erubi (~> 1.4)
+ rails-dom-testing (~> 2.0)
+ rails-html-sanitizer (~> 1.0, >= 1.0.3)
+ activejob (5.2.3)
+ activesupport (= 5.2.3)
+ globalid (>= 0.3.6)
+ activemodel (5.2.3)
+ activesupport (= 5.2.3)
+ activerecord (5.2.3)
+ activemodel (= 5.2.3)
+ activesupport (= 5.2.3)
+ arel (>= 9.0)
+ activestorage (5.2.3)
+ actionpack (= 5.2.3)
+ activerecord (= 5.2.3)
+ marcel (~> 0.3.1)
+ activesupport (5.2.3)
+ concurrent-ruby (~> 1.0, >= 1.0.2)
+ i18n (>= 0.7, < 2)
+ minitest (~> 5.1)
+ tzinfo (~> 1.1)
+ addressable (2.7.0)
+ public_suffix (>= 2.0.2, < 5.0)
+ ansi (1.5.0)
+ archive-zip (0.12.0)
+ io-like (~> 0.3.0)
+ arel (9.0.0)
+ autoprefixer-rails (9.6.5)
+ execjs
+ better_errors (2.5.1)
+ coderay (>= 1.0.0)
+ erubi (>= 1.0.0)
+ rack (>= 0.9.0)
+ bindex (0.8.1)
+ binding_of_caller (0.8.0)
+ debug_inspector (>= 0.0.1)
+ bootsnap (1.4.5)
+ msgpack (~> 1.0)
+ bootstrap (4.1.3)
+ autoprefixer-rails (>= 6.0.3)
+ popper_js (>= 1.12.9, < 2)
+ sass (>= 3.5.2)
+ builder (3.2.3)
+ byebug (11.0.1)
+ capybara (3.29.0)
+ addressable
+ mini_mime (>= 0.1.3)
+ nokogiri (~> 1.8)
+ rack (>= 1.6.0)
+ rack-test (>= 0.6.3)
+ regexp_parser (~> 1.5)
+ xpath (~> 3.2)
+ childprocess (3.0.0)
+ chromedriver-helper (2.1.1)
+ archive-zip (~> 0.10)
+ nokogiri (~> 1.8)
+ coderay (1.1.2)
+ concurrent-ruby (1.1.5)
+ crass (1.0.4)
+ debase (0.2.4.1)
+ debase-ruby_core_source (>= 0.10.2)
+ debase-ruby_core_source (0.10.6)
+ debug_inspector (0.0.3)
+ erubi (1.9.0)
+ execjs (2.7.0)
+ ffi (1.11.1)
+ formatador (0.2.5)
+ globalid (0.4.2)
+ activesupport (>= 4.2.0)
+ guard (2.15.1)
+ formatador (>= 0.2.4)
+ listen (>= 2.7, < 4.0)
+ lumberjack (>= 1.0.12, < 2.0)
+ nenv (~> 0.1)
+ notiffany (~> 0.0)
+ pry (>= 0.9.12)
+ shellany (~> 0.0)
+ thor (>= 0.18.1)
+ guard-compat (1.2.1)
+ guard-minitest (2.4.6)
+ guard-compat (~> 1.2)
+ minitest (>= 3.0)
+ i18n (1.7.0)
+ concurrent-ruby (~> 1.0)
+ io-like (0.3.0)
+ jbuilder (2.9.1)
+ activesupport (>= 4.2.0)
+ jquery-rails (4.3.5)
+ rails-dom-testing (>= 1, < 3)
+ railties (>= 4.2.0)
+ thor (>= 0.14, < 2.0)
+ jquery-turbolinks (2.1.0)
+ railties (>= 3.1.0)
+ turbolinks
+ listen (3.1.5)
+ rb-fsevent (~> 0.9, >= 0.9.4)
+ rb-inotify (~> 0.9, >= 0.9.7)
+ ruby_dep (~> 1.2)
+ loofah (2.3.0)
+ crass (~> 1.0.2)
+ nokogiri (>= 1.5.9)
+ lumberjack (1.0.13)
+ mail (2.7.1)
+ mini_mime (>= 0.1.1)
+ marcel (0.3.3)
+ mimemagic (~> 0.3.2)
+ method_source (0.9.2)
+ mimemagic (0.3.3)
+ mini_mime (1.0.2)
+ mini_portile2 (2.4.0)
+ minitest (5.12.2)
+ minitest-rails (5.2.0)
+ minitest (~> 5.10)
+ railties (~> 5.2.0)
+ minitest-reporters (1.4.1)
+ ansi
+ builder
+ minitest (>= 5.0)
+ ruby-progressbar
+ msgpack (1.3.1)
+ nenv (0.3.0)
+ nio4r (2.5.2)
+ nokogiri (1.10.4)
+ mini_portile2 (~> 2.4.0)
+ notiffany (0.1.3)
+ nenv (~> 0.1)
+ shellany (~> 0.0)
+ pg (1.1.4)
+ popper_js (1.14.5)
+ pry (0.12.2)
+ coderay (~> 1.1.0)
+ method_source (~> 0.9.0)
+ pry-rails (0.3.9)
+ pry (>= 0.10.4)
+ public_suffix (4.0.1)
+ puma (3.12.1)
+ rack (2.0.7)
+ rack-test (1.1.0)
+ rack (>= 1.0, < 3)
+ rails (5.2.3)
+ actioncable (= 5.2.3)
+ actionmailer (= 5.2.3)
+ actionpack (= 5.2.3)
+ actionview (= 5.2.3)
+ activejob (= 5.2.3)
+ activemodel (= 5.2.3)
+ activerecord (= 5.2.3)
+ activestorage (= 5.2.3)
+ activesupport (= 5.2.3)
+ bundler (>= 1.3.0)
+ railties (= 5.2.3)
+ sprockets-rails (>= 2.0.0)
+ rails-dom-testing (2.0.3)
+ activesupport (>= 4.2.0)
+ nokogiri (>= 1.6)
+ rails-html-sanitizer (1.3.0)
+ loofah (~> 2.3)
+ railties (5.2.3)
+ actionpack (= 5.2.3)
+ activesupport (= 5.2.3)
+ method_source
+ rake (>= 0.8.7)
+ thor (>= 0.19.0, < 2.0)
+ rake (13.0.0)
+ rb-fsevent (0.10.3)
+ rb-inotify (0.10.0)
+ ffi (~> 1.0)
+ regexp_parser (1.6.0)
+ ruby-debug-ide (0.7.0)
+ rake (>= 0.8.1)
+ ruby-progressbar (1.10.1)
+ ruby_dep (1.5.0)
+ rubyzip (2.0.0)
+ sass (3.7.4)
+ sass-listen (~> 4.0.0)
+ sass-listen (4.0.0)
+ rb-fsevent (~> 0.9, >= 0.9.4)
+ rb-inotify (~> 0.9, >= 0.9.7)
+ sass-rails (5.1.0)
+ railties (>= 5.2.0)
+ sass (~> 3.1)
+ sprockets (>= 2.8, < 4.0)
+ sprockets-rails (>= 2.0, < 4.0)
+ tilt (>= 1.1, < 3)
+ selenium-webdriver (3.142.6)
+ childprocess (>= 0.5, < 4.0)
+ rubyzip (>= 1.2.2)
+ shellany (0.0.1)
+ spring (2.1.0)
+ spring-watcher-listen (2.0.1)
+ listen (>= 2.7, < 4.0)
+ spring (>= 1.2, < 3.0)
+ sprockets (3.7.2)
+ concurrent-ruby (~> 1.0)
+ rack (> 1, < 3)
+ sprockets-rails (3.2.1)
+ actionpack (>= 4.0)
+ activesupport (>= 4.0)
+ sprockets (>= 3.0.0)
+ thor (0.20.3)
+ thread_safe (0.3.6)
+ tilt (2.0.10)
+ turbolinks (5.2.1)
+ turbolinks-source (~> 5.2)
+ turbolinks-source (5.2.0)
+ tzinfo (1.2.5)
+ thread_safe (~> 0.1)
+ uglifier (4.2.0)
+ execjs (>= 0.3.0, < 3)
+ web-console (3.7.0)
+ actionview (>= 5.0)
+ activemodel (>= 5.0)
+ bindex (>= 0.4.0)
+ railties (>= 5.0)
+ websocket-driver (0.7.1)
+ websocket-extensions (>= 0.1.0)
+ websocket-extensions (0.1.4)
+ xpath (3.2.0)
+ nokogiri (~> 1.8)
+
+PLATFORMS
+ ruby
+
+DEPENDENCIES
+ better_errors
+ binding_of_caller
+ bootsnap (>= 1.1.0)
+ bootstrap (~> 4.1.3)
+ byebug
+ capybara (>= 2.15)
+ chromedriver-helper
+ debase (>= 0.2.4.1)
+ guard
+ guard-minitest
+ jbuilder (~> 2.5)
+ jquery-rails
+ jquery-turbolinks
+ listen (>= 3.0.5, < 3.2)
+ minitest-rails
+ minitest-reporters
+ pg (>= 0.18, < 2.0)
+ pry-rails
+ puma (~> 3.11)
+ rails (~> 5.2.3)
+ ruby-debug-ide (>= 0.7.0)
+ sass-rails (~> 5.0)
+ selenium-webdriver
+ spring
+ spring-watcher-listen (~> 2.0.0)
+ turbolinks (~> 5)
+ tzinfo-data
+ uglifier (>= 1.3.0)
+ web-console (>= 3.3.0)
+
+RUBY VERSION
+ ruby 2.5.5p157
+
+BUNDLED WITH
+ 2.0.2
diff --git a/Guardfile b/Guardfile
new file mode 100644
index 0000000000..e34f706f4a
--- /dev/null
+++ b/Guardfile
@@ -0,0 +1,9 @@
+guard :minitest, autorun: false, spring: true do
+ watch(%r{^app/(.+).rb$}) { |m| "test/#{m[1]}_test.rb" }
+ watch(%r{^app/controllers/application_controller.rb$}) { 'test/controllers' }
+ watch(%r{^app/controllers/(.+)_controller.rb$}) { |m| "test/integration/#{m[1]}_test.rb" }
+ watch(%r{^app/views/(.+)_mailer/.+}) { |m| "test/mailers/#{m[1]}_mailer_test.rb" }
+ watch(%r{^lib/(.+).rb$}) { |m| "test/lib/#{m[1]}_test.rb" }
+ watch(%r{^test/.+_test.rb$})
+ watch(%r{^test/test_helper.rb$}) { 'test' }
+end
diff --git a/Rakefile b/Rakefile
new file mode 100644
index 0000000000..e85f913914
--- /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 0000000000..b16e53d6d5
--- /dev/null
+++ b/app/assets/config/manifest.js
@@ -0,0 +1,3 @@
+//= link_tree ../images
+//= link_directory ../javascripts .js
+//= link_directory ../stylesheets .css
diff --git a/app/assets/images/.keep b/app/assets/images/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/app/assets/javascripts/application.js b/app/assets/javascripts/application.js
new file mode 100644
index 0000000000..27c269c46f
--- /dev/null
+++ b/app/assets/javascripts/application.js
@@ -0,0 +1,19 @@
+// This is a manifest file that'll be compiled into application.js, which will include all the files
+// listed below.
+//
+// Any JavaScript/Coffee file within this directory, lib/assets/javascripts, or any plugin's
+// vendor/assets/javascripts directory can be referenced here using a relative path.
+//
+// It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the
+// compiled file. JavaScript code in this file should be added after the last require_* statement.
+//
+// Read Sprockets README (https://github.com/rails/sprockets#sprockets-directives) for details
+// about supported directives.
+ //= require jquery3
+ //= require popper
+ //= require bootstrap-sprockets
+//
+//= require rails-ujs
+//= require activestorage
+//= require turbolinks
+//= require_tree .
diff --git a/app/assets/javascripts/cable.js b/app/assets/javascripts/cable.js
new file mode 100644
index 0000000000..739aa5f022
--- /dev/null
+++ b/app/assets/javascripts/cable.js
@@ -0,0 +1,13 @@
+// Action Cable provides the framework to deal with WebSockets in Rails.
+// You can generate new channels where WebSocket features live using the `rails generate channel` command.
+//
+//= require action_cable
+//= require_self
+//= require_tree ./channels
+
+(function() {
+ this.App || (this.App = {});
+
+ App.cable = ActionCable.createConsumer();
+
+}).call(this);
diff --git a/app/assets/javascripts/channels/.keep b/app/assets/javascripts/channels/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/app/assets/javascripts/homepages.js b/app/assets/javascripts/homepages.js
new file mode 100644
index 0000000000..dee720facd
--- /dev/null
+++ b/app/assets/javascripts/homepages.js
@@ -0,0 +1,2 @@
+// Place all the behaviors and hooks related to the matching controller here.
+// All this logic will automatically be available in application.js.
diff --git a/app/assets/javascripts/users.js b/app/assets/javascripts/users.js
new file mode 100644
index 0000000000..dee720facd
--- /dev/null
+++ b/app/assets/javascripts/users.js
@@ -0,0 +1,2 @@
+// Place all the behaviors and hooks related to the matching controller here.
+// All this logic will automatically be available in application.js.
diff --git a/app/assets/javascripts/votes.js b/app/assets/javascripts/votes.js
new file mode 100644
index 0000000000..dee720facd
--- /dev/null
+++ b/app/assets/javascripts/votes.js
@@ -0,0 +1,2 @@
+// Place all the behaviors and hooks related to the matching controller here.
+// All this logic will automatically be available in application.js.
diff --git a/app/assets/javascripts/works.js b/app/assets/javascripts/works.js
new file mode 100644
index 0000000000..dee720facd
--- /dev/null
+++ b/app/assets/javascripts/works.js
@@ -0,0 +1,2 @@
+// Place all the behaviors and hooks related to the matching controller here.
+// All this logic will automatically be available in application.js.
diff --git a/app/assets/stylesheets/application.scss b/app/assets/stylesheets/application.scss
new file mode 100644
index 0000000000..d8d2fb2bf2
--- /dev/null
+++ b/app/assets/stylesheets/application.scss
@@ -0,0 +1,18 @@
+/*
+ * This is a manifest file that'll be compiled into application.css, which will include all the files
+ * listed below.
+ *
+ * Any CSS and SCSS 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/SCSS
+ * 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.
+ *
+ */
+/* Custom bootstrap variables must be set or imported *before* bootstrap. */
+@import "bootstrap";
+/* Import scss content */
+.text-monospace{font-family:SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace}.text-justify{text-align:justify !important}.text-nowrap{white-space:nowrap !important}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.text-left{text-align:left !important}.text-right{text-align:right !important}.text-center{text-align:center !important}@media (min-width: 576px){.text-sm-left{text-align:left !important}.text-sm-right{text-align:right !important}.text-sm-center{text-align:center !important}}@media (min-width: 768px){.text-md-left{text-align:left !important}.text-md-right{text-align:right !important}.text-md-center{text-align:center !important}}@media (min-width: 992px){.text-lg-left{text-align:left !important}.text-lg-right{text-align:right !important}.text-lg-center{text-align:center !important}}@media (min-width: 1200px){.text-xl-left{text-align:left !important}.text-xl-right{text-align:right !important}.text-xl-center{text-align:center !important}}.text-lowercase{text-transform:lowercase !important}.text-uppercase{text-transform:uppercase !important}.text-capitalize{text-transform:capitalize !important}.font-weight-light{font-weight:300 !important}.font-weight-normal{font-weight:400 !important}.font-weight-bold{font-weight:700 !important}.font-italic{font-style:italic !important}.text-white{color:#fff !important}.text-primary{color:#007bff !important}a.text-primary:hover,a.text-primary:focus{color:#0062cc !important}.text-secondary{color:#6c757d !important}a.text-secondary:hover,a.text-secondary:focus{color:#545b62 !important}.text-success{color:#28a745 !important}a.text-success:hover,a.text-success:focus{color:#1e7e34 !important}.text-info{color:#17a2b8 !important}a.text-info:hover,a.text-info:focus{color:#117a8b !important}.text-warning{color:#ffc107 !important}a.text-warning:hover,a.text-warning:focus{color:#d39e00 !important}.text-danger{color:#dc3545 !important}a.text-danger:hover,a.text-danger:focus{color:#bd2130 !important}.text-light{color:#f8f9fa !important}a.text-light:hover,a.text-light:focus{color:#dae0e5 !important}.text-dark{color:#343a40 !important}a.text-dark:hover,a.text-dark:focus{color:#1d2124 !important}.text-body{color:#212529 !important}.text-muted{color:#6c757d !important}.text-black-50{color:rgba(0,0,0,0.5) !important}.text-white-50{color:rgba(255,255,255,0.5) !important}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.visible{visibility:visible !important}.invisible{visibility:hidden !important}@media print{*,*::before,*::after{text-shadow:none !important;box-shadow:none !important}a:not(.btn){text-decoration:underline}abbr[title]::after{content:" (" attr(title) ")"}pre{white-space:pre-wrap !important}pre,blockquote{border:1px solid #adb5bd;page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}p,h2,h3{orphans:3;widows:3}h2,h3{page-break-after:avoid}@page{size:a3}body{min-width:992px !important}.container{min-width:992px !important}.navbar{display:none}.badge{border:1px solid #000}.table{border-collapse:collapse !important}.table td,.table th{background-color:#fff !important}.table-bordered th,.table-bordered td{border:1px solid #dee2e6 !important}.table-dark{color:inherit}.table-dark th,.table-dark td,.table-dark thead th,.table-dark tbody+tbody{border-color:#dee2e6}.table .thead-dark th{color:inherit;border-color:#dee2e6}}.user-votes__container{margin-top:2rem}.user-votes__header{color:black}.works-votes__container{margin-top:2rem}.works-votes__header{color:black}body,h1,h2,h3,h4,h5{font-family:"Gudea", sans-serif}h1,h2,h3,h4,h5{font-weight:bold}a,h2,h3{color:#26A69A}a:hover{color:#00796B;text-decoration:none}.alert{margin:0}.btn-primary{background-color:#26A69A;color:white;border-color:#26A69A}.btn:hover{background-color:#00796B;color:white}.table td{vertical-align:middle}.app-header__nav{display:flex;flex-direction:row;justify-content:space-between}.app-header__nav_item{margin-top:1rem}.app-header__nav_item .nav-link{color:#00796B}.app-header__site-nav-container .app-header__nav_item{margin-right:2rem}.app-header__user-nav-container .nav-item{margin-left:2rem}.list-group-item{border:none}.app-header__header{max-width:100%;background-color:#B2DFDB;margin-bottom:0.5rem;padding:2rem 1rem 0.5rem 1rem}.app-header__header h1{text-align:center;margin:25px auto 40px auto}.app-header__header h1 a{color:#FF5722;padding-right:25px;margin-right:15px;border-right:white 2px solid}.app-header__header h1 small{color:white}.app-header__header p{margin-bottom:5px}.alert__container{margin:2rem 0 1rem 0}.spotlight{padding:0 2rem 2rem 2rem}.spotlight__header{color:#26A69A}.spotlight__header--prefix{color:#424242}.spotlight__header--prefix::after{content:":"}.spotlight__link-to{border-bottom:2px solid}.spotlight__description{margin-bottom:0}.root__hr{margin:0px auto 3rem auto;color:#EEEEEE}.top-ten__container{display:grid;grid-template-columns:1fr 1fr 1fr;grid-column-gap:2em}.top-ten__header{color:#26A69A;border-bottom:2px solid;border-bottom-color:#B2DFDB}.top-ten__list{list-style-type:none;margin-left:0;padding-left:0}.top-ten__creator{color:slategrey}main{padding:2rem}
+@import "**/*";
diff --git a/app/assets/stylesheets/homepages.scss b/app/assets/stylesheets/homepages.scss
new file mode 100644
index 0000000000..2305c36d10
--- /dev/null
+++ b/app/assets/stylesheets/homepages.scss
@@ -0,0 +1,3 @@
+// Place all the styles related to the Homepages controller here.
+// They will automatically be included in application.css.
+// You can use Sass (SCSS) here: http://sass-lang.com/
diff --git a/app/assets/stylesheets/users.scss b/app/assets/stylesheets/users.scss
new file mode 100644
index 0000000000..31a2eacb84
--- /dev/null
+++ b/app/assets/stylesheets/users.scss
@@ -0,0 +1,3 @@
+// Place all the styles related to the Users controller here.
+// They will automatically be included in application.css.
+// You can use Sass (SCSS) here: http://sass-lang.com/
diff --git a/app/assets/stylesheets/votes.scss b/app/assets/stylesheets/votes.scss
new file mode 100644
index 0000000000..9a6720f80e
--- /dev/null
+++ b/app/assets/stylesheets/votes.scss
@@ -0,0 +1,3 @@
+// Place all the styles related to the Votes controller here.
+// They will automatically be included in application.css.
+// You can use Sass (SCSS) here: http://sass-lang.com/
diff --git a/app/assets/stylesheets/works.scss b/app/assets/stylesheets/works.scss
new file mode 100644
index 0000000000..5618452f3e
--- /dev/null
+++ b/app/assets/stylesheets/works.scss
@@ -0,0 +1,3 @@
+// Place all the styles related to the Works controller here.
+// They will automatically be included in application.css.
+// You can use Sass (SCSS) here: http://sass-lang.com/
diff --git a/app/channels/application_cable/channel.rb b/app/channels/application_cable/channel.rb
new file mode 100644
index 0000000000..d672697283
--- /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 0000000000..0ff5442f47
--- /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/application_controller.rb b/app/controllers/application_controller.rb
new file mode 100644
index 0000000000..09705d12ab
--- /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 0000000000..e69de29bb2
diff --git a/app/controllers/homepages_controller.rb b/app/controllers/homepages_controller.rb
new file mode 100644
index 0000000000..ef28e0d5bd
--- /dev/null
+++ b/app/controllers/homepages_controller.rb
@@ -0,0 +1,8 @@
+class HomepagesController < ApplicationController
+ def index
+ @works = Work.all
+ @movies = Work.category_list("movie").slice(0..9)
+ @books = Work.category_list("book").slice(0..9)
+ @albums = Work.category_list("album").slice(0..9)
+ end
+end
diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb
new file mode 100644
index 0000000000..d4377d7db0
--- /dev/null
+++ b/app/controllers/users_controller.rb
@@ -0,0 +1,47 @@
+class UsersController < ApplicationController
+ def index
+ @users = User.all
+ end
+
+ def show
+ @user = User.find_by(id: params[:id])
+ if @user.nil?
+ head :not_found
+ return
+ end
+ end
+
+ def login_form
+ @user = User.new
+ end
+
+ def login
+ username = params[:user][:username]
+ found_user = User.find_by(username: username)
+
+ if found_user
+ session[:user_id] = found_user.id
+ flash[:success] = "Successfully logged in as existing user #{found_user.username}"
+ else
+ @user = User.new(username: username)
+ if @user.save
+ session[:user_id] = @user.id
+ flash[:success] = "Successfully created new user #{@user.username} with ID #{@user.id}"
+ else
+ flash.now[:warning] = "A problem occurred: Could not create user"
+ render :login_form
+ return
+ end
+ end
+
+ return redirect_to root_path
+
+ end
+
+ def logout
+ session[:user_id] = nil
+ flash[:success] = "Successfully logged out"
+ return redirect_to root_path
+ end
+
+end
diff --git a/app/controllers/votes_controller.rb b/app/controllers/votes_controller.rb
new file mode 100644
index 0000000000..5dcd5e4df5
--- /dev/null
+++ b/app/controllers/votes_controller.rb
@@ -0,0 +1,28 @@
+class VotesController < ApplicationController
+
+ def upvote
+ if session[:user_id]
+ @vote = Vote.find_by(work_id: params[:id], user_id: session[:user_id])
+
+ if @vote
+ flash[:warning] = "A problem occurred: Could not upvote twice!"
+ else
+ @vote = Vote.new(work_id: params[:id], user_id: session[:user_id])
+
+ if @vote.save
+ flash[:success] = "Successfully upvoted!"
+ else
+ flash[:warning] = "A problem occurred: Could not upvote!"
+ end
+
+ end
+
+ else
+ flash[:warning] = "A problem occurred: You must log in to do that!"
+ end
+
+ return redirect_to work_path(params[:id])
+
+ end
+
+end
diff --git a/app/controllers/works_controller.rb b/app/controllers/works_controller.rb
new file mode 100644
index 0000000000..23d055b34c
--- /dev/null
+++ b/app/controllers/works_controller.rb
@@ -0,0 +1,76 @@
+class WorksController < ApplicationController
+
+ before_action :find_work, only: [:show, :edit, :update]
+
+ def index
+ @works = Work.all
+ @movies = Work.category_list("movie")
+ @books = Work.category_list("book")
+ @albums = Work.category_list("album")
+ end
+
+ def show
+ if @work.nil?
+ head :not_found
+ return
+ end
+ end
+
+ def new
+ @work = Work.new
+ end
+
+ def create
+ @work = Work.new( work_params )
+ if @work.save
+ flash[:success] = "Successfully created #{@work.category} #{@work.id}"
+ redirect_to work_path(@work.id)
+ return
+ else
+ flash.now[:warning] = "A problem occurred: Could not create #{@work.category}"
+ render :new
+ return
+ end
+ end
+
+ def edit
+
+ end
+
+ def update
+ if @work.nil?
+ redirect_to root_path
+ return
+ elsif @work.update( work_params )
+ redirect_to work_path(@work.id)
+ return
+ else
+ render :edit
+ return
+ end
+ end
+
+ def destroy
+ the_correct_work = Work.find_by( id: params[:id] )
+ if the_correct_work.nil?
+ redirect_to root_path
+ return
+ else
+ the_correct_work.destroy
+ flash[:success] = "Your work " + the_correct_work.title + " was successfully deleted!"
+ redirect_to root_path
+ return
+ end
+ end
+
+ private
+
+ def find_work
+ @work = Work.find_by(id: params[:id])
+ end
+
+ def work_params
+ return params.require(:work).permit(:category, :title, :creator, :publication_year, :description)
+ end
+
+end
diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb
new file mode 100644
index 0000000000..02d03d926e
--- /dev/null
+++ b/app/helpers/application_helper.rb
@@ -0,0 +1,12 @@
+module ApplicationHelper
+ def readable_date(date)
+ return "[unknown]" unless date
+ return (
+ "".html_safe +
+ date.to_date.to_s +
+ " ".html_safe
+ )
+ end
+end
diff --git a/app/helpers/homepages_helper.rb b/app/helpers/homepages_helper.rb
new file mode 100644
index 0000000000..4bd8098f37
--- /dev/null
+++ b/app/helpers/homepages_helper.rb
@@ -0,0 +1,2 @@
+module HomepagesHelper
+end
diff --git a/app/helpers/users_helper.rb b/app/helpers/users_helper.rb
new file mode 100644
index 0000000000..2310a240d7
--- /dev/null
+++ b/app/helpers/users_helper.rb
@@ -0,0 +1,2 @@
+module UsersHelper
+end
diff --git a/app/helpers/votes_helper.rb b/app/helpers/votes_helper.rb
new file mode 100644
index 0000000000..5a82eed07d
--- /dev/null
+++ b/app/helpers/votes_helper.rb
@@ -0,0 +1,2 @@
+module VotesHelper
+end
diff --git a/app/helpers/works_helper.rb b/app/helpers/works_helper.rb
new file mode 100644
index 0000000000..ccb78c2b73
--- /dev/null
+++ b/app/helpers/works_helper.rb
@@ -0,0 +1,2 @@
+module WorksHelper
+end
diff --git a/app/jobs/application_job.rb b/app/jobs/application_job.rb
new file mode 100644
index 0000000000..a009ace51c
--- /dev/null
+++ b/app/jobs/application_job.rb
@@ -0,0 +1,2 @@
+class ApplicationJob < ActiveJob::Base
+end
diff --git a/app/mailers/application_mailer.rb b/app/mailers/application_mailer.rb
new file mode 100644
index 0000000000..286b2239d1
--- /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 0000000000..10a4cba84d
--- /dev/null
+++ b/app/models/application_record.rb
@@ -0,0 +1,3 @@
+class ApplicationRecord < ActiveRecord::Base
+ self.abstract_class = true
+end
diff --git a/app/models/concerns/.keep b/app/models/concerns/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/app/models/user.rb b/app/models/user.rb
new file mode 100644
index 0000000000..ecb3e0cf48
--- /dev/null
+++ b/app/models/user.rb
@@ -0,0 +1,4 @@
+class User < ApplicationRecord
+ has_many :votes, :dependent => :restrict_with_error
+ validates :username, presence: true
+end
diff --git a/app/models/vote.rb b/app/models/vote.rb
new file mode 100644
index 0000000000..8c25a4a45e
--- /dev/null
+++ b/app/models/vote.rb
@@ -0,0 +1,7 @@
+class Vote < ApplicationRecord
+ belongs_to :work
+ belongs_to :user
+
+ validates_uniqueness_of :user, scope: [:work_id], :message => "has already voted for this work"
+
+end
diff --git a/app/models/work.rb b/app/models/work.rb
new file mode 100644
index 0000000000..3188785eae
--- /dev/null
+++ b/app/models/work.rb
@@ -0,0 +1,16 @@
+class Work < ApplicationRecord
+ has_many :votes, :dependent => :destroy
+ validates :title, presence: true
+ validates :category, presence: true
+
+ def self.category_list(category)
+ list = Work.where(category: category).sort_by{ |work| - work.votes.length }
+ return list
+ end
+
+ def self.spotlight
+ top = Work.all.sort_by {|work| - work.votes.length }.first
+ return top
+ end
+
+end
diff --git a/app/views/homepages/index.html.erb b/app/views/homepages/index.html.erb
new file mode 100644
index 0000000000..9b761d9d2e
--- /dev/null
+++ b/app/views/homepages/index.html.erb
@@ -0,0 +1,79 @@
+
+
+
+
+ <%= Work.spotlight.votes.length %> votes - <%= Work.spotlight.description %>
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/views/layouts/application.html.erb b/app/views/layouts/application.html.erb
new file mode 100644
index 0000000000..54677b5e0b
--- /dev/null
+++ b/app/views/layouts/application.html.erb
@@ -0,0 +1,92 @@
+
+
+
+ MediaRanker
+ <%= csrf_meta_tags %>
+ <%= csp_meta_tag %>
+
+ <%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %>
+ <%= javascript_include_tag 'application', 'data-turbolinks-track': 'reload' %>
+
+
+
+
+
+
+
+ <% if flash.any? %>
+
+
+ <% flash.each do |name, message| %>
+
+
+
<%= message %>
+
+ <% if @work && @work.errors.any? %>
+
+ <% @work.errors.each do |column, message| %>
+ <%= column %>: <%= message %>
+ <% end %>
+
+
+ <% elsif @user && @user.errors.any? %>
+
+ <% @user.errors.each do |column, message| %>
+ <%= column %>: <%= message %>
+ <% end %>
+
+
+ <% end %>
+
+
+ <% end %>
+
+
+ <% end %>
+
+ <%= yield %>
+
+
diff --git a/app/views/layouts/mailer.html.erb b/app/views/layouts/mailer.html.erb
new file mode 100644
index 0000000000..cbd34d2e9d
--- /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 0000000000..37f0bddbd7
--- /dev/null
+++ b/app/views/layouts/mailer.text.erb
@@ -0,0 +1 @@
+<%= yield %>
diff --git a/app/views/users/current.html.erb b/app/views/users/current.html.erb
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/app/views/users/index.html.erb b/app/views/users/index.html.erb
new file mode 100644
index 0000000000..f4606c41cc
--- /dev/null
+++ b/app/views/users/index.html.erb
@@ -0,0 +1,24 @@
+
+ List of Users
+
+
+
+ Username
+ Votes
+ Joined
+
+
+
+ <% @users.each do |user| %>
+
+ <%= link_to user.username, user_path(user.id) %>
+ <%= user.votes.length %>
+ <%= readable_date(user.created_at) %>
+
+ <% end %>
+
+
+
+ <%= link_to "Back to Media List", works_path, class: "btn btn-secondary" %>
+
+
\ No newline at end of file
diff --git a/app/views/users/login_form.html.erb b/app/views/users/login_form.html.erb
new file mode 100644
index 0000000000..5107c305dc
--- /dev/null
+++ b/app/views/users/login_form.html.erb
@@ -0,0 +1,27 @@
+
+ Log In
+
+ <%= form_with model: @user, url: login_path do |f| %>
+
+
+ <%= f.label :Username %>
+ <%= f.text_field :username, class: "form-control" %>
+
+
+
+ <%= f.submit "Log In", class: "btn btn-primary" %>
+
+
+ <% end %>
+
+
+ A note about logging in
+
+ There is no password field. In fact, there is no indication whatsoever that you are who you say you are. There's nothing special about users - username is just another piece of data that the user entered and we have to keep track of.
+
+
+
+ We'll learn more about security and authentication in the next couple weeks. For now, don't worry about it beyond what you can see here.
+
+
+
\ No newline at end of file
diff --git a/app/views/users/show.html.erb b/app/views/users/show.html.erb
new file mode 100644
index 0000000000..77d82e09e6
--- /dev/null
+++ b/app/views/users/show.html.erb
@@ -0,0 +1,34 @@
+
+ User Summary: <%= link_to @user.username, user_path(@user.id) %>
+ Joined site <%= @user.created_at %>
+
+
+
+
+
+
+ Media Title
+ Created By
+ Published
+ Category
+ Voted On
+
+
+
+ <% @user.votes.each do |vote| %>
+
+ <%= link_to Work.find_by(id: vote.work_id).title, work_path(vote.work_id) %>
+ <%= Work.find_by(id: vote.work_id).creator %>
+ <%= Work.find_by(id: vote.work_id).publication_year %>
+ <%= Work.find_by(id: vote.work_id).category %>
+ <%= Work.find_by(id: vote.work_id).created_at %>
+
+ <% end %>
+
+
+
+
+ <%= link_to "See all Users", users_path, class: "btn btn-secondary" %>
+ <%= link_to "Back to Media List", root_path, class: "btn btn-primary" %>
+
+
\ No newline at end of file
diff --git a/app/views/works/_form.html.erb b/app/views/works/_form.html.erb
new file mode 100644
index 0000000000..82f0c67891
--- /dev/null
+++ b/app/views/works/_form.html.erb
@@ -0,0 +1,31 @@
+<%= form_with model: @work, class: "work-form-class" do |f| %>
+
+
+ <%= f.label :Category %>
+ <%= f.select :category, ["album", "book", "movie"], {:include_blank => false},{:class =>"form-control"} %>
+
+
+
+ <%= f.label :Title %>
+ <%= f.text_field :title, class: "form-control" %>
+
+
+
+ <%= f.label :Creator %>
+ <%= f.text_field :creator, class: "form-control" %>
+
+
+
+ <%= f.label "Publication year" %>
+ <%= f.text_field :publication_year, class: "form-control" %>
+
+
+
+ <%= f.label :Description %>
+ <%= f.text_field :description, class: "form-control" %>
+
+
+
+ <%= f.submit submit_button_label, class: "btn btn-primary" %>
+
+<% end %>
\ No newline at end of file
diff --git a/app/views/works/edit.html.erb b/app/views/works/edit.html.erb
new file mode 100644
index 0000000000..09a63d5715
--- /dev/null
+++ b/app/views/works/edit.html.erb
@@ -0,0 +1,6 @@
+
+ Edit This <%= @work.category %>
+
+ <%= render partial: "form", locals: { submit_button_label: "Update Work", work_form_class: "edit-work-form" } %>
+
+
\ No newline at end of file
diff --git a/app/views/works/index.html.erb b/app/views/works/index.html.erb
new file mode 100644
index 0000000000..7a12a4e2b3
--- /dev/null
+++ b/app/views/works/index.html.erb
@@ -0,0 +1,84 @@
+
+ List of Works
+
+
+
+
+
+ <%= link_to "View top media", root_path, class: "btn btn-secondary" %>
+ <%= link_to "Add a new work", root_path, class: "btn btn-primary" %>
+
+
+
diff --git a/app/views/works/new.html.erb b/app/views/works/new.html.erb
new file mode 100644
index 0000000000..763f2d54bd
--- /dev/null
+++ b/app/views/works/new.html.erb
@@ -0,0 +1,7 @@
+
+ Add a new work
+
+ <%= render partial: "form", locals: { submit_button_label: "Create Work", work_form_class: "new-work-form" } %>
+
+
+
diff --git a/app/views/works/show.html.erb b/app/views/works/show.html.erb
new file mode 100644
index 0000000000..6c58db9c21
--- /dev/null
+++ b/app/views/works/show.html.erb
@@ -0,0 +1,33 @@
+
+
+ <%= @work.title %>
+ Created by:
+ Published:
+
+ <%= link_to "Back to media ranks", root_path, class: "btn btn-primary" %>
+ <%= link_to "Edit", edit_work_path(@work.id), class: "btn btn-primary" %>
+ <%= link_to "Upvote", upvote_path(@work.id), method: :post, class: "btn btn-primary", rel: "nofollow" %>
+ <%= link_to "Delete", work_path(@work.id), method: :delete, data: { confirm: "Are you sure?" }, class: "btn btn-danger", rel: "nofollow" %>
+
+
+
+
+
+
+
+
+ User
+ Date
+
+
+
+ <% @work.votes.each do |vote| %>
+
+ <%= link_to vote.user.username, user_path(vote.user_id) %>
+ <%= readable_date(vote.created_at) %>
+
+ <% end %>
+
+
+
+
\ No newline at end of file
diff --git a/bin/bundle b/bin/bundle
new file mode 100755
index 0000000000..f19acf5b5c
--- /dev/null
+++ b/bin/bundle
@@ -0,0 +1,3 @@
+#!/usr/bin/env ruby
+ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../Gemfile', __dir__)
+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..94fd4d7977
--- /dev/null
+++ b/bin/setup
@@ -0,0 +1,36 @@
+#!/usr/bin/env ruby
+require 'fileutils'
+include FileUtils
+
+# path to your application root.
+APP_ROOT = File.expand_path('..', __dir__)
+
+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..d89ee495fa
--- /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..58bfaed518
--- /dev/null
+++ b/bin/update
@@ -0,0 +1,31 @@
+#!/usr/bin/env ruby
+require 'fileutils'
+include FileUtils
+
+# path to your application root.
+APP_ROOT = File.expand_path('..', __dir__)
+
+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')
+
+ # Install JavaScript dependencies if using Yarn
+ # system('bin/yarn')
+
+ 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..460dd565b4
--- /dev/null
+++ b/bin/yarn
@@ -0,0 +1,11 @@
+#!/usr/bin/env ruby
+APP_ROOT = File.expand_path('..', __dir__)
+Dir.chdir(APP_ROOT) do
+ begin
+ exec "yarnpkg", *ARGV
+ 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..08db79cb25
--- /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 MediaRanker
+ 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.2
+
+ # Settings in config/environments/* take precedence over those specified here.
+ # Application configuration can go into files in config/initializers
+ # -- all .rb files in that directory are automatically loaded after loading
+ # the framework and any gems in your application.
+ end
+end
diff --git a/config/boot.rb b/config/boot.rb
new file mode 100644
index 0000000000..b9e460cef3
--- /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 0000000000..e6b43ef488
--- /dev/null
+++ b/config/cable.yml
@@ -0,0 +1,10 @@
+development:
+ adapter: async
+
+test:
+ adapter: async
+
+production:
+ adapter: redis
+ url: <%= ENV.fetch("REDIS_URL") { "redis://localhost:6379/1" } %>
+ channel_prefix: media-ranker_production
diff --git a/config/credentials.yml.enc b/config/credentials.yml.enc
new file mode 100644
index 0000000000..3172dab235
--- /dev/null
+++ b/config/credentials.yml.enc
@@ -0,0 +1 @@
+bZljqVPEIlbSqLWIHjTKRQ8tF9UVd/FjJFrkAfX3TLuyy2ghauMmwaPkyEiEyP6whprV4BjSRTqjYFVucamHZ9c40/zeCK1v7uONn188akDWhgJUajhxpAi3dsREvbDAEN60LO09qoiNVQ9K1tCCC3WD+k1vP5y4smcm1YrRQzb3vegntQbRV9Fn+Sm77cBzYakur4ry1TxtD3fcrlRy+PyAJSDb1qbs71/y4Uw/5PPPbCwpFI1ucuQI19u06Xt0lWUKPis+KS38sKzsMsIWwgB/UxaJDMXwRtgzwSr30cqsy/MdaWZKP/0UA3PmI9+dX9Y4L5UKvbM0YuJ/YawVnjEwOObDIuYkgTJIUqwhnuRcIkXmsom4rpSqiv2BRqRWYV/kZRXS4akdVp8D11hALbpPUKzscDzTa/VX--BuaJxXoA2GuvOVbx--8bkNLUhNu06XvFyzAWtWkA==
\ No newline at end of file
diff --git a/config/database.yml b/config/database.yml
new file mode 100644
index 0000000000..d9428f62bd
--- /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: media-ranker_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: media-ranker
+
+ # 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: media-ranker_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: media-ranker_production
+ username: media-ranker
+ password: <%= ENV['MEDIA-RANKER_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..1311e3e4ef
--- /dev/null
+++ b/config/environments/development.rb
@@ -0,0 +1,61 @@
+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.
+ # Run rails dev:cache to toggle caching.
+ 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.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 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
+
+ # 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..3c2fde1361
--- /dev/null
+++ b/config/environments/production.rb
@@ -0,0 +1,94 @@
+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 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
+
+ # 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
+
+ # 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 = "media-ranker_#{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..0a38fd3ce9
--- /dev/null
+++ b/config/environments/test.rb
@@ -0,0 +1,46 @@
+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.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
+
+ # 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
+
+ # Raises error for missing translations
+ # config.action_view.raise_on_missing_translations = true
+end
diff --git a/config/initializers/action_view.rb b/config/initializers/action_view.rb
new file mode 100644
index 0000000000..142d382f87
--- /dev/null
+++ b/config/initializers/action_view.rb
@@ -0,0 +1 @@
+Rails.application.config.action_view.form_with_generates_remote_forms = false
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/content_security_policy.rb b/config/initializers/content_security_policy.rb
new file mode 100644
index 0000000000..d3bcaa5ec8
--- /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
+# For further information see the following documentation
+# https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Security-Policy
+
+# Rails.application.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
+
+# If you are using UJS then enable automatic nonce generation
+# Rails.application.config.content_security_policy_nonce_generator = -> request { SecureRandom.base64(16) }
+
+# Report CSP violations to a specified URI
+# For further information see the following documentation:
+# https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Security-Policy-Report-Only
+# Rails.application.config.content_security_policy_report_only = true
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/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..a5eccf816b
--- /dev/null
+++ b/config/puma.rb
@@ -0,0 +1,34 @@
+# 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.
+#
+# preload_app!
+
+# 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..27904ba8ae
--- /dev/null
+++ b/config/routes.rb
@@ -0,0 +1,13 @@
+Rails.application.routes.draw do
+ # For details on the DSL available within this file, see http://guides.rubyonrails.org/routing.html
+ root to: "homepages#index"
+ resources :homepages, only: [:index]
+ resources :works
+ resources :users
+
+ get "/login", to: "users#login_form", as: "login"
+ post "/login", to: "users#login"
+ post "/logout", to: "users#logout", as: "logout"
+
+ post "/works/:id/upvote", to: "votes#upvote", as: "upvote"
+end
diff --git a/config/spring.rb b/config/spring.rb
new file mode 100644
index 0000000000..9fa7863f99
--- /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/config/storage.yml b/config/storage.yml
new file mode 100644
index 0000000000..d32f76e8fb
--- /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 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
+
+# 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
+
+# Use 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
+
+# mirror:
+# service: Mirror
+# primary: local
+# mirrors: [ amazon, google, microsoft ]
diff --git a/db/generate_starter_data.rb b/db/generate_starter_data.rb
index 19fca77936..d3a23c6bcd 100644
--- a/db/generate_starter_data.rb
+++ b/db/generate_starter_data.rb
@@ -13,7 +13,7 @@
CSV.open("db/media_seeds.csv", "w", :write_headers => true,
:headers => ["category", "title", "creator", "publication_year", "description"]) do |csv|
25.times do
- category = %w(album book).sample
+ category = %w(album book movie).sample
title = Faker::Coffee.blend_name
creator = Faker::Name.name
publication_year = rand(Date.today.year - 100..Date.today.year)
diff --git a/db/media-seeds.csv b/db/media-seeds.csv
deleted file mode 100644
index 5f5b252a25..0000000000
--- a/db/media-seeds.csv
+++ /dev/null
@@ -1,25 +0,0 @@
-category,title,creator,publication_year,description
-album,Blue Breaker,Dr. Sarai Langosh,1949,Et et expedita non aut quo.
-book,Joe Treat,Blaise Lesch,1968,Voluptatem adipisci qui velit.
-album,Kreb-Full-o Been,Ms. Trevion Buckridge,2016,Vero consectetur delectus consequatur id aut accusantium unde excepturi.
-album,Wake-up Pie,Timmy Streich I,1919,Voluptatem consequatur qui consectetur nisi officiis culpa.
-album,Major Cup,Jayde Bartoletti,1944,Quis recusandae cum est facere consequatur minima magni et.
-book,Summer Select,Ms. Gwendolyn Ortiz,1946,Et molestiae eos nam odit aut sed.
-album,Holiday Choice,Alexandria Lehner,1940,Excepturi voluptas ut voluptatum.
-book,Postmodern Blend,Meredith Brekke,1970,Dolorem fugit accusantium qui.
-book,Green Forrester,Raquel Hirthe,1933,Omnis qui quia odio.
-album,Winter Mug,Tia Weissnat II,1990,Laboriosam autem iusto quae sed voluptate et.
-book,Red Pie,Davon Kub,1961,Id dolorem qui laborum quia.
-album,Major Equinox,Queen Satterfield,1997,Fugit perferendis est quam sunt porro vel rerum.
-book,Melty Breaker,Montana Dickinson Sr.,1991,Perferendis harum fuga corporis.
-book,Winter Pie,Mr. Syble Kuhn,1970,Incidunt molestias deserunt laudantium.
-album,Goodbye Utopia,Orion Spencer,1962,Praesentium enim pariatur voluptatem sed quod dolorum.
-album,Green Select,Berneice Jenkins,1957,Hic repudiandae molestiae id nulla aliquid maiores necessitatibus.
-book,Blacktop Enlightenment,Seamus D'Amore,1928,Ea id cumque et pariatur magni nemo dolorem.
-album,Express Extract,Dorothy Jast I,1969,Dolores dolorum aut ea aperiam et voluptatem.
-album,Winter Been,Mackenzie Wilkinson,1932,Culpa repudiandae et at sint et amet fugiat et.
-book,Heart Mug,Orpha Douglas,2009,Qui voluptas alias quia.
-album,Blue Treat,Eliseo Gorczany,1979,Sit est quis veniam saepe.
-book,Hello Town,Laury Walter,2005,Est sed ut asperiores sed fugiat.
-album,Blacktop Choice,Casey Feil,2008,Temporibus ex maxime labore quam et natus quia ipsum.
-book,Huggy Star,Nigel Lesch DVM,1962,Voluptatem ea aspernatur nesciunt ipsa quis error corporis placeat.
diff --git a/db/media_seeds.csv b/db/media_seeds.csv
new file mode 100644
index 0000000000..989a4f3fe0
--- /dev/null
+++ b/db/media_seeds.csv
@@ -0,0 +1,26 @@
+category,title,creator,publication_year,description
+book,Holiday Solstice,Matilde Hoeger,1953,Cupiditate ut et nisi.
+album,Goodbye Select,Cody Lesch,1953,Animi modi voluptatibus beatae.
+book,Green Java,Kyong Rohan,2015,Saepe dignissimos et aut.
+book,Pumpkin-spice Utopia,Jani Satterfield Jr.,1952,Dolorum rerum quia nisi.
+movie,The Mug,Ja Grimes Jr.,2010,Mollitia iste quia optio.
+album,Summer Cowboy,Earnest Donnelly,2017,Tempora exercitationem earum doloremque.
+book,Holiday Look,Ernesto Leuschke,1957,Voluptatem fugit labore eaque.
+album,Green Utopia,Jermaine Turner,2016,Placeat mollitia ullam voluptas.
+book,Chocolate Star,Donita Nitzsche,1964,Odit iusto laboriosam fuga.
+book,Joe Nuts,Tonia DuBuque,1988,Qui repudiandae quae atque.
+book,Summer Breaker,Marcia Stark,1955,In consequatur vel iure.
+movie,Café Cake,Faviola King,2005,Placeat repellendus voluptatem voluptates.
+book,Cascara Look,Mr. Thurman Hartmann,1934,Doloribus reiciendis ea molestias.
+movie,Blacktop Forrester,Adan Rau,1920,Aut odio iste est.
+movie,Huggy Level,Concha Larkin DDS,1995,Et nam ut voluptatibus.
+book,Jacked Look,Fawn Erdman IV,1971,Id velit aut non.
+movie,Postmodern Bean,Jenell Goodwin,1997,Voluptas odio beatae repellendus.
+movie,Wake-up Symphony,Evelyn Koelpin,1948,Ut est dolorem officiis.
+book,Postmodern Cup,Jacklyn Homenick,1977,Voluptates quae porro at.
+movie,Seattle Pie,Tina Cronin,1988,Alias voluptatum laboriosam mollitia.
+movie,Winter Equinox,Bryant Lang III,1961,Provident aperiam quia dolorum.
+book,Blacktop Symphony,Mr. Aron O'Conner,2011,Possimus id est aut.
+movie,Holiday Solstice,Nathaniel Kuhn PhD,1972,Quia aut provident error.
+movie,Express Pie,Barbie Greenfelder,1970,Ipsa et non cum.
+album,Hello Treat,Jacquiline Reilly,1953,Accusamus id ut ut.
diff --git a/db/migrate/20191015225109_create_works.rb b/db/migrate/20191015225109_create_works.rb
new file mode 100644
index 0000000000..d6e24fdb58
--- /dev/null
+++ b/db/migrate/20191015225109_create_works.rb
@@ -0,0 +1,13 @@
+class CreateWorks < ActiveRecord::Migration[5.2]
+ def change
+ create_table :works do |t|
+ t.string :category
+ t.string :title
+ t.string :creator
+ t.integer :publication_year
+ t.string :description
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20191015225140_create_users.rb b/db/migrate/20191015225140_create_users.rb
new file mode 100644
index 0000000000..9c711e759b
--- /dev/null
+++ b/db/migrate/20191015225140_create_users.rb
@@ -0,0 +1,9 @@
+class CreateUsers < ActiveRecord::Migration[5.2]
+ def change
+ create_table :users do |t|
+ t.string :username
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20191015225155_create_votes.rb b/db/migrate/20191015225155_create_votes.rb
new file mode 100644
index 0000000000..aa134a1296
--- /dev/null
+++ b/db/migrate/20191015225155_create_votes.rb
@@ -0,0 +1,8 @@
+class CreateVotes < ActiveRecord::Migration[5.2]
+ def change
+ create_table :votes do |t|
+
+ t.timestamps
+ end
+ end
+end
diff --git a/db/migrate/20191015230825_add_reference_to_votes.rb b/db/migrate/20191015230825_add_reference_to_votes.rb
new file mode 100644
index 0000000000..40a9c8c89e
--- /dev/null
+++ b/db/migrate/20191015230825_add_reference_to_votes.rb
@@ -0,0 +1,6 @@
+class AddReferenceToVotes < ActiveRecord::Migration[5.2]
+ def change
+ add_reference :votes, :work, foreign_key: true
+ add_reference :votes, :user, foreign_key: true
+ end
+end
diff --git a/db/schema.rb b/db/schema.rb
new file mode 100644
index 0000000000..360935f625
--- /dev/null
+++ b/db/schema.rb
@@ -0,0 +1,45 @@
+# 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: 2019_10_15_230825) do
+
+ # These are extensions that must be enabled in order to support this database
+ enable_extension "plpgsql"
+
+ create_table "users", force: :cascade do |t|
+ t.string "username"
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ end
+
+ create_table "votes", force: :cascade do |t|
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ t.bigint "work_id"
+ t.bigint "user_id"
+ t.index ["user_id"], name: "index_votes_on_user_id"
+ t.index ["work_id"], name: "index_votes_on_work_id"
+ end
+
+ create_table "works", force: :cascade do |t|
+ t.string "category"
+ t.string "title"
+ t.string "creator"
+ t.integer "publication_year"
+ t.string "description"
+ t.datetime "created_at", null: false
+ t.datetime "updated_at", null: false
+ end
+
+ add_foreign_key "votes", "users"
+ add_foreign_key "votes", "works"
+end
diff --git a/db/seeds.rb b/db/seeds.rb
new file mode 100644
index 0000000000..c5e8fa2019
--- /dev/null
+++ b/db/seeds.rb
@@ -0,0 +1,45 @@
+# 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 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)
+
+require 'csv'
+
+MEDIA_FILE = Rails.root.join('db', 'media_seeds.csv')
+puts "Loading raw work data from #{MEDIA_FILE}"
+
+work_failures = []
+CSV.foreach(MEDIA_FILE, :headers => true) do |row|
+ work = Work.new
+ work.category = row['category']
+ work.title = row['title']
+ work.creator = row['creator']
+ work.publication_year = row['publication_year'].to_i
+ work.description = row['description']
+
+ successful = work.save
+ if !successful
+ work_failures << work
+ puts "Failed to save work: #{work.inspect}"
+ else
+ puts "Created work: #{work.inspect}"
+ end
+end
+
+puts "Added #{Work.count} work records"
+puts "#{work_failures.length} works 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..1deff4255b
--- /dev/null
+++ b/package.json
@@ -0,0 +1,5 @@
+{
+ "name": "media-ranker",
+ "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/storage/.keep b/storage/.keep
new file mode 100644
index 0000000000..e69de29bb2
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/homepages_controller_test.rb b/test/controllers/homepages_controller_test.rb
new file mode 100644
index 0000000000..cef86edeae
--- /dev/null
+++ b/test/controllers/homepages_controller_test.rb
@@ -0,0 +1,7 @@
+require "test_helper"
+
+describe HomepagesController do
+ # it "does a thing" do
+ # value(1+1).must_equal 2
+ # end
+end
diff --git a/test/controllers/users_controller_test.rb b/test/controllers/users_controller_test.rb
new file mode 100644
index 0000000000..b58feb8fcc
--- /dev/null
+++ b/test/controllers/users_controller_test.rb
@@ -0,0 +1,7 @@
+require "test_helper"
+
+describe UsersController do
+ # it "does a thing" do
+ # value(1+1).must_equal 2
+ # end
+end
diff --git a/test/controllers/votes_controller_test.rb b/test/controllers/votes_controller_test.rb
new file mode 100644
index 0000000000..f7e8eae4dd
--- /dev/null
+++ b/test/controllers/votes_controller_test.rb
@@ -0,0 +1,35 @@
+require "test_helper"
+
+describe VotesController do
+ describe "upvote" do
+ it "cannot be performed if no user logged in" do
+ work = works(:list_work_4)
+
+ expect {
+ post upvote_path(work.id)
+ }.must_differ "Vote.count", 0
+ must_redirect_to work_path(work.id)
+ end
+
+ it "can upvote a work by a logged-in user if the work hasn't been upvoted" do
+ user = perform_login(users(:user1))
+ work = works(:list_work_4)
+
+ expect {
+ post upvote_path(work.id)
+ }.must_differ "Vote.count", 1
+ must_redirect_to work_path(work.id)
+ end
+
+ it "doesn't allow an user to upvote the same work more one time" do
+ user = perform_login(users(:user1))
+ work = works(:list_work_4)
+ vote = Vote.create(user: user, work: work)
+
+ expect {
+ post upvote_path(work.id)
+ }.must_differ "Vote.count", 0
+ must_redirect_to work_path(work.id)
+ end
+ end
+end
diff --git a/test/controllers/works_controller_test.rb b/test/controllers/works_controller_test.rb
new file mode 100644
index 0000000000..5279080847
--- /dev/null
+++ b/test/controllers/works_controller_test.rb
@@ -0,0 +1,7 @@
+require "test_helper"
+
+describe WorksController do
+ # it "does a thing" do
+ # value(1+1).must_equal 2
+ # end
+end
diff --git a/test/fixtures/.keep b/test/fixtures/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/fixtures/files/.keep b/test/fixtures/files/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/fixtures/users.yml b/test/fixtures/users.yml
new file mode 100644
index 0000000000..44b0632ce4
--- /dev/null
+++ b/test/fixtures/users.yml
@@ -0,0 +1,8 @@
+user1:
+ username: user1
+
+user2:
+ username: user2
+
+user3:
+ username: user3
\ No newline at end of file
diff --git a/test/fixtures/votes.yml b/test/fixtures/votes.yml
new file mode 100644
index 0000000000..daf5b98444
--- /dev/null
+++ b/test/fixtures/votes.yml
@@ -0,0 +1,24 @@
+vote1:
+ user: user1
+ work: list_work_1
+
+vote2:
+ user: user2
+ work: list_work_1
+
+vote3:
+ user: user3
+ work: list_work_1
+
+vote4:
+ user: user1
+ work: list_work_2
+
+vote5:
+ user: user2
+ work: list_work_2
+
+vote6:
+ user: user1
+ work: list_work_3
+
diff --git a/test/fixtures/works.yml b/test/fixtures/works.yml
new file mode 100644
index 0000000000..7b7cc07776
--- /dev/null
+++ b/test/fixtures/works.yml
@@ -0,0 +1,15 @@
+list_work_1:
+ title: POODR1
+ category: ruby
+
+list_work_2:
+ title: POODR2
+ category: ruby
+
+list_work_3:
+ title: POODR3
+ category: ruby
+
+list_work_4:
+ title: POODR4
+ category: ruby
diff --git a/test/helpers/.keep b/test/helpers/.keep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/helpers/application_helper_test.rb b/test/helpers/application_helper_test.rb
new file mode 100644
index 0000000000..96d3717a9c
--- /dev/null
+++ b/test/helpers/application_helper_test.rb
@@ -0,0 +1,21 @@
+require "test_helper"
+
+describe ApplicationHelper, :helper do
+ describe "readable_date" do
+ it "produces a tag with the full timestamp" do
+ date = Date.today - 14
+
+ result = readable_date(date)
+
+ expect(result).must_include date.to_s
+ end
+
+ it "returns [unknown] if the date is nil" do
+ date = nil
+
+ result = readable_date(date)
+
+ expect(result).must_equal "[unknown]"
+ end
+ end
+end
\ No newline at end of file
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/user_test.rb b/test/models/user_test.rb
new file mode 100644
index 0000000000..1760b165f1
--- /dev/null
+++ b/test/models/user_test.rb
@@ -0,0 +1,28 @@
+require "test_helper"
+
+describe User do
+ describe "validations" do
+ it "can be valid" do
+ is_valid = users(:user1).valid?
+ assert(is_valid)
+ end
+
+ it "is invalid if there is no username" do
+ user = users(:user1)
+ user.username = nil
+ is_valid = user.valid?
+ refute(is_valid)
+ end
+ end
+
+ describe "relationships" do
+ it 'can set the vote through "votes"' do
+ Vote.destroy_all
+ vote = Vote.new(work: works(:list_work_1))
+ user = users(:user1)
+ user.votes = [vote]
+
+ expect(user.votes.first).must_equal vote
+ end
+ end
+end
diff --git a/test/models/vote_test.rb b/test/models/vote_test.rb
new file mode 100644
index 0000000000..ef860c73c9
--- /dev/null
+++ b/test/models/vote_test.rb
@@ -0,0 +1,73 @@
+require "test_helper"
+
+describe Vote do
+ describe "validations" do
+ it "can be valid" do
+ is_valid = votes(:vote1).valid?
+ assert(is_valid)
+ end
+
+ it "is invalid if there is no user" do
+ vote = votes(:vote1)
+ vote.user = nil
+ is_valid = vote.valid?
+ refute(is_valid)
+ end
+
+ it "is invalid if there is no work" do
+ vote = votes(:vote1)
+ vote.work = nil
+ is_valid = vote.valid?
+ refute(is_valid)
+ end
+
+ it "is invalid if a vote has the same user and work with an existed vote" do
+ vote = votes(:vote1)
+ same_vote = Vote.create(user: vote.user, work: vote.work)
+ is_valid = same_vote.valid?
+ refute(is_valid)
+ end
+ end
+
+ describe "relationships" do
+ describe "belongs to work" do
+ it "can set the work_id through work" do
+ user = users(:user1)
+ work = works(:list_work_4)
+ vote = Vote.new(user: user)
+ vote.work = work
+
+ expect(vote.work_id).must_equal work.id
+ end
+
+ it "can set the work through work_id" do
+ user = users(:user1)
+ work = works(:list_work_4)
+ vote = Vote.new(user: user)
+ vote.work_id = work.id
+
+ expect(vote.work).must_equal work
+ end
+ end
+
+ describe "belongs to user" do
+ it "can set the user_id through user" do
+ user = users(:user1)
+ work = works(:list_work_4)
+ vote = Vote.new(work: work)
+ vote.user = user
+
+ expect(vote.user_id).must_equal user.id
+ end
+
+ it "can set the work through work_id" do
+ user = users(:user1)
+ work = works(:list_work_4)
+ vote = Vote.new(work: work)
+ vote.user_id = user.id
+
+ expect(vote.user).must_equal user
+ end
+ end
+ end
+end
diff --git a/test/models/work_test.rb b/test/models/work_test.rb
new file mode 100644
index 0000000000..80c9fd17ad
--- /dev/null
+++ b/test/models/work_test.rb
@@ -0,0 +1,82 @@
+require "test_helper"
+
+describe Work do
+ describe "validations" do
+ it "can be valid" do
+ is_valid = works(:list_work_1).valid?
+ assert(is_valid)
+ end
+
+ it "is invalid if there is no title" do
+ work = works(:list_work_1)
+ work.title = nil
+ is_valid = work.valid?
+ refute(is_valid)
+ end
+
+ it "is invalid if there is no category" do
+ work = works(:list_work_1)
+ work.category = nil
+ is_valid = work.valid?
+ refute(is_valid)
+ end
+ end
+
+ describe "relationships" do
+ it 'can set the vote through "votes"' do
+ Vote.destroy_all
+ vote = Vote.new(user: users(:user1))
+ work = works(:list_work_1)
+ work.votes = [vote]
+
+ expect(work.votes.first).must_equal vote
+ end
+ end
+
+ describe "category list" do
+ it "returns an array of work instances and the array is sorted by votes'count of the work" do
+ work1 = works(:list_work_1)
+ work2 = works(:list_work_2)
+ work3 = works(:list_work_3)
+ work4 = works(:list_work_4)
+
+ list = Work.category_list("ruby")
+
+ correct_order_list = [work1, work2, work3, work4].sort_by { |work| -work.votes.length }
+
+ expect(list).must_be_instance_of Array
+
+ list.each_with_index do |work, index|
+ expect(work).must_be_instance_of Work
+ expect(work.category).must_equal "ruby"
+ expect(work).must_equal correct_order_list[index]
+ end
+ end
+
+ it "returns an empty array if no work is found in a category" do
+ empty_list = Work.category_list("music")
+ expect(empty_list).must_equal []
+ end
+
+ end
+
+ describe "spotlight" do
+ it "returns a work with the most votes" do
+ works = Work.all
+ list = works.sort_by{|work| work.votes.length}
+
+ expect(Work.spotlight).must_be_instance_of Work
+ expect(Work.spotlight).must_equal list.last
+ end
+
+ it "returns the first work if more than one works having the same amount of votes" do
+ Work.destroy_all
+
+ work1 = Work.create(title: "game1", category: "game")
+ work2 = Work.create(title: "game2", category: "game")
+
+ expect(Work.spotlight).must_be_instance_of Work
+ expect(Work.spotlight).must_equal work1
+ end
+ 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..d6644ec7aa
--- /dev/null
+++ b/test/test_helper.rb
@@ -0,0 +1,26 @@
+ENV['RAILS_ENV'] ||= 'test'
+require_relative '../config/environment'
+require 'rails/test_help'
+require 'minitest/rails'
+require 'minitest/autorun'
+require 'minitest/reporters'
+
+class ActiveSupport::TestCase
+ # 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...
+ def perform_login(user = nil)
+ user ||= User.first
+
+ login_data = {
+ user: {
+ username: user.username,
+ },
+ }
+
+ post login_path, params: login_data
+ expect(session[:user_id]).must_equal user.id
+ return user
+ end
+end
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