diff --git a/.circleci/config.yml b/.circleci/config.yml new file mode 100644 index 00000000..594b0cce --- /dev/null +++ b/.circleci/config.yml @@ -0,0 +1,93 @@ +version: 2 + +orbs: + browser-tools: circleci/browser-tools@1.4.0 + +jobs: + build: + working_directory: ~/otb-api + + # Primary container image where all commands run + + docker: + - image: cimg/ruby:3.1.2-browsers + environment: + PGUSER: root + RAILS_ENV: test + DB_HOSTNAME: 127.0.0.1 + DB_USERNAME: user + DB_PASSWORD: password + TEST_DB_NAME: otb + MYSQL_ALLOW_EMPTY_PASSWORD: true + CI: 'circleci' + BROWSER_PATH: /usr/bin/google-chrome + + - image: circleci/postgres:10 + environment: + POSTGRES_USER: user + POSTGRES_DB: otb + POSTGRES_PASSWORD: password + + - image: circleci/mysql:latest + command: [--default-authentication-plugin=mysql_native_password] + environment: + MYSQL_DATABASE: otb + MYSQL_USER: user + MYSQL_PASSWORD: password + MYSQL_ALLOW_EMPTY_PASSWORD: yes + MYSQL_ROOT_PASSWORD: password + + steps: + - checkout + - run: + name: Install dependencies + command: | + wget -q -O - https://dl.google.com/linux/linux_signing_key.pub | sudo apt-key add - + sudo apt update + sudo apt install -y postgresql-client || true + sudo apt install -y imagemagick libappindicator1 fonts-liberation + wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb + sudo dpkg -i google-chrome*.deb + gem install bundle + bundle check --path=vendor/bundle || bundle install --path=vendor/bundle --jobs 4 --retry 3 + + - run: + name: Make Tmp Directory + command: | + sudo mkdir -p /data/tmp + sudo chmod 777 /data/tmp + sudo chown $USER:$USER /data/tmp + sudo mkdir -p public/storage/tmp + sudo chmod 777 public/storage/tmp + sudo chown $USER:$USER public/storage/tmp + + # Test building MySQL + # TODO: Figure out how to allow the database user to create new tenants in + # the MySQL Docker instance. For now, we'll just test that it can setup the + # initial database using MySQL. + # - run: + # name: Wait for DB + # command: dockerize -wait tcp://127.0.0.1:3306 -timeout 1m + + # - run: + # name: MySQL Setup + # command: | + # export DB_ADAPTER=mysql2 + # bundle exec rake db:drop RAILS_ENV=test DB_ADAPTER=mysql2 + # bundle exec rake db:create RAILS_ENV=test DB_ADAPTER=mysql2 + # bundle exec rake db:schema:load RAILS_ENV=test DB_ADAPTER=mysql2 + # bundle exec rake db:migrate RAILS_ENV=test DB_ADAPTER=mysql2 + + # Test using PostgreSQL + - run: + name: PostgreSQL Setup + command: | + export DB_ADAPTER=postgresql + bundle exec rake db:drop RAILS_ENV=test DB_ADAPTER=postgresql + bundle exec rake db:create RAILS_ENV=test DB_ADAPTER=postgresql + bundle exec rake db:schema:load RAILS_ENV=test DB_ADAPTER=postgresql + bundle exec rake db:migrate RAILS_ENV=test DB_ADAPTER=postgresql + + - run: + name: Parallel RSpec with PostgreSQL + command: DB_ADAPTER=postgresql bundle exec rspec spec/ diff --git a/.coveralls.yml b/.coveralls.yml deleted file mode 100644 index 6e649991..00000000 --- a/.coveralls.yml +++ /dev/null @@ -1 +0,0 @@ -service_name: travis-ci \ No newline at end of file diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 00000000..93ded0d9 --- /dev/null +++ b/.dockerignore @@ -0,0 +1,54 @@ +# Git +.git +.gitignore + +# Documentation +README.md +CHANGELOG.md +LICENSE + +# Development files +.env* + +# Dependencies +node_modules +vendor/bundle/.bundle + +# Build artifacts +tmp/* +log/* +coverage/* +public/storage/* +storage/* + +# Test files +spec/ +test/ +.rspec + +# Development tools +.vscode/ +.idea/ +*.swp +*.swo +*~ + +# OS generated files +.DS_Store +.DS_Store? +._* +.Spotlight-V100 +.Trashes +ehthumbs.db +Thumbs.db + +# Rails specific +/log/*.log +/tmp +/storage +/public/storage +/coverage + +# Docker +Dockerfile* +docker-compose* \ No newline at end of file diff --git a/.github/workflows/deploy.yml b/.github/workflows/deploy.yml new file mode 100644 index 00000000..4faa2c7f --- /dev/null +++ b/.github/workflows/deploy.yml @@ -0,0 +1,112 @@ +name: Deploy + +on: + push: + branches: [ develop, main ] + workflow_dispatch: + inputs: + environment: + description: 'Environment to deploy to' + required: true + default: 'staging' + type: choice + options: + - staging + - production + +env: + RAILS_ENV: production + +permissions: + id-token: write # Required for AWS OIDC + contents: read + +jobs: + test: + runs-on: ubuntu-latest + + services: + postgres: + image: postgres:13 + env: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: otb_test + options: >- + --health-cmd pg_isready + --health-interval 10s + --health-timeout 5s + --health-retries 5 + ports: + - 5432:5432 + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Set up Ruby + uses: ruby/setup-ruby@v1 + with: + ruby-version: 3.1.2 + bundler-cache: true + + - name: Install system dependencies + run: | + sudo apt-get update + sudo apt-get install -y postgresql-client imagemagick libpq-dev gdal-bin libgdal-dev + + - name: Create directories + run: | + sudo mkdir -p /data/tmp + sudo chmod 777 /data/tmp + mkdir -p public/storage/tmp + + - name: Set up database + run: | + export DB_ADAPTER=postgresql + until pg_isready -h localhost -p 5432 -U postgres; do sleep 2; done + bundle exec rake db:create RAILS_ENV=test DB_ADAPTER=postgresql + bundle exec rake db:schema:load RAILS_ENV=test DB_ADAPTER=postgresql + + - name: Run tests + run: | + DB_ADAPTER=postgresql bundle exec rspec --format progress + + deploy: + needs: test + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Configure AWS credentials + uses: aws-actions/configure-aws-credentials@v4 + with: + role-to-assume: ${{ secrets.AWS_ROLE }} + aws-region: us-east-1 + role-session-name: GitHub-OIDC-OTB + + - name: Deploy to ECS + env: + AWS_ECR: ${{ secrets.AWS_ECR }} + AWS_ECS_CLUSTER_DEV: ${{ secrets.AWS_ECS_CLUSTER_DEV }} + AWS_ECS_CLUSTER_PROD: ${{ secrets.AWS_ECS_CLUSTER_PROD }} + AWS_ECS_SERVICE_DEV: ${{ secrets.AWS_ECS_SERVICE_DEV }} + AWS_ECS_SERVICE_PROD: ${{ secrets.AWS_ECS_SERVICE_PROD }} + AWS_REGION: ${{ secrets.AWS_REGION }} + BRANCH: ${{ github.ref_name }} + run: | + chmod +x build.sh + ./build.sh + + - name: Notify deployment status + if: always() + run: | + branch="${{ github.ref_name }}" + environment=$([ "$branch" == "main" ] && echo "production" || echo "staging") + if [[ "${{ job.status }}" == "success" ]]; then + echo "Successfully deployed to $environment via ECS" + else + echo "Deployment to $environment failed" + fi \ No newline at end of file diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml new file mode 100644 index 00000000..55109b48 --- /dev/null +++ b/.github/workflows/test.yml @@ -0,0 +1,81 @@ +name: Test + +on: + push: + branches: [ main, develop ] + pull_request: + branches: [ main, develop ] + +env: + RAILS_ENV: test + PGUSER: postgres + DB_HOSTNAME: localhost + DB_USERNAME: postgres + DB_PASSWORD: postgres + TEST_DB_NAME: otb_test + CI: 'github-actions' + BROWSER_PATH: /usr/bin/google-chrome + +jobs: + test: + runs-on: ubuntu-latest + + services: + postgres: + image: postgres:13 + env: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: otb_test + options: >- + --health-cmd pg_isready + --health-interval 10s + --health-timeout 5s + --health-retries 5 + ports: + - 5432:5432 + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Set up Ruby + uses: ruby/setup-ruby@v1 + with: + ruby-version: 3.1.2 + bundler-cache: true + + - name: Install system dependencies + run: | + sudo apt-get update + sudo apt-get install -y postgresql-client imagemagick libpq-dev gdal-bin libgdal-dev fonts-liberation + wget -q https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb + sudo dpkg -i google-chrome-stable_current_amd64.deb || true + sudo apt-get install -f -y + + - name: Create required directories + run: | + sudo mkdir -p /data/tmp + sudo chmod 777 /data/tmp + sudo chown $USER:$USER /data/tmp + mkdir -p public/storage/tmp + chmod 777 public/storage/tmp + + - name: Wait for database to be ready + run: | + until pg_isready -h localhost -p 5432 -U postgres; do + echo "Waiting for PostgreSQL..." + sleep 2 + done + + - name: Set up PostgreSQL database + run: | + export DB_ADAPTER=postgresql + bundle exec rake db:drop RAILS_ENV=test DB_ADAPTER=postgresql || true + bundle exec rake db:create RAILS_ENV=test DB_ADAPTER=postgresql + bundle exec rake db:schema:load RAILS_ENV=test DB_ADAPTER=postgresql + bundle exec rake db:migrate RAILS_ENV=test DB_ADAPTER=postgresql + + - name: Run RSpec tests + run: | + DB_ADAPTER=postgresql bundle exec rspec --format progress \ No newline at end of file diff --git a/.gitignore b/.gitignore index e95e6825..8930b17f 100644 --- a/.gitignore +++ b/.gitignore @@ -1,12 +1,15 @@ *.DS_Store coverage -config/database.yml log config/secrets.yml tmp config/initializers/auth.rb public/uploads/ +public/storage/* .vscode *.env* -.ruby-version +.bundle +.coveralls.yml +/config/master.key +storage diff --git a/.ruby-version b/.ruby-version new file mode 100644 index 00000000..ef538c28 --- /dev/null +++ b/.ruby-version @@ -0,0 +1 @@ +3.1.2 diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 35575793..00000000 --- a/.travis.yml +++ /dev/null @@ -1,28 +0,0 @@ -language: ruby -rvm: - - 2.6.3 - -services: - - postgresql - - mysql - -before_install: - - cp config/database.yml.travis config/database.yml - - cp config/secrets.yml.dist config/secrets.yml - - gem install bundler - -env: - - DB=mysql - - DB=postgres - -before_script: - - sudo service postgresql restart - - psql -c 'create database otb_test;' -U postgres - - mysql -e 'create database otb_test' - -script: - - bundle install - - bundle exec rake db:drop - - bundle exec rake db:create - - bundle exec rake db:schema:load - - bundle exec rspec ./spec/requests diff --git a/Capfile b/Capfile index 51956b36..5fad1378 100644 --- a/Capfile +++ b/Capfile @@ -29,7 +29,7 @@ install_plugin Capistrano::SCM::Git # require 'capistrano/rvm' require 'capistrano/rbenv' set :rbenv_type, :user # or :system, depends on your rbenv setup -set :rbenv_ruby, '2.6.3' +set :rbenv_ruby, '3.0.2' # require 'capistrano/chruby' require 'capistrano/bundler' # require 'capistrano/rails/assets' diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 00000000..a2610361 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,34 @@ +FROM ruby:3.1.2-slim + +RUN apt-get update -qq && \ + apt-get install -y --no-install-recommends \ + build-essential \ + curl \ + git \ + postgresql-client \ + libpq-dev \ + libgdal-dev \ + gdal-bin \ + imagemagick \ + libmagickwand-dev \ + libvips \ + pkg-config \ + && rm -rf /var/lib/apt/lists/* + +WORKDIR /rails + +COPY Gemfile Gemfile.lock ./ + +RUN gem install bundler:2.2.22 +RUN bundle install + +COPY . . + +RUN mkdir -p /data/tmp public/storage/tmp tmp/pids && \ + chmod 777 /data/tmp public/storage/tmp tmp/pids + +EXPOSE 3000 + +RUN chmod +x ./entrypoint.sh + +ENTRYPOINT ["./entrypoint.sh"] \ No newline at end of file diff --git a/Gemfile b/Gemfile index 4d012e24..d948983e 100644 --- a/Gemfile +++ b/Gemfile @@ -9,32 +9,42 @@ end # Bundle edge Rails instead: gem 'rails', github: 'rails/rails' -gem 'rails', '~> 5.2.0' -gem "rack", ">= 2.0.6" +# gem 'rails', '~> 7.0.4' +gem 'rails', '~> 6.1.0' + +gem 'rack', '>= 2.0.6' gem 'pg' gem 'mysql2' # Multitenancy for Rails and ActiveRecord -gem 'apartment' +gem 'ros-apartment', require: 'apartment' # For JSONAPI responses -gem 'active_model_serializers', '~> 0.10.0.rc3' -gem 'acts-as-taggable-on', '~> 5.0' +gem 'active_model_serializers', '~> 0.10.12' +# For pagination +gem 'kaminari' +# gem 'pagy', '~> 5.10' # Use Puma as the app server gem 'puma', '~> 4.3.0' # Use Redis adapter to run Action Cable in production gem 'redis', '~> 3.0' -gem "actionview", ">= 5.2.2.1" +gem 'actionview', '>= 5.2.2.1' # Use ActiveModel has_secure_password # gem 'bcrypt', '~> 3.1.7' -# Social Auth # gem 'ecds_rails_auth_engine', path: '../ecds_auth_engine' -gem 'ecds_rails_auth_engine', git: 'https://github.com/ecds/ecds_rails_auth_engine.git', :tag => 'v0.1.5' -gem 'cancancan', '~> 2.0' +gem 'ecds_rails_auth_engine', git: 'https://github.com/ecds/ecds_rails_auth_engine.git', branch: 'feature/fauxoauth' # Active Storage will land in 5.2 gem 'carrierwave', '~> 1.0' gem 'mini_magick' -gem 'carrierwave-base64' +gem 'image_processing', '~> 1.2' +gem 'ferrum' +gem 'aws-sdk-s3', '~> 1' + +# RGeo is a geospatial data library for Ruby. +# https://github.com/rgeo/rgeo +gem 'rgeo' +gem 'ipinfo-rails' + # Vidoe provider APIs gem 'vimeo' @@ -45,20 +55,20 @@ gem 'youtube_rails' gem 'rack-cors' # TODO: should probably only require this for :test -gem 'faker', git: 'https://github.com/stympy/faker.git', branch: 'master' +gem 'faker' 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] - gem "test-prof" + # gem "test-prof" end group :development do - gem 'listen', '>= 3.0.5', '< 3.2' + gem 'listen' # 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' - gem 'rspec-rails', '~> 3.5' + gem 'rspec-rails', '~> 5.1.2' # Use Capistrano for deployment gem 'capistrano-rails' gem 'capistrano-rbenv', '~> 2.0' @@ -67,13 +77,19 @@ end group :test do - gem "factory_bot" + gem 'factory_bot' gem 'factory_bot_rails' - gem 'shoulda-matchers', git: 'https://github.com/thoughtbot/shoulda-matchers.git', branch: 'rails-5' + gem 'shoulda-matchers', '~> 4.5.1' #git: 'https://github.com/thoughtbot/shoulda-matchers.git', branch: 'rails-5' gem 'database_cleaner' - gem 'coveralls', require: false gem 'webmock' + gem 'coveralls', require: false + gem 'simplecov', require: false + gem 'simplecov-lcov', require: false + gem 'term-ansicolor' end # Windows does not include zoneinfo files, so bundle the tzinfo-data gem gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby] +gem 'net-smtp', require: false +gem 'net-imap', require: false +gem 'net-pop', require: false diff --git a/Gemfile.backup b/Gemfile.backup new file mode 100644 index 00000000..eb885ebc --- /dev/null +++ b/Gemfile.backup @@ -0,0 +1,80 @@ +# frozen_string_literal: true + +source 'https://rubygems.org' + +git_source(:github) do |repo_name| + repo_name = "#{repo_name}/#{repo_name}" unless repo_name.include?('/') + "https://github.com/#{repo_name}.git" +end + + +# Bundle edge Rails instead: gem 'rails', github: 'rails/rails' +gem 'rails', '~> 5.2.0' +gem "rack", ">= 2.0.6" +gem 'pg' +gem 'mysql2' +# Multitenancy for Rails and ActiveRecord +gem 'apartment' +# For JSONAPI responses +gem 'active_model_serializers', '~> 0.10.0.rc3' +gem 'acts-as-taggable-on', '~> 5.0' +# Use Puma as the app server +gem 'puma', '~> 4.3.0' +# Use Redis adapter to run Action Cable in production +gem 'redis', '~> 3.0' +gem "actionview", ">= 5.2.2.1" +# Use ActiveModel has_secure_password +# gem 'bcrypt', '~> 3.1.7' + +# Social Auth +# gem 'ecds_rails_auth_engine', path: '../ecds_auth_engine' +gem 'ecds_rails_auth_engine', git: 'https://github.com/ecds/ecds_rails_auth_engine.git', :tag => 'v0.1.5' +gem 'cancancan', '~> 2.0' + +# Active Storage will land in 5.2 +gem 'carrierwave', '~> 1.0' +gem 'mini_magick' +gem 'carrierwave-base64' + +# Vidoe provider APIs +gem 'vimeo' +gem 'yt' +gem 'youtube_rails' + +# Use Rack CORS for handling Cross-Origin Resource Sharing (CORS), making cross-origin AJAX possible +gem 'rack-cors' + +# TODO: should probably only require this for :test +gem 'faker', git: 'https://github.com/stympy/faker.git', branch: 'main' + +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] + gem "test-prof" +end + +group :development do + 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' + gem 'rspec-rails', '~> 3.5' + # Use Capistrano for deployment + gem 'capistrano-rails' + gem 'capistrano-rbenv', '~> 2.0' + gem 'capistrano-passenger' +end + + +group :test do + gem "factory_bot" + gem 'factory_bot_rails' + gem 'shoulda-matchers', git: 'https://github.com/thoughtbot/shoulda-matchers.git', branch: 'rails-5' + gem 'database_cleaner' + gem 'coveralls', require: false + gem 'webmock' +end + +# Windows does not include zoneinfo files, so bundle the tzinfo-data gem +gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby] +gem "mimemagic", ">= 0.3.5" diff --git a/Gemfile.lock b/Gemfile.lock index 4a680fd4..48a30662 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -1,348 +1,438 @@ GIT remote: https://github.com/ecds/ecds_rails_auth_engine.git - revision: 834d77e7ec3a18e21cd490b91ff99770ea0afaad - tag: v0.1.5 + revision: fe56d7d214403689a0ab0ab79cbc9d343fdc63f6 + branch: feature/fauxoauth specs: - ecds_rails_auth_engine (0.1.5) - cancancan (~> 2.0) - rails (~> 5.2.0) - rails_api_auth - -GIT - remote: https://github.com/stympy/faker.git - revision: 7966190d46fd8165b58f3be1bffe41d37ccaea7c - branch: master - specs: - faker (2.10.2) - i18n (>= 1.6, < 2) - -GIT - remote: https://github.com/thoughtbot/shoulda-matchers.git - revision: 4b160bd19ecca7f97d7ac22dccd5fde9b0da5a9f - branch: rails-5 - specs: - shoulda-matchers (3.1.2) - activesupport (>= 4.2.0) + ecds_rails_auth_engine (0.2.0) + cancancan + httparty + jwt + rails GEM remote: https://rubygems.org/ specs: - actioncable (5.2.4.1) - actionpack (= 5.2.4.1) + IPinfo (1.0.1) + faraday (~> 1.0) + json (~> 2.1) + lru_redux (~> 1.1) + actioncable (6.1.6.1) + actionpack (= 6.1.6.1) + activesupport (= 6.1.6.1) nio4r (~> 2.0) websocket-driver (>= 0.6.1) - actionmailer (5.2.4.1) - actionpack (= 5.2.4.1) - actionview (= 5.2.4.1) - activejob (= 5.2.4.1) + actionmailbox (6.1.6.1) + actionpack (= 6.1.6.1) + activejob (= 6.1.6.1) + activerecord (= 6.1.6.1) + activestorage (= 6.1.6.1) + activesupport (= 6.1.6.1) + mail (>= 2.7.1) + actionmailer (6.1.6.1) + actionpack (= 6.1.6.1) + actionview (= 6.1.6.1) + activejob (= 6.1.6.1) + activesupport (= 6.1.6.1) mail (~> 2.5, >= 2.5.4) rails-dom-testing (~> 2.0) - actionpack (5.2.4.1) - actionview (= 5.2.4.1) - activesupport (= 5.2.4.1) - rack (~> 2.0, >= 2.0.8) + actionpack (6.1.6.1) + actionview (= 6.1.6.1) + activesupport (= 6.1.6.1) + rack (~> 2.0, >= 2.0.9) rack-test (>= 0.6.3) rails-dom-testing (~> 2.0) - rails-html-sanitizer (~> 1.0, >= 1.0.2) - actionview (5.2.4.1) - activesupport (= 5.2.4.1) + rails-html-sanitizer (~> 1.0, >= 1.2.0) + actiontext (6.1.6.1) + actionpack (= 6.1.6.1) + activerecord (= 6.1.6.1) + activestorage (= 6.1.6.1) + activesupport (= 6.1.6.1) + nokogiri (>= 1.8.5) + actionview (6.1.6.1) + activesupport (= 6.1.6.1) builder (~> 3.1) erubi (~> 1.4) rails-dom-testing (~> 2.0) - rails-html-sanitizer (~> 1.0, >= 1.0.3) - active_model_serializers (0.10.10) - actionpack (>= 4.1, < 6.1) - activemodel (>= 4.1, < 6.1) + rails-html-sanitizer (~> 1.1, >= 1.2.0) + active_model_serializers (0.10.13) + actionpack (>= 4.1, < 7.1) + activemodel (>= 4.1, < 7.1) case_transform (>= 0.2) jsonapi-renderer (>= 0.1.1.beta1, < 0.3) - activejob (5.2.4.1) - activesupport (= 5.2.4.1) + activejob (6.1.6.1) + activesupport (= 6.1.6.1) globalid (>= 0.3.6) - activemodel (5.2.4.1) - activesupport (= 5.2.4.1) - activerecord (5.2.4.1) - activemodel (= 5.2.4.1) - activesupport (= 5.2.4.1) - arel (>= 9.0) - activestorage (5.2.4.1) - actionpack (= 5.2.4.1) - activerecord (= 5.2.4.1) - marcel (~> 0.3.1) - activesupport (5.2.4.1) + activemodel (6.1.6.1) + activesupport (= 6.1.6.1) + activerecord (6.1.6.1) + activemodel (= 6.1.6.1) + activesupport (= 6.1.6.1) + activestorage (6.1.6.1) + actionpack (= 6.1.6.1) + activejob (= 6.1.6.1) + activerecord (= 6.1.6.1) + activesupport (= 6.1.6.1) + marcel (~> 1.0) + mini_mime (>= 1.1.0) + activesupport (6.1.6.1) concurrent-ruby (~> 1.0, >= 1.0.2) - i18n (>= 0.7, < 2) - minitest (~> 5.1) - tzinfo (~> 1.1) - acts-as-taggable-on (5.0.0) - activerecord (>= 4.2.8) - addressable (2.7.0) + i18n (>= 1.6, < 2) + minitest (>= 5.1) + tzinfo (~> 2.0) + zeitwerk (~> 2.3) + addressable (2.8.0) public_suffix (>= 2.0.2, < 5.0) - airbrussh (1.4.0) + airbrussh (1.4.1) sshkit (>= 1.6.1, != 1.7.0) - apartment (2.2.1) - activerecord (>= 3.1.2, < 6.0) - parallel (>= 0.7.1) - public_suffix (>= 2) - rack (>= 1.3.6) - arel (9.0.0) - bcrypt (3.1.13) - bcrypt (3.1.13-java) + aws-eventstream (1.2.0) + aws-partitions (1.613.0) + aws-sdk-core (3.131.5) + aws-eventstream (~> 1, >= 1.0.2) + aws-partitions (~> 1, >= 1.525.0) + aws-sigv4 (~> 1.1) + jmespath (~> 1, >= 1.6.1) + aws-sdk-kms (1.58.0) + aws-sdk-core (~> 3, >= 3.127.0) + aws-sigv4 (~> 1.1) + aws-sdk-s3 (1.114.0) + aws-sdk-core (~> 3, >= 3.127.0) + aws-sdk-kms (~> 1) + aws-sigv4 (~> 1.4) + aws-sigv4 (1.5.1) + aws-eventstream (~> 1, >= 1.0.2) builder (3.2.4) - cancancan (2.3.0) - capistrano (3.12.1) + cancancan (3.4.0) + capistrano (3.17.0) airbrussh (>= 1.0.0) i18n rake (>= 10.0.0) sshkit (>= 1.9.0) - capistrano-bundler (1.6.0) + capistrano-bundler (2.1.0) capistrano (~> 3.1) - capistrano-passenger (0.2.0) + capistrano-passenger (0.2.1) capistrano (~> 3.0) - capistrano-rails (1.4.0) + capistrano-rails (1.6.2) capistrano (~> 3.1) - capistrano-bundler (~> 1.1) - capistrano-rbenv (2.1.6) + capistrano-bundler (>= 1.1, < 3) + capistrano-rbenv (2.2.0) capistrano (~> 3.1) sshkit (~> 1.3) - carrierwave (1.3.1) + carrierwave (1.3.2) activemodel (>= 4.0.0) activesupport (>= 4.0.0) mime-types (>= 1.16) - carrierwave-base64 (2.8.0) - carrierwave (>= 0.8.0) - mime-types (~> 3.0) - mimemagic (~> 0.3.2) + ssrf_filter (~> 1.0) case_transform (0.2) activesupport - concurrent-ruby (1.1.6) - coveralls (0.8.23) - json (>= 1.8, < 3) - simplecov (~> 0.16.1) - term-ansicolor (~> 1.3) - thor (>= 0.19.4, < 2.0) - tins (~> 1.6) - crack (0.4.3) - safe_yaml (~> 1.0.0) + cliver (0.3.2) + concurrent-ruby (1.1.10) + coveralls (0.7.1) + multi_json (~> 1.3) + rest-client + simplecov (>= 0.7) + term-ansicolor + thor + crack (0.4.5) + rexml crass (1.0.6) - database_cleaner (1.8.3) - diff-lcs (1.3) - docile (1.3.2) - erubi (1.9.0) - factory_bot (5.1.1) - activesupport (>= 4.2.0) - factory_bot_rails (5.1.1) - factory_bot (~> 5.1.0) - railties (>= 4.2.0) - ffi (1.12.2) - ffi (1.12.2-java) - ffi (1.12.2-x64-mingw32) - ffi (1.12.2-x86-mingw32) - globalid (0.4.2) - activesupport (>= 4.2.0) + database_cleaner (2.0.1) + database_cleaner-active_record (~> 2.0.0) + database_cleaner-active_record (2.0.1) + activerecord (>= 5.a) + database_cleaner-core (~> 2.0.0) + database_cleaner-core (2.0.1) + diff-lcs (1.5.0) + docile (1.4.0) + domain_name (0.5.20190701) + unf (>= 0.0.5, < 1.0.0) + erubi (1.10.0) + factory_bot (6.2.1) + activesupport (>= 5.0.0) + factory_bot_rails (6.2.0) + factory_bot (~> 6.2.0) + railties (>= 5.0.0) + faker (3.5.2) + i18n (>= 1.8.11, < 2) + faraday (1.10.0) + faraday-em_http (~> 1.0) + faraday-em_synchrony (~> 1.0) + faraday-excon (~> 1.1) + faraday-httpclient (~> 1.0) + faraday-multipart (~> 1.0) + faraday-net_http (~> 1.0) + faraday-net_http_persistent (~> 1.0) + faraday-patron (~> 1.0) + faraday-rack (~> 1.0) + faraday-retry (~> 1.0) + ruby2_keywords (>= 0.0.4) + faraday-em_http (1.0.0) + faraday-em_synchrony (1.0.0) + faraday-excon (1.1.0) + faraday-httpclient (1.0.1) + faraday-multipart (1.0.4) + multipart-post (~> 2) + faraday-net_http (1.0.1) + faraday-net_http_persistent (1.2.0) + faraday-patron (1.0.0) + faraday-rack (1.0.0) + faraday-retry (1.0.3) + ferrum (0.11) + addressable (~> 2.5) + cliver (~> 0.3) + concurrent-ruby (~> 1.1) + websocket-driver (>= 0.6, < 0.8) + ffi (1.15.5) + globalid (1.0.0) + activesupport (>= 5.0) hashdiff (1.0.1) - httparty (0.13.7) - json (~> 1.8) + http-accept (1.7.0) + http-cookie (1.0.5) + domain_name (~> 0.5) + httparty (0.20.0) + mime-types (~> 3.0) multi_xml (>= 0.5.2) httpclient (2.8.3) - i18n (1.8.2) + i18n (1.12.0) concurrent-ruby (~> 1.0) - json (1.8.6) - json (1.8.6-java) + image_processing (1.12.2) + mini_magick (>= 4.9.5, < 5) + ruby-vips (>= 2.0.17, < 3) + ipinfo-rails (1.0.1) + IPinfo (~> 1.0.1) + rack (~> 2.0) + jmespath (1.6.1) + json (2.6.2) jsonapi-renderer (0.2.2) - listen (3.1.5) - rb-fsevent (~> 0.9, >= 0.9.4) - rb-inotify (~> 0.9, >= 0.9.7) - ruby_dep (~> 1.2) - loofah (2.4.0) + jwt (2.4.1) + kaminari (1.2.2) + activesupport (>= 4.1.0) + kaminari-actionview (= 1.2.2) + kaminari-activerecord (= 1.2.2) + kaminari-core (= 1.2.2) + kaminari-actionview (1.2.2) + actionview + kaminari-core (= 1.2.2) + kaminari-activerecord (1.2.2) + activerecord + kaminari-core (= 1.2.2) + kaminari-core (1.2.2) + listen (3.7.1) + rb-fsevent (~> 0.10, >= 0.10.3) + rb-inotify (~> 0.9, >= 0.9.10) + loofah (2.18.0) crass (~> 1.0.2) nokogiri (>= 1.5.9) + lru_redux (1.1.0) mail (2.7.1) mini_mime (>= 0.1.1) - marcel (0.3.3) - mimemagic (~> 0.3.2) - method_source (0.9.2) - mime-types (3.3.1) + marcel (1.0.2) + method_source (1.0.0) + mime-types (3.4.1) mime-types-data (~> 3.2015) - mime-types-data (3.2019.1009) - mimemagic (0.3.4) - mini_magick (4.10.1) - mini_mime (1.0.2) - mini_portile2 (2.4.0) - minitest (5.14.0) + mime-types-data (3.2022.0105) + mini_magick (4.11.0) + mini_mime (1.1.2) + minitest (5.16.2) + multi_json (1.15.0) multi_xml (0.6.0) - multipart-post (2.1.1) - mysql2 (0.5.3) - mysql2 (0.5.3-x64-mingw32) - mysql2 (0.5.3-x86-mingw32) - mysql2 (0.5.3-x86-mswin32-60) - net-scp (2.0.0) - net-ssh (>= 2.6.5, < 6.0.0) - net-ssh (5.2.0) - nio4r (2.5.2) - nio4r (2.5.2-java) - nokogiri (1.10.9) - mini_portile2 (~> 2.4.0) - nokogiri (1.10.9-java) - nokogiri (1.10.9-x64-mingw32) - mini_portile2 (~> 2.4.0) - nokogiri (1.10.9-x86-mingw32) - mini_portile2 (~> 2.4.0) - oauth (0.5.4) - parallel (1.19.1) - pg (1.2.3) - pg (1.2.3-x64-mingw32) - pg (1.2.3-x86-mingw32) - public_suffix (4.0.3) - puma (4.3.3) + multipart-post (2.2.3) + mysql2 (0.5.4) + net-imap (0.3.1) + net-protocol + net-pop (0.1.2) + net-protocol + net-protocol (0.1.3) + timeout + net-scp (1.2.1) + net-ssh (>= 2.6.5) + net-smtp (0.3.2) + net-protocol + net-ssh (7.0.1) + netrc (0.11.0) + nio4r (2.5.8) + nokogiri (1.13.8-aarch64-linux) + racc (~> 1.4) + nokogiri (1.13.8-arm64-darwin) + racc (~> 1.4) + nokogiri (1.13.8-x86_64-darwin) + racc (~> 1.4) + nokogiri (1.13.8-x86_64-linux) + racc (~> 1.4) + oauth (0.5.10) + parallel (1.22.1) + pg (1.4.2) + public_suffix (4.0.7) + puma (4.3.12) nio4r (~> 2.0) - puma (4.3.3-java) - nio4r (~> 2.0) - rack (2.2.2) + racc (1.6.0) + rack (2.2.4) rack-cors (1.1.1) rack (>= 2.0.0) - rack-test (1.1.0) - rack (>= 1.0, < 3) - rails (5.2.4.1) - actioncable (= 5.2.4.1) - actionmailer (= 5.2.4.1) - actionpack (= 5.2.4.1) - actionview (= 5.2.4.1) - activejob (= 5.2.4.1) - activemodel (= 5.2.4.1) - activerecord (= 5.2.4.1) - activestorage (= 5.2.4.1) - activesupport (= 5.2.4.1) - bundler (>= 1.3.0) - railties (= 5.2.4.1) + rack-test (2.0.2) + rack (>= 1.3) + rails (6.1.6.1) + actioncable (= 6.1.6.1) + actionmailbox (= 6.1.6.1) + actionmailer (= 6.1.6.1) + actionpack (= 6.1.6.1) + actiontext (= 6.1.6.1) + actionview (= 6.1.6.1) + activejob (= 6.1.6.1) + activemodel (= 6.1.6.1) + activerecord (= 6.1.6.1) + activestorage (= 6.1.6.1) + activesupport (= 6.1.6.1) + bundler (>= 1.15.0) + railties (= 6.1.6.1) sprockets-rails (>= 2.0.0) rails-dom-testing (2.0.3) activesupport (>= 4.2.0) nokogiri (>= 1.6) - rails-html-sanitizer (1.3.0) + rails-html-sanitizer (1.4.3) loofah (~> 2.3) - rails_api_auth (0.1.0) - bcrypt (~> 3.1.7) - httparty (~> 0.13.3) - rails (>= 3.2.6, < 6) - railties (5.2.4.1) - actionpack (= 5.2.4.1) - activesupport (= 5.2.4.1) + railties (6.1.6.1) + actionpack (= 6.1.6.1) + activesupport (= 6.1.6.1) method_source - rake (>= 0.8.7) - thor (>= 0.19.0, < 2.0) - rake (13.0.1) - rb-fsevent (0.10.3) + rake (>= 12.2) + thor (~> 1.0) + rake (13.0.6) + rb-fsevent (0.11.1) rb-inotify (0.10.1) ffi (~> 1.0) redis (3.3.5) - rspec-core (3.9.1) - rspec-support (~> 3.9.1) - rspec-expectations (3.9.1) + rest-client (2.1.0) + http-accept (>= 1.7.0, < 2.0) + http-cookie (>= 1.0.2, < 2.0) + mime-types (>= 1.16, < 4.0) + netrc (~> 0.8) + rexml (3.2.5) + rgeo (2.4.0) + ros-apartment (2.11.0) + activerecord (>= 5.0.0, < 7.1) + parallel (< 2.0) + public_suffix (>= 2.0.5, < 5.0) + rack (>= 1.3.6, < 3.0) + rspec-core (3.11.0) + rspec-support (~> 3.11.0) + rspec-expectations (3.11.0) diff-lcs (>= 1.2.0, < 2.0) - rspec-support (~> 3.9.0) - rspec-mocks (3.9.1) + rspec-support (~> 3.11.0) + rspec-mocks (3.11.1) diff-lcs (>= 1.2.0, < 2.0) - rspec-support (~> 3.9.0) - rspec-rails (3.9.1) - actionpack (>= 3.0) - activesupport (>= 3.0) - railties (>= 3.0) - rspec-core (~> 3.9.0) - rspec-expectations (~> 3.9.0) - rspec-mocks (~> 3.9.0) - rspec-support (~> 3.9.0) - rspec-support (3.9.2) - ruby_dep (1.5.0) - safe_yaml (1.0.5) - simplecov (0.16.1) + rspec-support (~> 3.11.0) + rspec-rails (5.1.2) + actionpack (>= 5.2) + activesupport (>= 5.2) + railties (>= 5.2) + rspec-core (~> 3.10) + rspec-expectations (~> 3.10) + rspec-mocks (~> 3.10) + rspec-support (~> 3.10) + rspec-support (3.11.0) + ruby-vips (2.1.4) + ffi (~> 1.12) + ruby2_keywords (0.0.5) + shoulda-matchers (4.5.1) + activesupport (>= 4.2.0) + simplecov (0.21.2) docile (~> 1.1) - json (>= 1.8, < 3) - simplecov-html (~> 0.10.0) - simplecov-html (0.10.2) - spring (2.1.0) + simplecov-html (~> 0.11) + simplecov_json_formatter (~> 0.1) + simplecov-html (0.12.3) + simplecov-lcov (0.8.0) + simplecov_json_formatter (0.1.4) + spring (2.1.1) spring-watcher-listen (2.0.1) listen (>= 2.7, < 4.0) spring (>= 1.2, < 3.0) - sprockets (4.0.0) + sprockets (4.1.1) concurrent-ruby (~> 1.0) rack (> 1, < 3) - sprockets-rails (3.2.1) - actionpack (>= 4.0) - activesupport (>= 4.0) + sprockets-rails (3.4.2) + actionpack (>= 5.2) + activesupport (>= 5.2) sprockets (>= 3.0.0) - sshkit (1.21.0) + sshkit (1.21.2) net-scp (>= 1.1.2) net-ssh (>= 2.8.0) + ssrf_filter (1.0.7) sync (0.5.0) term-ansicolor (1.7.1) tins (~> 1.0) - test-prof (0.11.3) - thor (1.0.1) - thread_safe (0.3.6) - thread_safe (0.3.6-java) - tins (1.24.1) + thor (1.2.1) + timeout (0.3.0) + tins (1.31.1) sync - tzinfo (1.2.6) - thread_safe (~> 0.1) - tzinfo-data (1.2020.1) - tzinfo (>= 1.0.0) + tzinfo (2.0.5) + concurrent-ruby (~> 1.0) + unf (0.1.4) + unf_ext + unf_ext (0.0.8.2) vimeo (1.5.4) httparty (>= 0.4.5) httpclient (>= 2.1.5.2) json (>= 1.1.9) multipart-post (>= 1.0.1) oauth (>= 0.4.3) - webmock (3.8.3) - addressable (>= 2.3.6) + webmock (3.14.0) + addressable (>= 2.8.0) crack (>= 0.3.2) hashdiff (>= 0.4.0, < 2.0.0) - websocket-driver (0.7.1) - websocket-extensions (>= 0.1.0) - websocket-driver (0.7.1-java) + websocket-driver (0.7.5) websocket-extensions (>= 0.1.0) - websocket-extensions (0.1.4) + websocket-extensions (0.1.5) youtube_rails (1.2.2) - yt (0.32.6) + yt (0.33.4) activesupport + zeitwerk (2.6.0) PLATFORMS - java - ruby - x64-mingw32 - x86-mingw32 - x86-mswin32 + aarch64-linux + arm64-darwin-24 + x86_64-darwin-20 + x86_64-darwin-21 + x86_64-linux DEPENDENCIES actionview (>= 5.2.2.1) - active_model_serializers (~> 0.10.0.rc3) - acts-as-taggable-on (~> 5.0) - apartment - cancancan (~> 2.0) + active_model_serializers (~> 0.10.12) + aws-sdk-s3 (~> 1) capistrano-passenger capistrano-rails capistrano-rbenv (~> 2.0) carrierwave (~> 1.0) - carrierwave-base64 coveralls database_cleaner ecds_rails_auth_engine! factory_bot factory_bot_rails - faker! - listen (>= 3.0.5, < 3.2) + faker + ferrum + image_processing (~> 1.2) + ipinfo-rails + kaminari + listen mini_magick mysql2 + net-imap + net-pop + net-smtp pg puma (~> 4.3.0) rack (>= 2.0.6) rack-cors - rails (~> 5.2.0) + rails (~> 6.1.0) redis (~> 3.0) - rspec-rails (~> 3.5) - shoulda-matchers! + rgeo + ros-apartment + rspec-rails (~> 5.1.2) + shoulda-matchers (~> 4.5.1) + simplecov + simplecov-lcov spring spring-watcher-listen (~> 2.0.0) - test-prof + term-ansicolor tzinfo-data vimeo webmock @@ -350,4 +440,4 @@ DEPENDENCIES yt BUNDLED WITH - 2.1.4 + 2.2.22 diff --git a/Gemfile.original b/Gemfile.original new file mode 100644 index 00000000..e7d8b54d --- /dev/null +++ b/Gemfile.original @@ -0,0 +1,78 @@ +# frozen_string_literal: true + +source 'https://rubygems.org' + +git_source(:github) do |repo_name| + repo_name = "#{repo_name}/#{repo_name}" unless repo_name.include?('/') + "https://github.com/#{repo_name}.git" +end + +# Bundle edge Rails instead: gem 'rails', github: 'rails/rails' +gem 'rails', '~> 5.2.0' +gem "rack", ">= 2.0.6" +gem 'pg' +gem 'mysql2' +# Multitenancy for Rails and ActiveRecord +gem 'apartment', '~> 2.2.0' +# For JSONAPI responses +gem 'active_model_serializers', '~> 0.10.0.rc3' +gem 'acts-as-taggable-on', '~> 5.0' +# Use Puma as the app server +gem 'puma', '~> 4.3.0' +# Use Redis adapter to run Action Cable in production +gem 'redis', '~> 3.0' +gem "actionview", ">= 5.2.2.1" +# Use ActiveModel has_secure_password +# gem 'bcrypt', '~> 3.1.7' + +# Social Auth +# gem 'ecds_rails_auth_engine', path: '../ecds_auth_engine' +gem 'ecds_rails_auth_engine', git: 'https://github.com/ecds/ecds_rails_auth_engine.git', :tag => 'v0.1.5' +gem 'cancancan', '~> 2.0' + +# Active Storage will land in 5.2 +gem 'carrierwave', '~> 1.0' +gem 'mini_magick' +gem 'carrierwave-base64' + +# Video provider APIs +gem 'vimeo' +gem 'yt' +gem 'youtube_rails' + +# Use Rack CORS for handling Cross-Origin Resource Sharing (CORS), making cross-origin AJAX possible +gem 'rack-cors' + +# Faker gem - use version compatible with Ruby 2.7+ +gem 'faker', '~> 2.20' + +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] + gem "test-prof" +end + +group :development do + 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' + gem 'rspec-rails', '~> 3.5' + # Use Capistrano for deployment + gem 'capistrano-rails' + gem 'capistrano-rbenv', '~> 2.0' + gem 'capistrano-passenger' +end + +group :test do + gem "factory_bot", "~> 5.2" + gem 'factory_bot_rails', '~> 5.2' + gem 'shoulda-matchers', '~> 4.0' + gem 'database_cleaner' + gem 'coveralls', require: false + gem 'webmock' +end + +# Windows does not include zoneinfo files, so bundle the tzinfo-data gem +gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby] +gem "mimemagic", ">= 0.3.5" \ No newline at end of file diff --git a/README.md b/README.md index b12d07b9..634afa9c 100644 --- a/README.md +++ b/README.md @@ -4,19 +4,59 @@ The OpenTourBuilder API server provides a multi-tenant REST API for geographic t ## Requirements -- Ruby 2.4.1+ -- 5.2.x -- PostgreSQL 9.6.9 - - Plugins* - - pgcrypto - - uuid-ossp - - plpgsql +- rbenv +- Ruby 3.0.2 -\* Database plugins are enabled during the install process +## Install PostgreSQL client + +~~~bash +sudo apt install postgresql-client +~~~ + +## Install GDAL + +~~~bash +sudo apt install libpq-dev gdal-bin libgdal-dev +~~~ + +## Install Headless Chrome + +This is mostly taken from [this blog post](https://geekflare.com/install-chromium-ubuntu-centos/) and works on Ubuntu 20.04 + +~~~bash +sudo apt update +sudo apt install -y libappindicator1 fonts-liberation +wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb +sudo dpkg -i google-chrome*.deb +~~~ + +I always get some errors about missing/mis-matched dependencies. The following will fix that. + +~~~bash +sudo apt install -f +~~~ + +And try to install it again. + +~~~bash +sudo dpkg -i google-chrome*.deb +~~~ + +Assuming all went well, verify the install be running: + +~~~bash +google-chrome-stable -version +~~~ + +You should see something like: + +~~~bash +Google Chrome 91.0.4472.106 +~~~ ## Build Status -[![Build Status](https://travis-ci.com/ecds/otb-api-server.svg?branch=develop)](https://travis-ci.com/ecds/otb-api-server) +TODO: Add CircleCI ## Installation @@ -70,6 +110,7 @@ cap deploy staging We use the [Git-Flow](https://danielkummer.github.io/git-flow-cheatsheet/) branching model. Please submit pull requests against the develop branch. ### Code of conduct + [Code of Conduct](CODE_OF_CONDUCT.md) ## License diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb index f380099e..9a16d01c 100644 --- a/app/controllers/application_controller.rb +++ b/app/controllers/application_controller.rb @@ -5,4 +5,27 @@ class ApplicationController < ActionController::API include Response include ExceptionHandler include EcdsRailsAuthEngine::CurrentUser + if Rails.env == 'test' + include ActiveStorage::SetCurrent + end + + before_action :set_no_cache_control, only: [:index, :show] + + def set_no_cache_control + # Prevent the client from caching GET responses. + # If you, for example, look at a tour at https://battle-of-atlanta.opentour.site + # and your browser caches the API responses. then you go edit that tour + # at https://opentour.site/admin/battle-or-atlanta, your browser will use some of + # those previously cached responses. the problem is, those cached responses have a response header + # + # access-control-allow-origin: https://battle-of-atlanta.opentour.site + # + # But now your origin is https://opentour.site and the browser blocks the response + # and throws a cross origin error + response.headers['Cache-Control'] = 'no-cache, no-store, max-age=0, must-revalidate' + response.headers['Pragma'] = 'no-cache' + response.headers['Expires'] = '-1' + expires_now() + stale?(SecureRandom.hex(10)) + end end diff --git a/app/controllers/concerns/exception_handler.rb b/app/controllers/concerns/exception_handler.rb index 71fa04b5..7cd90aec 100644 --- a/app/controllers/concerns/exception_handler.rb +++ b/app/controllers/concerns/exception_handler.rb @@ -5,17 +5,17 @@ module ExceptionHandler # provides the more graceful `included` method extend ActiveSupport::Concern - included do - rescue_from ActiveRecord::RecordNotFound do |e| - json_response({ message: e.message }, :not_found) - end + # included do + # rescue_from ActiveRecord::RecordNotFound do |e| + # json_response({ message: e.message }, :not_found) + # end - rescue_from ActiveRecord::RecordInvalid do |e| - json_response({ message: e.message }, :unprocessable_entity) - end + # rescue_from ActiveRecord::RecordInvalid do |e| + # json_response({ message: e.message }, :unprocessable_entity) + # end - rescue_from ActionController::ParameterMissing do |e| - json_response({ message: e.message }, :unprocessable_entity) - end - end + # rescue_from ActionController::ParameterMissing do |e| + # json_response({ message: e.message }, :unprocessable_entity) + # end + # end end diff --git a/app/controllers/v1/tours_controller.rb b/app/controllers/v1/tours_controller.rb deleted file mode 100644 index 07fa72ee..00000000 --- a/app/controllers/v1/tours_controller.rb +++ /dev/null @@ -1,9 +0,0 @@ -# frozen_string_literal: true - -# app/controllers/v1/tours_controller.rb -# This is only for testing calls to versioned endpoints. -class V1::ToursController < ApplicationController - def index - json_response(message: Faker::Movies::HitchhikersGuideToTheGalaxy.quote) - end -end diff --git a/app/controllers/v3/authenticated_controller.rb b/app/controllers/v3/authenticated_controller.rb deleted file mode 100644 index 939c10d2..00000000 --- a/app/controllers/v3/authenticated_controller.rb +++ /dev/null @@ -1,14 +0,0 @@ -# frozen_string_literal: true - -# app/controllers/v3/authenticated_controller.rb -module V3 - class AuthenticatedController < V3Controller - # include RailsApiAuth::Authentication - - before_action :authenticate! - - def index - render json: { success: true } - end - end -end diff --git a/app/controllers/v3/flat_pages_controller.rb b/app/controllers/v3/flat_pages_controller.rb index 22d78d34..f1f736bc 100644 --- a/app/controllers/v3/flat_pages_controller.rb +++ b/app/controllers/v3/flat_pages_controller.rb @@ -1,59 +1,72 @@ # frozen_string_literal: true -class V3::FlatPagesController < V3Controller - before_action :set_flat_page, only: [:show, :update, :destroy] - authorize_resource +module V3 + class FlatPagesController < V3Controller + before_action :set_record, only: [:show, :update, :destroy] + #authorize_resource - # GET /v3/flat_pages - def index - @flat_pages = FlatPage.all - - render json: @flat_pages - end - - # GET /v3/flat_pages/1 - def show - render json: @flat_page - end + # GET /v3/records + def index + @records = if current_user.current_tenant_admin? + FlatPage.all + elsif current_user.tours.present? + current_user.tours.map { |tour| tour.flat_pages }.flatten.uniq + else + Tour.published.map { |tour| tour.flat_pages }.flatten.uniq + end + render json: @records + end - # POST /v3/flat_pages - def create - @flat_page = FlatPage.new(flat_page_params) + # POST /v3/records + def create + if @allowed + @record = FlatPage.new(record_params) - if @flat_page.save - render json: @flat_page, status: :created, location: "/#{Apartment::Tenant.current}/flat-pages/#{@flat_page.id}" - else - render json: @flat_page.errors, status: :unprocessable_entity + if @record.save + render json: @record, status: :created, location: "/#{Apartment::Tenant.current}/flat-pages/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end end - end - # PATCH/PUT /v3/flat_pages/1 - def update - if @flat_page.update(flat_page_params) - render json: @flat_page - else - render json: @flat_page.errors, status: :unprocessable_entity + # PATCH/PUT /v3/records/1 + def update + if @allowed + if @record.update(record_params) + render json: @record + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end end - end - # DELETE /v3/flat_pages/1 - def destroy - @flat_page.destroy - end + # DELETE /v3/records/1 - private - # Use callbacks to share common setup or constraints between actions. - def set_flat_page - @flat_page = FlatPage.find(params[:id]) - end - # Only allow a trusted parameter "white list" through. - def flat_page_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :title, :content, :tours - ] - ) - end + private + # Use callbacks to share common setup or constraints between actions. + def set_record + _record = FlatPage.find(params[:id]) + @record = _record.published || @allowed ? _record : FlatPage.new(id: params[:id]) + end + + # Only allow a trusted parameter "white list" through. + def record_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :title, :body, :tours + ] + ) + end + + def allowed? + @allowed = current_user&.current_tenant_admin? || current_user.tours&.any? { |tour| Tour.all.include?(tour) } + end + end end diff --git a/app/controllers/v3/geojson_tours_controller.rb b/app/controllers/v3/geojson_tours_controller.rb index 63ad37bc..3a00ff4e 100644 --- a/app/controllers/v3/geojson_tours_controller.rb +++ b/app/controllers/v3/geojson_tours_controller.rb @@ -1,30 +1,69 @@ # frozen_string_literal: true # -# +# Endpoint that returns a tour serialized as GeoJSON # module V3 - class GeojsonToursController < V3Controller - skip_authorization_check + class GeojsonToursController < ApplicationController + include ActionView::Helpers::SanitizeHelper + def show @tour = Tour.find(params[:id]) - render json: { type: 'FeatureCollection', features: @tour.stops.map { |s| feature(s) } }.to_json + # if @tour.published + geojosn = { + type: 'FeatureCollection', + crs: { + type: 'name', + properties: { + name: 'urn:ogc:def:crs:EPSG::4326' + } + }, + meta: meta_content, + features: @tour.tour_stops.map { |tour_stop| feature(tour_stop.position, tour_stop.stop) } + } + render json: geojosn.to_json + # render json: { type: 'FeatureCollection', meta: meta_content, features: @tour.tour_stops.map { |tour_stop| feature(tour_stop.position, tour_stop.stop) } }.to_json + # else + # head 401 + # end end private - def feature(stop) + def meta_content + { + title: @tour.title, + description: sanitize(@tour.description), + images: @tour.media.map do |m| + { + caption: m.caption, + full: m.files[:desktop], + thumb: m.files[:mobile] + } + end + } + end + + def feature(position, stop) + stop.media.map { |m| m.caption = nil if m.caption.blank? } { type: 'Feature', geometry: { - type: 'Point', - coordinates: [stop.lng.to_f, stop.lat.to_f] - }, - properties: { - title: stop.title, - description: stop.description, - images: stop.media.map(&:desktop).map { |m| "#{request.protocol}#{request.host}/#{m}" } - } + type: 'Point', + coordinates: [stop.lng.to_f, stop.lat.to_f] + }, + properties: { + title: stop.title, + description: sanitize(stop.description), + position: position, + images: stop.media.map do |m| + { + caption: m.caption, + full: m.files[:desktop], + thumb: m.files[:mobile] + } + end + } } end end diff --git a/app/controllers/v3/map_icons_controller.rb b/app/controllers/v3/map_icons_controller.rb new file mode 100644 index 00000000..e1f1e72f --- /dev/null +++ b/app/controllers/v3/map_icons_controller.rb @@ -0,0 +1,38 @@ +module V3 + class MapIconsController < V3Controller + + def index + render json: MapIcon.all + end + + def create + if crud_allowed? + @record = MapIcon.new(record_params) + if @record.save + render json: @record, status: :created + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end + end + + private + + # Only allow a trusted parameter "white list" through. + def record_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :base_sixty_four, :filename, :stop + ] + ) + end + + def set_record + _record = MapIcon.find(params[:id]) + @record = _record&.published || @allowed ? _record : MapIcon.new(id: params[:id]) + end + end +end diff --git a/app/controllers/v3/map_overlays_controller.rb b/app/controllers/v3/map_overlays_controller.rb new file mode 100644 index 00000000..0f71ed97 --- /dev/null +++ b/app/controllers/v3/map_overlays_controller.rb @@ -0,0 +1,33 @@ +module V3 + class MapOverlaysController < V3Controller + def create + if crud_allowed? + @record = MapOverlay.new(record_params) + if @record.save + render json: @record, status: :created + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end + end + + private + + # Only allow a trusted parameter "white list" through. + def record_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :south, :east, :north, :west, :base_sixty_four, :filename, :tour, :stop + ] + ) + end + + def set_record + _record = MapOverlay.find(params[:id]) + @record = _record&.published || @allowed ? _record : MapOverlay.new(id: params[:id]) + end + end +end diff --git a/app/controllers/v3/media_controller.rb b/app/controllers/v3/media_controller.rb index 0b81bf0e..a6b09925 100644 --- a/app/controllers/v3/media_controller.rb +++ b/app/controllers/v3/media_controller.rb @@ -3,75 +3,86 @@ # app/controllers/v3/media_controller.rb module V3 class MediaController < V3Controller - before_action :set_medium, only: [:show, :update, :destroy] - authorize_resource + # include Pagy::Backend + before_action :set_record, only: [:show, :update, :destroy, :file] + # GET /media def index # TODO: This ins not ideal, we use these `not_in_*` scopes to make the list of media avaliable to add # to a stop or tour. But the paramerter does not make sense when just looking at it. Needs clearer language. - @media = if params[:stop_id] - Medium.not_in_stop(params[:stop_id]).or(Medium.no_stops) - elsif params[:tour_id] - Medium.not_in_tour(params[:tour_id]).or(Medium.no_tours) - else + @media = if (current_user && current_user.current_tenant_admin?) Medium.all + else + Medium.all.map { |medium| medium if medium.published }.compact + end + # pagy, @media = pagy(@media) + # pagy_headers_merge(pagy) + if params[:page].present? + @media = @media.page params[:page] + self.set_pagination_header end render json: @media end - + # GET /media/1 def show - if @medium.published || current_user.id.present? - render json: @medium + if @record.published || current_user.id.present? + render json: @record else - head 401 + render json: { data: { id: 0, type: 'media', attributes: { title: '....' } } } end end # POST /media def create - @medium = Medium.new(medium_params) + if allowed? + @record = Medium.new(record_params) - if @medium.save - render json: @medium, status: :created, location: "/#{Apartment::Tenant.current}/media/#{@medium.id}" + if @record.save + render json: @record, status: :created, location: "/#{Apartment::Tenant.current}/media/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end else - render json: @medium.errors, status: :unprocessable_entity + render json: {}, status: :unauthorized end end - # PATCH/PUT /media/1 - def update - if @medium.update(update_medium_params) - render json: @medium - else - render json: @medium.errors, status: :unprocessable_entity - end + # Use callbacks to share common setup or constraints between actions. + def set_record + @record = Medium.find(params[:id]) end - # DELETE /media/1 - def destroy - @medium.destroy + # Only allow a trusted parameter "white list" through. + def record_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :title, :caption, :original_image, :stops, :tours, :video, :stop_id, :tour_id, :base_sixty_four, :video_provider, :embed, :filename + ] + ) end - # private - # Use callbacks to share common setup or constraints between actions. - def set_medium - @medium = Medium.find(params[:id]) - end + private - # Only allow a trusted parameter "white list" through. - def medium_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :title, :caption, :original_image, :stops, :tours, :video, :stop_id, :tour_id - ] - ) - end + def set_pagination_header(name=:media, options = {}) + scope = instance_variable_get("@#{name}") + request_params = request.query_parameters + url_without_params = request.original_url.slice(0..(request.original_url.index("?")-1)) unless request_params.empty? + url_without_params ||= request.original_url - def update_medium_params - self.medium_params.except(:original_image) - end + page = {} + page[:first] = 1 if scope.total_pages > 1 && !scope.first_page? + page[:last] = scope.total_pages if scope.total_pages > 1 && !scope.last_page? + page[:next] = scope.current_page + 1 unless scope.last_page? + page[:prev] = scope.current_page - 1 unless scope.first_page? + pagination_links = [] + page.each do |k, v| + new_request_hash= request_params.merge({:page => v}) + pagination_links << "<#{url_without_params}?#{new_request_hash.to_param}>; rel=\"#{k}\"" + end + headers["Link"] = pagination_links.join(", ") + end end end diff --git a/app/controllers/v3/modes_controller.rb b/app/controllers/v3/modes_controller.rb index 3dcc7531..678e29de 100644 --- a/app/controllers/v3/modes_controller.rb +++ b/app/controllers/v3/modes_controller.rb @@ -2,15 +2,14 @@ # app/controllers/v3/modes_controller.rb module V3 - class ModesController < V3Controller - # before_action :set_mode, only: [:show, :update, :destroy] - authorize_resource - + class ModesController < ApplicationController # GET /modes def index - @modes = Mode.all + json_response Mode.all + end - render json: @modes + def show + json_response Mode.find(params[:id]) end end end diff --git a/app/controllers/v3/stop_media_controller.rb b/app/controllers/v3/stop_media_controller.rb index 0e808ce1..3f42d292 100644 --- a/app/controllers/v3/stop_media_controller.rb +++ b/app/controllers/v3/stop_media_controller.rb @@ -1,67 +1,36 @@ -class V3::StopMediaController < V3Controller - before_action :set_stop_medium, only: [:show, :update, :destroy] - authorize_resource - - # GET /v3/stop_media - def index - @stop_media = if params[:stop_id] && params[:medium_id] - StopMedium.where(stop_id: params[:stop_id]).where(medium_id: params[:medium_id]).first || {} - else - StopMedium.all +module V3 + class StopMediaController < V3::TourRelationsController + # GET /v3/stop_media + def index + # @stop_media = if params[:tour_id] && params[:medium_id] + # StopMedium.where(tour_id: params[:tour_id]).where(medium_id: params[:medium_id]).first || {} + # else + # StopMedium.all + # end + + @stop_media = StopMedium.all + + unless current_user&.current_tenant_admin? || current_user.tours.present? + @stop_media = @stop_media.reject { |stop_medium| !stop_medium.stop.published } + end + + render json: @stop_media end - render json: @stop_media - end - - # GET /v3/stop_media/1 - def show - render json: @stop_medium, - include: [ - # 'media' - ] - end - - # POST /v3/stop_media - def create - @stop_medium = StopMedium.new(stop_medium_params) - if @stop_medium.save - render json: @stop_medium, status: :created, location: "/#{Apartment::Tenant.current}/stop-medium/#{@stop_medium.id}" - else - render json: @stop_medium.errors, status: :unprocessable_entity - end - end - - # PATCH/PUT /v3/stop_media/1 - def update - if @stop_medium.update(stop_medium_params) - render json: @stop_medium - else - render json: @stop_medium.errors, status: :unprocessable_entity - end + private + # Only allow a trusted parameter "white list" through. + def record_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :medium, :stop, :position + ] + ) + end + + def set_record + _record = StopMedium.find(params[:id]) + @record = _record&.published || @allowed ? _record : StopMedium.new(id: params[:id]) + end end - - # DELETE /v3/stop_media/1 - def destroy - @stop_medium.destroy - end - - private - # Use callbacks to share common setup or constraints between actions. - def set_stop_medium - @stop_medium = StopMedium.find(params[:id]) - end - - # Only allow a trusted parameter "white list" through. - def stop_medium_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :medium, :stop, :position, :medium_id, :stop_id - ] - ) - end - - def set_stop_medium - @stop_medium = StopMedium.find_by!(id: params[:id]) - end end diff --git a/app/controllers/v3/stops_controller.rb b/app/controllers/v3/stops_controller.rb index 6b3b359c..6d7b94f7 100644 --- a/app/controllers/v3/stops_controller.rb +++ b/app/controllers/v3/stops_controller.rb @@ -1,88 +1,73 @@ # frozen_string_literal: true # /app/controllers/v3/stops_controller.rb -# module V3 -class V3::StopsController < V3Controller - # before_action :set_tour - before_action :set_stop, only: [:show, :update, :destroy] - authorize_resource - +module V3 + class StopsController < V3::TourRelationsController # GET /stops - def index - @stops = if params[:tour_id] - Stop.not_in_tour(params[:tour_id]).or(Stop.no_tours) - elsif params[:slug] - # stop = StopSlug.find_by(slug: params[:slug]).stop - stop = Stop.by_slug_and_tour(params[:slug], params[:tour_id]) - else - Stop.all + def index + @records = if current_user.current_tenant_admin? + Stop.all + elsif current_user.tours.present? + current_user.tours.map { |tour| tour.stops }.flatten.uniq + else + Tour.published.map { |tour| tour.stops }.flatten.uniq + end + render json: @records end - render json: @stops, - include: [ - 'media', - 'stop_media' - ] - end - - # GET /stops/1 - def show - render json: @stop, - include: [ - 'media', - 'stop_media' - ] - end - # POST /stops - def create - @stop = Stop.new(stop_params) - if @stop.save - render json: @stop, status: :created, location: "/#{Apartment::Tenant.current}/#{@stop.id}" - else - render json: @stop.errors, status: :unprocessable_entity + # POST /stops + def create + if crud_allowed? + @record = Stop.new(stop_params) + if @record.save + render json: @record, status: :created, location: "/#{Apartment::Tenant.current}/#{@record.id}" + end + else + head 401 + end end - end - # PATCH/PUT /stops/1 - def update - if @stop.update(stop_params) - render json: @stop, location: "/#{Apartment::Tenant.current}/stops/#{@stop.id}" - else - render json: @stop.errors, status: :unprocessable_entity + # PATCH/PUT /stops/1 + def update + if crud_allowed? + if @record&.update(stop_params) + render json: @record, location: "/#{Apartment::Tenant.current}/stops/#{@record.id}" + end + else + head 401 + end end - end - - # DELETE /stops/1 - def destroy - @stop.destroy - end - - private - # Only allow a trusted parameter "white list" through. - def stop_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :title, :description, :lat, :lng, - :parking_lat, :parking_lng, :media, - :address, :tours, :direction_notes, - :meta_description - ] - ) + def destroy + if !crud_allowed? + head 401 + elsif crud_allowed? && @record.orphaned + @record.destroy + elsif crud_allowed? && !@record.orphaned + head 405 + end end - # Use callbacks to share common setup or constraints between actions. + private - def set_tour - @tour = Tour.find(params[:tour_id]) - end + # Only allow a trusted parameter "white list" through. + def stop_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :title, :description, :lat, :lng, + :parking_lat, :parking_lng, :media, + :address, :tours, :direction_notes, + :meta_description, :parking_address, + :icon_color, :map_icon + ] + ) + end - def set_stop - @stop = Stop.find(params[:id]) - end - - def set_tour_stop - @stop = @tour.stops.find_by!(id: params[:id]) if @tour - end + # Callbacks + def set_record + _record = Stop.find_by(id: params[:id]) + @record = _record&.published || @allowed ? _record : Stop.new(id: params[:id]) + end + end end diff --git a/app/controllers/v3/themes_controller.rb b/app/controllers/v3/themes_controller.rb index 33833f6a..c2739267 100644 --- a/app/controllers/v3/themes_controller.rb +++ b/app/controllers/v3/themes_controller.rb @@ -3,55 +3,35 @@ # app/controllers/v3/themes_controller.rb module V3 class ThemesController < V3Controller - before_action :set_theme, only: [:show, :update, :destroy] - authorize_resource - # GET /themes def index - @themes = Theme.all - - render json: @themes + render json: Theme.all end # GET /themes/1 def show - render json: @theme + render json: @record end # POST /themes def create - @theme = Theme.new(theme_params) - - if @theme.save - render json: @theme, status: :created, location: @theme - else - render json: @theme.errors, status: :unprocessable_entity - end + head 405 end # PATCH/PUT /themes/1 def update - if @theme.update(theme_params) - render json: @theme - else - render json: @theme.errors, status: :unprocessable_entity - end + head 405 end # DELETE /themes/1 def destroy - @theme.destroy + head 405 end private - # Use callbacks to share common setup or constraints between actions. - def set_theme - @theme = Theme.find(params[:id]) - end - # Only allow a trusted parameter "white list" through. - def theme_params - params.fetch(:theme, {}) + def set_record + @record = Theme.find(params[:id]) end -end + end end diff --git a/app/controllers/v3/tour_authors_controller.rb b/app/controllers/v3/tour_authors_controller.rb new file mode 100644 index 00000000..bd8e4b73 --- /dev/null +++ b/app/controllers/v3/tour_authors_controller.rb @@ -0,0 +1,44 @@ +module V3 + class TourAuthorsController < ApplicationController + before_action :set_tour_author, only: [:show] + + # GET /tour_authors + def index + if current_user&.current_tenant_admin? + render json: TourAuthor.all + else + head 401 + end + end + + # GET /tour_authors/1 + def show + if current_user&.current_tenant_admin? + render json: @tour_author + else + head 401 + end + end + + # POST /tour_authors + def create + head 405 + end + + # PATCH/PUT /tour_set_admins/1 + def update + head 405 + end + + # DELETE /tour_set_admins/1 + def destroy + head 405 + end + + private + # Use callbacks to share common setup or constraints between actions. + def set_tour_author + @tour_author = TourAuthor.find(params[:id]) + end + end +end diff --git a/app/controllers/v3/tour_collections_controller.rb b/app/controllers/v3/tour_collections_controller.rb deleted file mode 100644 index 8fbbd9ca..00000000 --- a/app/controllers/v3/tour_collections_controller.rb +++ /dev/null @@ -1,54 +0,0 @@ -# frozen_string_literal: true - - class V3::TourCollectionsController < V3Controller - before_action :set_tour_collection, only: [:show, :update, :destroy] - authorize_resource - - # GET /v3/tour_collections - def index - @tour_collections = TourCollection.all - - render json: @tour_collections - end - - # GET /v3/tour_collections/1 - def show - render json: @tour_collection - end - - # POST /v3/tour_collections - def create - @tour_collection = TourCollection.new(tour_collection_params) - - if @tour_collection.save - render json: @tour_collection, status: :created, location: @tour_collection - else - render json: @tour_collection.errors, status: :unprocessable_entity - end - end - - # PATCH/PUT /v3/tour_collections/1 - def update - if @tour_collection.update(tour_collection_params) - render json: @tour_collection - else - render json: @tour_collection.errors, status: :unprocessable_entity - end - end - - # DELETE /v3/tour_collections/1 - def destroy - @tour_collection.destroy - end - - private - # Use callbacks to share common setup or constraints between actions. - def set_tour_collection - @tour_collection = TourCollection.find(params[:id]) - end - - # Only allow a trusted parameter "white list" through. - def tour_collection_params - params.fetch(:tour_collection, {}) - end - end diff --git a/app/controllers/v3/tour_flat_pages_controller.rb b/app/controllers/v3/tour_flat_pages_controller.rb index 003be43e..4126f7c9 100644 --- a/app/controllers/v3/tour_flat_pages_controller.rb +++ b/app/controllers/v3/tour_flat_pages_controller.rb @@ -1,59 +1,74 @@ # frozen_string_literal: true -class V3::TourFlatPagesController < V3Controller - before_action :set_tour_flat_page, only: [:show, :update, :destroy] - authorize_resource +# /app/controllers/v3/tour_stops_controller.rb +module V3 + class TourFlatPagesController < V3Controller + # GET /stops + def index + @tour_flat_pages = TourFlatPage.all - # GET /v3/tour_flat_pages - def index - @tour_flat_pages = TourFlatPage.all + unless current_user&.current_tenant_admin? || current_user.tours.present? + @tour_flat_pages = @tour_flat_pages.reject { |tour_flat_page| !tour_flat_page.tour.published } + end - render json: @tour_flat_pages - end + render json: @tour_flat_pages + end - # GET /v3/tour_flat_pages/1 - def show - render json: @tour_flat_page - end + # GET /stops/1 + def show + if @record&.tour.published || allowed? + render json: @record + else + render json: { data: {} } + end + # render json: { data: {} } if @record.nil? + # render json: @record, include: ['stop'] + end - # POST /v3/tour_flat_pages - def create - @tour_flat_page = TourFlatPage.new(tour_flat_page_params) + # POST /stops + def create + # Not created via the API + head 405 + end - if @tour_flat_page.save - render json: @tour_flat_page, status: :created, location: "/#{Apartment::Tenant.current}/flat-pages/#{@tour_flat_page.id}" - else - render json: @tour_flat_page.errors, status: :unprocessable_entity + # PATCH/PUT /stops/1 + def update + if @allowed + if @record.update(tour_stop_params) + render json: @record, location: "/#{Apartment::Tenant.current}/tour_stops/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end end - end - # PATCH/PUT /v3/tour_flat_pages/1 - def update - if @tour_flat_page.update(tour_flat_page_params) - render json: @tour_flat_page - else - render json: @tour_flat_page.errors, status: :unprocessable_entity + # DELETE /stops/1 + def destroy + # Not deleted via the API + head 405 end - end - # DELETE /v3/tour_flat_pages/1 - def destroy - @tour_flat_page.destroy - end + private - private - # Use callbacks to share common setup or constraints between actions. - def set_tour_flat_page - @tour_flat_page = TourFlatPage.find(params[:id]) - end + # Only allow a trusted parameter "white list" through. + def tour_stop_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :stop, :tour, :position + ] + ) + end - # Only allow a trusted parameter "white list" through. - def tour_flat_page_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :tour, :flat_page, :position - ] - ) - end + def set_record + @record = TourFlatPage.find(params[:id]) + end + + def allowed? + @allowed = current_user&.current_tenant_admin? || current_user.tours&.any? { |tour| Tour.all.include?(tour) } + return @allowed + end + end end diff --git a/app/controllers/v3/tour_media_controller.rb b/app/controllers/v3/tour_media_controller.rb index caec3225..730e12b9 100644 --- a/app/controllers/v3/tour_media_controller.rb +++ b/app/controllers/v3/tour_media_controller.rb @@ -1,65 +1,36 @@ -class V3::TourMediaController < V3Controller - before_action :set_tour_medium, only: [:show, :update, :destroy] - authorize_resource - - # GET /v3/tour_media - def index - @tour_media = if params[:tour_id] && params[:medium_id] - TourMedium.where(tour_id: params[:tour_id]).where(medium_id: params[:medium_id]).first || {} - else - TourMedium.all - end - - render json: @tour_media - end - - # GET /v3/tour_media/1 - def show - render json: @tour_medium - end - - # POST /v3/tour_media - def create - @tour_medium = TourMedium.new(tour_medium_params) - - if @tour_medium.save - render json: @tour_medium, status: :created, location: @tour_medium - else - render json: @tour_medium.errors, status: :unprocessable_entity - end - end - - # PATCH/PUT /v3/tour_media/1 - def update - if @tour_medium.update(tour_medium_params) - render json: @tour_medium - else - render json: @tour_medium.errors, status: :unprocessable_entity +module V3 + class TourMediaController < V3::TourRelationsController + # GET /v3/tour_media + def index + # @tour_media = if params[:tour_id] && params[:medium_id] + # TourMedium.where(tour_id: params[:tour_id]).where(medium_id: params[:medium_id]).first || {} + # else + # TourMedium.all + # end + + @tour_media = TourMedium.all + + unless current_user&.current_tenant_admin? || current_user.tours.present? + @tour_media = @tour_media.reject { |tour_medium| !tour_medium.tour.published } + end + + render json: @tour_media end - end - # DELETE /v3/tour_media/1 - def destroy - @tour_medium.destroy + private + # Only allow a trusted parameter "white list" through. + def record_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :medium, :tour, :position + ] + ) + end + + def set_record + _record = TourMedium.find(params[:id]) + @record = _record&.published || @allowed ? _record : TourMedium.new(id: params[:id]) + end end - - private - # Use callbacks to share common setup or constraints between actions. - def set_tour_medium - @tour_medium = TourMedium.find(params[:id]) - end - - # Only allow a trusted parameter "white list" through. - def tour_medium_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :medium, :tour, :position - ] - ) - end - - def set_tour_medium - @tour_medium = TourMedium.find_by!(id: params[:id]) - end end diff --git a/app/controllers/v3/tour_modes_controller.rb b/app/controllers/v3/tour_modes_controller.rb deleted file mode 100644 index e1e7f7f5..00000000 --- a/app/controllers/v3/tour_modes_controller.rb +++ /dev/null @@ -1,15 +0,0 @@ -# frozen_string_literal: true - -# app/controllers/v3/tour_modes_controller.rb -module V3 - class TourModesController < V3Controller - authorize_resource - - # GET /tour_sets - def index - @tour_modes = TourMode.all - - render json: @tour_modes - end -end -end diff --git a/app/controllers/v3/tour_relations_controller.rb b/app/controllers/v3/tour_relations_controller.rb new file mode 100644 index 00000000..a4a94b86 --- /dev/null +++ b/app/controllers/v3/tour_relations_controller.rb @@ -0,0 +1,21 @@ +# frozen_string_literal: true + +# /app/controllers/v3/tour_relations_controller.rb +module V3 + class TourRelationsController < V3Controller + + def destroy + head 405 + end + + def allowed? + set_record if @record.nil? && params[:id].present? + @allowed = @record&.published || crud_allowed? + end + + def crud_allowed? + current_user&.current_tenant_admin? || + current_user.tours&.any? { |tour| Tour.all.include?(tour) } + end + end +end diff --git a/app/controllers/v3/tour_set_admins_controller.rb b/app/controllers/v3/tour_set_admins_controller.rb index 14d05fe7..33f9b2a1 100644 --- a/app/controllers/v3/tour_set_admins_controller.rb +++ b/app/controllers/v3/tour_set_admins_controller.rb @@ -2,55 +2,55 @@ module V3 class TourSetAdminsController < V3Controller - before_action :set_tour_set_admin, only: [:show, :update, :destroy] - authorize_resource - # GET /tour_set_admins def index - @tour_set_admins = TourSetAdmin.all - - render json: @tour_set_admins + if current_user&.super || current_user&.current_tenant_admin? + render json: TourSetAdmin.all + else + head 401 + end end # GET /tour_set_admins/1 def show - render json: @tour_set_admin + head 405 + # if current_user&.super || current_user&.current_tenant_admin? + # render json: @record + # else + # head 401 + # end end # POST /tour_set_admins def create - @tour_set_admin = TourSetAdmin.new(tour_set_admin_params) - - if @tour_set_admin.save - render json: @tour_set_admin, status: :created, location: @tour_set_admin - else - render json: @tour_set_admin.errors, status: :unprocessable_entity - end + head 405 + # @record = TourSetAdmin.new(tour_set_admin_params) + + # if @record.save + # render json: @record, status: :created, location: @record + # else + # render json: serialize_errors, status: :unprocessable_entity + # end end # PATCH/PUT /tour_set_admins/1 def update - if @tour_set_admin.update(tour_set_admin_params) - render json: @tour_set_admin - else - render json: @tour_set_admin.errors, status: :unprocessable_entity - end + head 405 end # DELETE /tour_set_admins/1 def destroy - @tour_set_admin.destroy + head 405 end private - # Use callbacks to share common setup or constraints between actions. - def set_tour_set_admin - @tour_set_admin = TourSetAdmin.find(params[:id]) - end - # Only allow a trusted parameter "white list" through. - def tour_set_admin_params - params.fetch(:tour_set_admin, {}) + # def tour_set_admin_params + # params.fetch(:tour_set_admin, {}) + # end + + def set_record + @record = TourSetAdmin.find(params[:id]) end end end diff --git a/app/controllers/v3/tour_sets_controller.rb b/app/controllers/v3/tour_sets_controller.rb index cfbbb149..600c7f65 100644 --- a/app/controllers/v3/tour_sets_controller.rb +++ b/app/controllers/v3/tour_sets_controller.rb @@ -4,76 +4,100 @@ # app/controllers/v3/tour_sets.rb module V3 class TourSetsController < V3Controller - before_action :set_tour_set, only: [:show, :update, :destroy] - authorize_resource - # GET /tour_sets def index - @tour_sets = [] + @records = [] if params[:subdir] && params[:subdir] != 'public' - @tour_sets = TourSet.find_by(subdir: params[:subdir]) - elsif current_user.super - @tour_sets = TourSet.all - elsif current_user.id.present? - @tour_sets = current_user.tour_sets + @records = TourSet.where(subdir: params[:subdir]) + if !@records.first&.published_tours&.empty? || current_user&.tour_sets.include?(@records.first) || current_user&.super + render json: @records + else + render json: TourSet.none + end + return + elsif current_user&.tour_sets.present? && !current_user.super + @records = published.concat(current_user.tour_sets).uniq else - #TourSet.all.reject {|ts| p ts.tours.empty?} - @tour_sets = TourSet.all.reject { |ts| ts.published_tours.empty? } + @records = TourSet.all end - - if current_user.current_tenant_admin? || current_user.super - render json: @tour_sets, include: [ 'admins' ] + + if current_user.tour_sets.present? || current_user.super + render json: @records, include: [ 'admins' ] else - render json: @tour_sets + if current_user&.tour_sets.empty? + @records = published + end + render json: @records end end # GET /tour_sets/1 def show - render json: @tour_set + if @allowed + render json: @record + else + render json: { data: { id: 0, type: 'tour_sets', attributes: { name: '....' } } } + end end # POST /tour_sets def create - @tour_set = TourSet.new(tour_set_params) + if crud_allowed? + @record = TourSet.new(record_params) - if @tour_set.save - # render json: @tour_set, status: :created, location: @tour_set - response = render json: @tour_set, status: :created, location: "/#{Apartment::Tenant.current}/#{@tour_set.id}" - return response + if @record.save + render json: @record, status: :created, location: "/#{Apartment::Tenant.current}/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end else - render json: @tour_set.errors, status: :unprocessable_entity + head 401 end end # PATCH/PUT /tour_sets/1 def update - if @tour_set.update(tour_set_params) - # render json: @tour_set - head :no_content + if crud_allowed? + if @record.update(record_params) + render json: @record + else + render json: serialize_errors, status: :unprocessable_entity + end else - render json: @tour_set.errors, status: :unprocessable_entity + head 401 end end - # DELETE /tour_sets/1 - def destroy - @tour_set.destroy - end - private # Use callbacks to share common setup or constraints between actions. - def set_tour_set - @tour_set = TourSet.find(params[:id]) + def set_record + @record = TourSet.find(params[:id]) + end + + def allowed? + set_record if @record.nil? && params[:id].present? + @allowed = if @record.nil? + crud_allowed? + else + current_user&.current_tenant_admin? || @record.published_tours.present? || current_user.tour_sets.include?(@record) + end + end + + def crud_allowed? + current_user&.super || (current_user.present? && @record&.in?(current_user.tour_sets)) + end + + def published + TourSet.all.reject { |tour_set| tour_set.published_tours.empty? } end # Only allow a trusted parameter "white list" through. - def tour_set_params + def record_params ActiveModelSerializers::Deserialization .jsonapi_parse( params, only: [ - :name, :tours, :admins + :name, :tours, :admins, :base_sixty_four, :logo_title ] ) end diff --git a/app/controllers/v3/tour_stops_controller.rb b/app/controllers/v3/tour_stops_controller.rb index 15e2a491..ecad029f 100644 --- a/app/controllers/v3/tour_stops_controller.rb +++ b/app/controllers/v3/tour_stops_controller.rb @@ -1,68 +1,86 @@ # frozen_string_literal: true # /app/controllers/v3/tour_stops_controller.rb -class V3::TourStopsController < V3Controller - before_action :set_tour_stop, only: [:show, :update, :destroy] - authorize_resource +module V3 + class TourStopsController < V3Controller + # GET /stops + def index + @records = if params[:fastboot] == 'true' + nil + elsif params[:tour] && params[:slug] + tour = Tour.find(params[:tour]) + if tour.published || allowed? + stop = Stop.by_slug_and_tour(params[:slug], params[:tour]).first + TourStop.find_by(tour: Tour.find(params[:tour]), stop: stop) + else + {} + end + elsif current_user.current_tenant_admin? + TourStop.all + else + Tour.published.map { |tour| tour.tour_stops }.flatten.uniq + end + if @records.nil? + render json: { data: { type: 'tour_stops', id: 0 } } + else + render json: @records, include: ['stop'] + end + end - # GET /stops - def index - @tour_stops = if params[:tour_id] && params[:stop_id] - TourStop.where(tour: Tour.find(params[:tour_id])).where(stop: Stop.find(params[:stop_id])).first || {} - elsif params[:tour] && params[:slug] - # stop = StopSlug.find_by(slug: params[:slug]) - stop = Stop.by_slug_and_tour(params[:slug], params[:tour]).first - # TourStop.where(tour: Tour.find(params[:tour])).where(stop: stop).first - TourStop.find_by(tour: Tour.find(params[:tour]), stop: stop) - else - TourStop.all + # GET /stops/1 + def show + if @record&.tour.published || allowed? + render json: @record + else + render json: { data: {} } + end + # render json: { data: {} } if @record.nil? + # render json: @record, include: ['stop'] end - render json: @tour_stops, include: ['stop'] - end - # GET /stops/1 - def show - render json: @tour_stop, include: ['stop'] - end + # POST /stops + def create + # Not created via the API + head 405 + end - # POST /stops - def create - @tour_stop = TourStop.new(tour_stop_params) - if @tour_stop.save - render json: @tour_stop, status: :created, location: @tour_stop - else - render json: @tour_stop.errors, status: :unprocessable_entity + # PATCH/PUT /stops/1 + def update + if @allowed + if @record.update(tour_stop_params) + render json: @record, location: "/#{Apartment::Tenant.current}/tour_stops/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end end - end - # PATCH/PUT /stops/1 - def update - if @tour_stop.update(tour_stop_params) - # render json: @stop - head :no_content - else - render json: @tour_stop.errors, status: :unprocessable_entity + # DELETE /stops/1 + def destroy + head 405 end - end - # DELETE /stops/1 - def destroy - @tour_stop.destroy - end + private - private + # Only allow a trusted parameter "white list" through. + def tour_stop_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :stop, :tour, :position + ] + ) + end - # Only allow a trusted parameter "white list" through. - def tour_stop_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :stop, :tour, :position - ] - ) - end + def set_record + @record = TourStop.find(params[:id]) + end - def set_tour_stop - @tour_stop = TourStop.find_by!(id: params[:id]) - end + def allowed? + @allowed = current_user&.current_tenant_admin? || current_user.tours&.any? { |tour| Tour.all.include?(tour) } + return @allowed + end + end end diff --git a/app/controllers/v3/tours_controller.rb b/app/controllers/v3/tours_controller.rb index 075094f8..12abeec4 100644 --- a/app/controllers/v3/tours_controller.rb +++ b/app/controllers/v3/tours_controller.rb @@ -1,111 +1,102 @@ # frozen_string_literal: true # app/controllers/v3/tours_controller.rb -# module V3 -class V3::ToursController < V3Controller - before_action :set_tour, only: [:show, :update, :destroy] - # authorize_resource - load_and_authorize_resource - - # GET /tours - def index - @tours = if (params[:slug]) - tour = Slug.find_by(slug: params[:slug]).tour - if tour.published || current_user.current_tenant_admin? - tour +module V3 + class ToursController < V3Controller + # GET /tours + def index + @records = if (params[:slug]) + @record = Slug.find_by(slug: params[:slug]).tour + if @record.published || crud_allowed? + @record + else + nil + end + elsif (current_user && current_user.current_tenant_admin?) + Tour.all + elsif (current_user && current_user.id) + (current_user.tours + Tour.published).uniq else - nil + Tour.published + end + if @records.nil? + render json: { data: { id: 0, type: 'tours', attributes: { title: '....' } } } + else + render json: @records, each_serializer: V3::TourBaseSerializer end - elsif (current_user.current_tenant_admin?) - Tour.all - elsif (current_user.id) - current_user.tours - else - Tour.published - end - - if @tours.nil? - render json: {error: "not-found"}.to_json, status: 404 - else - render json: @tours, - include: [ - 'tour_stops', - 'stops', - 'stops.media', - 'stops.stop_media', - 'mode', - 'modes', - 'theme', - 'media', - 'tour_media', - 'flat_pages', - 'tour_flat_pages' - ] end - end + # GET /tours/1 + def show + request_loc = if request.env['ipinfo'].respond_to?('longitude') + { centerLng: request.env['ipinfo'].longitude, centerLat: request.env['ipinfo'].latitude } + else + { centerLng: -84.38979, centerLat: 33.75432 } + end - # GET /tours/1 - def show - render json: @tour, - include: [ - 'tour_stops', - 'stops', - 'stops.media', - 'stops.stop_media', - 'mode', - 'modes', - 'theme', - 'media', - 'tour_media', - 'flat_pages', - 'tour_flat_pages' - ] - end + if @record&.published || crud_allowed? + render json: @record, loc: request_loc + else + render json: { data: { id: 0, type: 'tours', attributes: { title: '....' } } } + end + end - # POST /tours - def create - @tour = Tour.new(tour_params) - if @tour.save - response = render json: @tour, status: :created, location: "/#{Apartment::Tenant.current}/tours/#{@tour.id}" - return response - else - render json: @tour.errors, status: :unprocessable_entity + # POST /tours + def create + if crud_allowed? + @record = Tour.new(tour_params) + if @record.save + render json: @record, status: :created, location: "/#{Apartment::Tenant.current}/tours/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end end - end - # PATCH/PUT /tours/1 - def update - if @tour.update(tour_params) - render json: @tour, location: "/#{Apartment::Tenant.current}/tours/#{@tour.id}" - else - render json: @tour.errors, status: :unprocessable_entity + # PATCH/PUT /tours/1 + def update + if crud_allowed? + if @record.update(tour_params) + render json: @record, location: "/#{Apartment::Tenant.current}/tours/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end + else + head 401 + end end - end - # DELETE /tours/1 - def destroy - @tour.destroy - end + private + # Only allow a trusted parameter "white list" through. + def tour_params + ActiveModelSerializers::Deserialization + .jsonapi_parse( + params, only: [ + :title, :description, + :is_geo, :modes, :published, :theme_id, + :mode, :meta_description, :stops, + :media, :users, :flat_pages, :map_type, + :theme, :use_directions, :default_lng, + :link_address, :link_text, :restrict_bounds, + :restrict_bounds_to_overlay, :blank_map + ] + ) + end - private + def set_record + _record = Tour.find(params[:id]) + @record = _record&.published || @allowed ? _record : Tour.new(id: params[:id]) + end - # Use callbacks to share common setup or constraints between actions. - def set_tour - @tour = Tour.includes(:tour_stops).find(params[:id]) - end + # def allowed? + # @allowed = crud_allowed? || + # end - # Only allow a trusted parameter "white list" through. - def tour_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :title, :description, - :is_geo, :modes, :published, :theme_id, - :mode, :meta_description, :stops, - :media, :authors, :flat_pages, :map_type, - :theme - ] - ) - end -end \ No newline at end of file + def crud_allowed? + set_record if @record.nil? && params[:id].present? + current_user&.current_tenant_admin? || current_user.tours.include?(@record) + end + end +end diff --git a/app/controllers/v3/users_controller.rb b/app/controllers/v3/users_controller.rb index dae270c8..1a7bbd3c 100644 --- a/app/controllers/v3/users_controller.rb +++ b/app/controllers/v3/users_controller.rb @@ -1,98 +1,88 @@ # frozen_string_literal: true # app/controllers/v3/users_controller.rb +# +# Endpoints for User Model +# module V3 class UsersController < V3Controller - before_action :set_user, only: [:show, :update, :destroy] before_action :authenticate!, only: :me - authorize_resource # GET /users - # def index - # @users = User.all - - # render json: @users - # end def index - if current_user.present? && params['me'] - render json: current_user, include: ['tours', 'tour_sets', 'login'] - elsif current_user.current_tenant_admin? - render json: User.all - else - render json: { message: 'You are not autorized to to view this resource.' }.to_json, status: 401 + if current_user.present? + if params['me'] + render json: current_user + elsif current_user.current_tenant_admin? + render json: User.all + else + render json: { data: [] } + end end end # GET /users/1 def show - if current_user == @user || current_user.super - render json: @user + if current_user == @record || current_user.super + render json: @record, include_tours: true else render json: { message: 'You are not autorized to to view this resource.' }.to_json, status: 401 end end + # TODO: Is this endpoint ever used? # POST /users def create - @user = User.new(user_params) + if current_user&.super + @record = User.new(user_params) - if @user.save && user.create_login(login_params) - render json: @user, status: :created, location: @user + if @record.save + render json: @record, status: :created, location: "/#{Apartment::Tenant.current}/users/#{@record.id}" + else + render json: serialize_errors, status: :unprocessable_entity + end else - render json: @user.errors, status: :unprocessable_entity + head 401 end end # PATCH/PUT /users/1 def update - if @user.update(user_params) - render json: @user + if current_user&.super || current_user == @record + if @record.update(user_params) + render json: @record + else + render json: serialize_errors, status: :unprocessable_entity + end else - render json: @user.errors, status: :unprocessable_entity + head 401 end end # DELETE /users/1 def destroy - @user.destroy - end - - def me - user = @current_login.user - if user.nil? - render json: 'Invalid api token', status: :foo + if current_user&.super + @record.destroy else - render json: user + head 401 end end private - - # Use callbacks to share common setup or constraints between actions. - def set_user - @user = User.find(params[:id]) - end - # Only allow a trusted parameter "white list" through. def user_params ActiveModelSerializers::Deserialization .jsonapi_parse( params, only: [ - :displayname, :identification, :password, + :display_name, :identification, :password, :password_confirmation, :uid, :tour_sets, - :tours, :super + :tours, :super, :email, :terms_accepted ] ) end - def login_params - ActiveModelSerializers::Deserialization - .jsonapi_parse( - params, only: [ - :identification, :password, - :password_confirmation, :uid - ] - ) + def set_record + @record = User.find(params[:id]) end end end diff --git a/app/controllers/v3_controller.rb b/app/controllers/v3_controller.rb index 10c4bd48..d87c08df 100644 --- a/app/controllers/v3_controller.rb +++ b/app/controllers/v3_controller.rb @@ -1,9 +1,63 @@ # frozen_string_literal: true class V3Controller < ApplicationController - check_authorization + include EcdsRailsAuthEngine::CurrentUser + before_action :allowed?, only: [:show, :create, :update, :destroy] + before_action :set_record, only: [:show, :update, :destroy] - rescue_from CanCan::AccessDenied do |exception| - head 401 + # GET //1 + def show + render json: @record end + + # POST /v3/tour_media + def create + render json: {}, status: :unauthorized + end + + # PATCH/PUT /media/1 + def update + if crud_allowed? + if @record.update(record_params) + render json: @record + else + render json: serialize_errors, status: :unprocessable_entity + end + else + render json: {}, status: :unauthorized + end + end + + def destroy + if crud_allowed? + @record.destroy + else + render json: {}, status: :unauthorized + end + end + + def serialize_errors + errors = [] + if @record&.errors + @record.errors.full_messages.each do |error| + errors.push({ + detail: error, + source: { + pointer: 'data/attributes' + } + }) + end + end + { errors: errors } + end + + private + + def allowed? + @allowed = @record&.published || crud_allowed? + end + + def crud_allowed? + current_user&.current_tenant_admin? || current_user.tours.present? + end end diff --git a/app/models/ability.rb b/app/models/ability.rb deleted file mode 100644 index 1817308e..00000000 --- a/app/models/ability.rb +++ /dev/null @@ -1,45 +0,0 @@ -# frozen_string_literal: true - -class Ability - include CanCan::Ability - - def initialize(user) - can :read, Tour, published: true - can :read, Slug, published: true - can :read, Theme - can :read, Mode - can :read, TourMode - can :read, Stop - can :read, TourStop - can :read, TourMedium - can :read, StopMedium - can :read, Medium - can :read, FlatPage - can [:read], TourSet - return unless user.id.present? - can [:read, :edit, :update], Tour - can [:manage], TourMedium - can [:manage], Medium - can [:manage], TourStop - can [:manage], Stop - can [:manage], StopMedium - can [:manage], FlatPage - can [:manage], TourFlatPage - can [:manage], Mode - can [:manage], TourMode - can [:read], User - return unless user.current_tenant_admin? - can [:manage], Tour - can [:manage], TourSetAdmin - can [:read, :edit, :update], TourSet - can [:read, :edit, :update], User - # can :manage, Stop - # can :manage, TourStop - # can :manage, TourMedium - # can :manage, StopMedium - # can :manage, Medium - # can :manage, FlatPage - return unless user.super - can :manage, :all - end -end diff --git a/app/models/concerns/video_props.rb b/app/models/concerns/video_props.rb index bc82c22c..619b72e6 100644 --- a/app/models/concerns/video_props.rb +++ b/app/models/concerns/video_props.rb @@ -1,79 +1,83 @@ # frozen_string_literal: true +require 'open-uri' + module VideoProps extend ActiveSupport::Concern def self.props(medium) return if medium.video.nil? - if self.is_vimeo(medium) - medium.provider = 'vimeo' - medium.video = vimeo_id(medium) + case medium.video_provider + when 'keiner' + nil + when 'vimeo' metadata = HTTParty.get("https://vimeo.com/api/oembed.json?url=https%3A//vimeo.com/video/#{medium.video}") medium.title = metadata['title'] medium.caption = metadata['description'] - image = metadata['thumbnail_url'] - medium.remote_original_image_url = metadata['thumbnail_url'] - medium.embed = "" - - elsif self.is_youtube(medium) - medium.provider = 'youtube' - medium.video = youtube_id(medium) + medium.embed = "//player.vimeo.com/video/#{medium.video}" + thumbnail_width = metadata['thumbnail_width'] + thumbnail_height = metadata['thumbnail_height'] + scale_by = 1000 / thumbnail_width + thumbnail_url = "#{metadata['thumbnail_url'].split('_')[0]}_#{thumbnail_width * scale_by}x#{thumbnail_height * scale_by}" + downloaded_image = URI.open(thumbnail_url) + when 'youtube' begin metadata = Yt::Video.new(id: medium.video) medium.title = metadata.title medium.caption = metadata.description - medium.remote_original_image_url = "https://img.youtube.com/vi/#{medium.video}/0.jpg" - medium.embed = %Q[" - # end - # medium.save - # end - - # - - def self.is_youtube(medium) - if medium.provider == 'vimeo' - return false end - # FIXME Youtube is always going to return 200 - (medium.video.include? 'youtube.com') || (medium.video.include? 'youtu.be') || (!medium.video.include?('iframe') && HTTParty.get("https://www.youtube.com/watch?v=#{medium.video}").code == 200) - end - def self.is_vimeo(medium) - if medium.provider == 'youtube' - return false - end - (medium.video.include? 'vimeo') || (!medium.video.include?('iframe') && HTTParty.get("https://vimeo.com/#{medium.video}").code == 200) - end + return if downloaded_image.nil? - def self.youtube_id(medium) - if medium.video.include? 'iframe' - YouTubeRails.extract_video_id(Nokogiri::HTML(medium.video).xpath('//iframe')[0]['src']) - elsif medium.video.include?('youtu') - YouTubeRails.extract_video_id(medium.video) - else - medium.video - end + medium.filename = "#{medium.video}.jpg" + medium.base_sixty_four = encode_image(downloaded_image) + medium.attach_file unless medium.file.attached? end - def self.vimeo_id(medium) - if medium.video.include? 'iframe' - Nokogiri::HTML(medium.video).xpath('//iframe')[0]['src'].split('/')[-1] - else - /\d{7,10}/.match(medium.video)[0] - # /https?:\/\/{?:www\.}?vimeo.com\/{?:channels\/(?:\w+\/)?|groups\/([^/]*)\/videos\/|album\/(\d+)\/video\/|)(\d+)(?:$|\/|\?)/.match(medium.video)[0] + def self.encode_image(downloaded_image) + begin + if downloaded_image.is_a? StringIO + base_sixty_four = Base64.encode64(downloaded_image.read) + else + base_sixty_four = Base64.encode64(downloaded_image.open.read) + downloaded_image.unlink + end + rescue NoMethodError + base_sixty_four = Base64.encode64(downloaded_image) end + base_sixty_four end end diff --git a/app/models/flat_page.rb b/app/models/flat_page.rb index 1eb2e524..cba879a5 100644 --- a/app/models/flat_page.rb +++ b/app/models/flat_page.rb @@ -4,4 +4,16 @@ class FlatPage < ApplicationRecord has_many :tour_flat_pages has_many :tours, through: :tour_flat_pages validates :title, presence: true + + def slug + title ? title.parameterize_intl : '' + end + + def orphaned + tours.empty? + end + + def published + tours.any? { |tour| tour.published } + end end diff --git a/app/models/map_icon.rb b/app/models/map_icon.rb new file mode 100644 index 00000000..e5376c03 --- /dev/null +++ b/app/models/map_icon.rb @@ -0,0 +1,19 @@ +class MapIcon < MediumBaseRecord + validate :check_dimensions + + has_one :stop + + def published + stop.published + end + + def check_dimensions + return if base_sixty_four.nil? + + file = MiniMagick::Image.read(Base64.decode64(base_sixty_four)) + + if file[:height] > 80 || file[:width] > 80 + errors.add(:base, 'Icons should be no bigger that 80 by 80 pixels') + end + end +end diff --git a/app/models/map_overlay.rb b/app/models/map_overlay.rb new file mode 100644 index 00000000..7fca50c3 --- /dev/null +++ b/app/models/map_overlay.rb @@ -0,0 +1,26 @@ +# frozen_string_literal: true + +# +# Model calss for map overlays. +# +class MapOverlay < MediumBaseRecord + before_create :set_initial_bounds + + belongs_to :tour, optional: true + belongs_to :stop, optional: true + + def published + tour.published + end + + def set_initial_bounds + return if tour&.bounds.nil? || tour&.stop_count < 2 + + if tour + self.south = self.tour.bounds[:south] + self.north = self.tour.bounds[:north] + self.east = self.tour.bounds[:east] + self.west = self.tour.bounds[:west] + end + end +end diff --git a/app/models/medium.rb b/app/models/medium.rb index 9c43678d..48cdca2c 100644 --- a/app/models/medium.rb +++ b/app/models/medium.rb @@ -1,84 +1,75 @@ # frozen_string_literal: true # Model for media associated with stops. -class Medium < ApplicationRecord +class Medium < MediumBaseRecord include VideoProps include Rails.application.routes.url_helpers - before_validation :props - mount_base64_uploader :original_image, MediumUploader + before_create :props + before_save :add_widths + before_update :replace_video + + # has_one_attached :file do |attachable| + # attachable.variant :mobile, resize: '200x200' + # attachable.variant :tablet, resize: '300x300' + # attachable.variant :desktop, resize: '750x750' + # end + + # mount_base64_uploader :original_image, MediumUploader has_many :stop_media has_many :stops, through: :stop_media has_many :tour_media has_many :tours, through: :tour_media - validates_presence_of :original_image + enum video_provider: { keiner: 0, vimeo: 1, youtube: 2, soundcloud: 3 } attr_accessor :insecure - # TODO: This is not ideal, we use these `not_in_*` scopes to make the list of media avaliable to add - # to a stop or tour. But the paramerter does not make sense when just looking at it. Needs clearer language. - scope :not_in_stop, lambda { |stop_id| includes(:stop_media).where.not(stop_media: { stop_id: stop_id }) } - scope :not_in_tour, lambda { |tour_id| includes(:tour_media).where.not(tour_media: { tour_id: tour_id }) } - scope :no_stops, lambda { includes(:stop_media).where(stop_media: { stop_id: nil }) } - scope :no_tours, lambda { includes(:tour_media).where(tour_media: { tour_id: nil }) } - scope :orphan, -> { no_tours.no_stops } - scope :published_by_tour, lambda { includes(:tours).where(tours: { published: true }) } - scope :published_by_stop, -> { joins(:stops).merge(Stop.published) } - - def props return if self.video.nil? || self.video.empty? VideoProps.props(self) end - def desktop - original_image.desktop.url - end - - def tablet - original_image.tablet.url - end - - def mobile - original_image.mobile_list_thumb.url - end - - def mobile_thumb - original_image.mobile_list_thumb.url - end - def published - # This works and is shorter, but I think the longer way is more readable/clear. - # tours.published.present? || stops { |s| s.tours.published }.present? - tours.collect(&:published).include?(true) || stops.map {|s| s.tours.collect(&:published)}.flatten.include?(true) + tours.any? { |tour| tour.published } || stops.any? { |stop| stop.published } end - def srcset - "#{ENV['BASE_URL']}#{self.mobile} #{mobile_width}w, \ - #{ENV['BASE_URL']}#{self.tablet} #{tablet_width}w, \ - #{ENV['BASE_URL']}#{self.desktop} #{desktop_width}w" - end + def files + return nil if !self.file.attached? - def srcset_sizes - "(max-width: 680px) #{mobile_width}px, (max-width: 880px) #{tablet_width}px, #{desktop_width}px" + if file.content_type.include?('gif') + return { + lqip: file.variant(resize_to_limit: [50, 50], coalesce: true, layers: 'Optimize', deconstruct: true, loader: { page: nil }).processed.url, + mobile: file.variant(resize_to_limit: [300, 300], coalesce: true, layers: 'Optimize', deconstruct: true, loader: { page: nil }).processed.url, + tablet: file.variant(resize_to_limit: [400, 400], coalesce: true, layers: 'Optimize', deconstruct: true, loader: { page: nil }).processed.url, + desktop: file.variant(resize_to_limit: [750, 750], coalesce: true, layers: 'Optimize', deconstruct: true, loader: { page: nil }).processed.url + } + end + { + lqip: file.variant(resize_to_limit: [5, 5]).processed.url, + mobile: file.variant(resize_to_limit: [300, 300]).processed.url, + tablet: file.variant(resize_to_limit: [400, 400]).processed.url, + desktop: file.variant(resize_to_limit: [750, 750]).processed.url + } end - def insecure - "#{ENV['INSECURE_IMAGE_BASE_URL']}#{self.desktop}" + def orphaned + tours.empty? && stops.empty? end - def base64 - if self.original_image.file && File.file?(self.original_image.file.path) - return Base64.encode64(self.original_image.file.read) + def replace_video + if video.present? && base_sixty_four.present? + attach_file end - nil end - # private + def add_widths + return unless file.attached? - # def has_image - # original_image.present? || remote_original_image_url.present? - # end -end \ No newline at end of file + self.lqip_width = MiniMagick::Image.open(files[:lqip])[:width] || 50 + self.mobile_width = MiniMagick::Image.open(files[:mobile])[:width] || 300 + self.tablet_width = MiniMagick::Image.open(files[:tablet])[:width] || 400 + self.desktop_width = MiniMagick::Image.open(files[:desktop])[:width] || 750 + end +end diff --git a/app/models/medium_base_record.rb b/app/models/medium_base_record.rb new file mode 100755 index 00000000..64f0e8c9 --- /dev/null +++ b/app/models/medium_base_record.rb @@ -0,0 +1,91 @@ +# frozen_string_literal: true + +# Base class for models. +class MediumBaseRecord < ApplicationRecord + self.abstract_class = true + validate :check_content_type + + before_create :attach_file + before_destroy :purge + + validates_presence_of :filename + + # has_one_attached "#{Apartment::Tenant.current.underscore}_file" + has_one_attached 'file' + + attr_accessor :content_type + + # def image_url + # return nil unless file.attached? + + # file.url + # end + + def tmp_file_path + return Rails.root.join('public', 'storage', 'tmp', filename) if self.filename + nil + end + + def original_image_url + file.url + end + + # + # Create and attach file from Base64 string. + # + # This should only be called once when a new medium obeject is created via the API + # It assumes + # + # Some code taken from https://github.com/rootstrap/active-storage-base64/blob/v1.2.0/lib/active_storage_support/base64_attach.rb#L17-L32 + # + # + def attach_file + return if base_sixty_four.nil? + + # file.blob.delete if file.attached? + + self.parse_base64 + File.open(tmp_file_path, 'wb') do |f| + f.write(Base64.decode64(base_sixty_four)) + end + + self.file.attach( + io: File.open(tmp_file_path), + filename: filename, + content_type: self.content_type + ) + + self.base_sixty_four = nil + end + + def remove_tmp_file + File.delete(tmp_file_path) if File.exists?(tmp_file_path) + end + + def purge + remove_tmp_file + file.blob.delete if file.attached? + end + + def check_content_type + return if base_sixty_four.nil? + + self.parse_base64 + + if self.content_type.include?('jp2') + errors.add(:base, 'JPEG 2000 fils are not supported. Plese convert the image to a reqular JPEG or WebP format.') + end + end + + private + + def parse_base64 + if base_sixty_four.include?('data:') + headers, self.base_sixty_four = base_sixty_four.split(',') + headers =~ /^data:(.*?)$/ + self.content_type = Regexp.last_match(1).split(';base64').first + else + self.content_type = 'image/jpeg' + end + end +end diff --git a/app/models/slug.rb b/app/models/slug.rb index 9fbc0d78..f4d062c9 100644 --- a/app/models/slug.rb +++ b/app/models/slug.rb @@ -4,9 +4,9 @@ class Slug < ApplicationRecord belongs_to :tour validates :slug, uniqueness: true - attr_accessor :published + # attr_accessor :published - def published - tour.published - end + # def published + # tour.published + # end end diff --git a/app/models/stop.rb b/app/models/stop.rb index afde7bd6..3069a1c1 100644 --- a/app/models/stop.rb +++ b/app/models/stop.rb @@ -9,19 +9,17 @@ class Stop < ApplicationRecord has_many :stop_media has_many :media, through: :stop_media belongs_to :medium, optional: true + belongs_to :map_icon, optional: true has_many :stop_slugs, dependent: :delete_all + before_validation -> { self.title ||= 'untitled' } + validates :title, presence: true - # validates :title, uniqueness: true after_initialize :default_values + before_create :ensure_icon_color after_save :ensure_slug - before_validation -> { self.title ||= 'untitled' } - - scope :not_in_tour, lambda { |tour_id| includes(:tour_stops).where.not(tour_stops: { tour_id: tour_id }) } - scope :no_tours, lambda { includes(:tour_stops).where(tour_stops: { tour_id: nil }) } - scope :published, lambda { includes(:tours).where(tours: { published: true }) } scope :by_slug_and_tour, lambda { |slug, tour_id| joins(:stop_slugs).joins(:tours).where('stop_slugs.slug = ?', slug).where('tour_stops.tour_id = ?', tour_id) } def sanitized_description @@ -33,37 +31,31 @@ def sanitized_direction_notes end def slug - title.parameterize + title ? title.parameterize_intl : '' end def splash - if medium.present? - return medium + splash_medium = if medium.present? + medium elsif stop_media.present? - return stop_media.order(:position).first.medium + stop_media.order(:position).first.medium + else + nil end - nil - end - def splash_height - splash.nil? ? nil : splash.desktop_height - end - - def splash_width - splash.nil? ? nil : splash.desktop_width - end - - def insecure_splash - if !stop_media.empty? - return medium.nil? ? stop_media.order(:position).first.medium.insecure : medium.insecure + if splash_medium&.files + return { title: splash_medium.title, caption: splash_medium.caption, url: splash_medium.files[:desktop] } end nil end - def is_published - tours.published.present? + def orphaned + tours.empty? end + def published + tours.any? { |tour| tour.published } + end private @@ -72,6 +64,10 @@ def default_values end def ensure_slug - StopSlug.find_or_create_by(slug: self.slug, stop: self) + tour_stops.each { |ts| ts.save } + end + + def ensure_icon_color + self.icon_color = '#D32F2F' if icon_color.nil? end end diff --git a/app/models/stop_medium.rb b/app/models/stop_medium.rb index df0c21fc..b8980451 100644 --- a/app/models/stop_medium.rb +++ b/app/models/stop_medium.rb @@ -5,16 +5,20 @@ class StopMedium < ApplicationRecord belongs_to :medium belongs_to :stop + def published + stop&.published + end + after_create do self.position = self.position.nil? ? self.stop.media.length : self.position self.save end - before_destroy do - if self.medium.nil? || self.stop.nil? - nil - else - self.medium.stops.length == 1 ? self.medium.destroy! : nil - end - end + # before_destroy do + # if self.medium.nil? || self.stop.nil? + # nil + # else + # self.medium.stops.length == 1 ? self.medium.destroy! : nil + # end + # end end diff --git a/app/models/stop_slug.rb b/app/models/stop_slug.rb index 44a5c955..332f30e9 100644 --- a/app/models/stop_slug.rb +++ b/app/models/stop_slug.rb @@ -1,4 +1,5 @@ class StopSlug < ApplicationRecord belongs_to :stop + belongs_to :tour # validates :slug, uniqueness: true end diff --git a/app/models/stop_tag.rb b/app/models/stop_tag.rb deleted file mode 100644 index 4172fffa..00000000 --- a/app/models/stop_tag.rb +++ /dev/null @@ -1,6 +0,0 @@ -# frozen_string_literal: true - -# app/models/stop_tag.rb -class StopTag < ApplicationRecord - validates :title, presence: true -end diff --git a/app/models/tour.rb b/app/models/tour.rb index ecb563e5..d6116d49 100644 --- a/app/models/tour.rb +++ b/app/models/tour.rb @@ -1,11 +1,15 @@ # frozen_string_literal: true +require 'uri' + # Model class for a tour. class Tour < ApplicationRecord include HtmlSaintizer + + has_many :tour_stops, autosave: true, dependent: :destroy has_many :stops, -> { distinct }, through: :tour_stops - has_many :tour_modes + has_many :tour_modes, autosave: true, dependent: :destroy has_many :modes, through: :tour_modes belongs_to :mode, default: -> { Mode.last } has_many :tour_media @@ -14,28 +18,45 @@ class Tour < ApplicationRecord has_many :tour_flat_pages has_many :flat_pages, through: :tour_flat_pages has_many :tour_authors - has_many :authors, through: :tour_authors, source: :user + has_many :users, through: :tour_authors has_many :slugs, dependent: :delete_all - # has_many :authors, through: :tour_authors, foreign_key: :user_id + has_one :map_overlay + + # TODO: why does the CircleCI env need to serialize here? + if ENV['CI'] == 'circleci' + serialize :saved_stop_order, Array + end # belongs_to :splash_image_medium_id, class_name: 'Medium' belongs_to :theme, default: -> { Theme.first } - validates :title, presence: true + + enum default_lng: { + "en-US": 0, "fr-FR": 1, "de-DE": 2, "pl-PL": 3, "nl-NL": 4, "fi-FI": 5, "sv-SE": 6, "it-IT": 7, "es-ES": 8, "pt-PT": 9, + "ru-RU": 10, "pt-BR": 11, "es-MX": 12, "zh-CN": 13, "zh-TW": 14, "ja-JP": 15, "ko-KR": 16 + } + + validates :title, presence: true, uniqueness: { case_sensitive: false } before_validation -> { self.mode ||= Mode.last } before_validation -> { self.theme ||= Theme.first } before_validation -> { self.title ||= 'untitled' } + before_validation :update_saved_stop_order + before_save :calculate_duration + before_save :check_url + before_save :check_for_overlay after_save :ensure_slug after_create :add_modes scope :published, -> { where(published: true) } + scope :mapable, -> { where(is_geo: true) } + scope :has_stops, -> { includes(:stops).where.not(stops: { id: nil }) } def sanitized_description HtmlSaintizer.accessable(description) end def slug - title.parameterize + title.parameterize_intl end def tenant @@ -58,31 +79,80 @@ def theme_title end def splash - if medium.present? - return medium + splash_medium = if medium.present? + medium elsif tour_media.present? - return tour_media.order(:position).first.medium + tour_media.order(:position).first.medium + else + nil + end + + if splash_medium + return { title: splash_medium.title, caption: splash_medium.caption, url: splash_medium.files[:desktop] } end nil end - def splash_height - splash.nil? ? nil : splash.desktop_height + def stop_count + self.stops.count end - def splash_width - splash.nil? ? nil : splash.desktop_width + def bounds + if self.restrict_bounds_to_overlay && self.map_overlay.present? + box = RGeo::Cartesian::BoundingBox.create_from_points( + RGeo::Geographic.spherical_factory.point(self.map_overlay.east.to_f, self.map_overlay.south.to_f), + RGeo::Geographic.spherical_factory.point(self.map_overlay.west.to_f, self.map_overlay.north.to_f) + ) + + return { + south: box.min_y - (box.y_span / 8), + north: box.max_y + (box.y_span / 8), + east: box.max_x + (box.x_span / 8), + west: box.min_x - (box.x_span / 8), + centerLat: box.center_y, + centerLng: box.center_x + } + elsif stops.empty? + return nil + end + + points = stops.map { |stop| RGeo::Geographic.spherical_factory.point(stop.lng, stop.lat) } + box = RGeo::Cartesian::BoundingBox.create_from_points(points.pop, points.pop) + points.each { |point| box.add(point) } + + { + south: box.min_y - (box.y_span / 8), + north: box.max_y + (box.y_span / 8), + east: box.max_x + (box.x_span / 8), + west: box.min_x - (box.x_span / 8), + centerLat: box.center_y, + centerLng: box.center_x + } end - def insecure_splash - if !tour_media.empty? - return medium.nil? ? tour_media.order(:position).first.medium.insecure : medium.insecure + def calculate_duration + return unless published + + return if stops.count < 2 + + return if mode.nil? + + return if mode.title.nil? + + return unless self.will_save_change_to_published? || self.will_save_change_to_saved_stop_order? || self.will_save_change_to_mode_id? + + durations = [] + destinations = tour_stops.order(:position).map { |tour_stop| [tour_stop.stop.lat, tour_stop.stop.lng] } + + # The direction matrix API limits the number of destinations to 25. + # Calculate the duration in chunks to stay below the limit. + destinations.each_slice(24) do |group| + origin = group.shift + g_directions = GoogleDirections.new(origin, group, group.count + 1, mode.title) + durations.push(g_directions.duration) end - nil - end - def stop_count - self.stops.count + self.duration = durations.compact.sum.zero? ? nil : durations.sum end private @@ -96,4 +166,31 @@ def add_modes self.modes << m end end + + def check_url + return if link_address.blank? + + uri = URI(link_address) + + self.link_address = "http://#{link_address}" if uri.scheme.nil? + end + + def update_saved_stop_order + self.saved_stop_order = self.tour_stops.order(:position).map(&:stop_id) + end + + def check_for_overlay + if self.restrict_bounds_to_overlay && self.map_overlay.nil? + self.restrict_bounds_to_overlay = false + # self.restrict_bounds = false + end + + if !self.restrict_bounds_to_overlay_was && self.restrict_bounds_to_overlay && self.map_overlay.present? + self.restrict_bounds = false + end + + if self.restrict_bounds && !self.restrict_bounds_was + self.restrict_bounds_to_overlay = false + end + end end diff --git a/app/models/tour_collection.rb b/app/models/tour_collection.rb deleted file mode 100644 index 17092ecc..00000000 --- a/app/models/tour_collection.rb +++ /dev/null @@ -1,4 +0,0 @@ -# frozen_string_literal: true - -class TourCollection < ApplicationRecord -end diff --git a/app/models/tour_medium.rb b/app/models/tour_medium.rb index 2996f8a0..13c44b5b 100644 --- a/app/models/tour_medium.rb +++ b/app/models/tour_medium.rb @@ -11,11 +11,15 @@ class TourMedium < ApplicationRecord self.save end - before_destroy do - if self.medium.nil? || self.tour.nil? - nil - else - self.medium.tours.length == 1 ? self.medium.destroy! : nil - end + # before_destroy do + # if self.medium.nil? || self.tour.nil? + # nil + # else + # self.medium.tours.length == 1 ? self.medium.destroy! : nil + # end + # end + + def published + tour&.published end end diff --git a/app/models/tour_set.rb b/app/models/tour_set.rb index c1ccefc7..ad205817 100644 --- a/app/models/tour_set.rb +++ b/app/models/tour_set.rb @@ -2,25 +2,26 @@ # Model class for tour sets. This is the main model for "instances" of Open Tour Builder. class TourSet < ApplicationRecord - mount_uploader :logo, LogoUploader - mount_uploader :footer_logo, LogoFooterUploader - has_many :tour_set_admins - has_many :admins, through: :tour_set_admins, source: :user before_save :set_subdir + before_save :attach_file after_create :create_tenant after_create :create_defaults before_destroy :drop_tenant + validates :name, presence: true, uniqueness: true - # attr_accessor :footer_logo_width - # attr_accessor :footer_logo_height + + has_one_attached 'logo' + + has_many :tour_set_admins + has_many :admins, through: :tour_set_admins, source: :user + attr_accessor :published_tours - # validate :validate_footer_logo_dimensions, if :uploading? def published_tours begin Apartment::Tenant.switch! self.subdir tours = [] - Tour.published.each do |t| + Tour.published.has_stops.each do |t| tour = { title: t.title, slug: t.slug @@ -33,10 +34,39 @@ def published_tours end end + def mapable_tours + begin + Apartment::Tenant.switch! self.subdir + tours = [] + Tour.published.has_stops.mapable.each do |t| + tour = { + title: t.title, + slug: t.slug, + center: { lat: t.bounds[:centerLat], lng: t.bounds[:centerLng] } + } + tours.push(tour) + end + tours + rescue Apartment::TenantNotFound => error + # self.delete + end + end + + def logo_url + Apartment::Tenant.switch! 'public' + begin + return logo.url if logo.attached? + rescue URI::InvalidURIError + # FIXME: This seems to be a problem when testing? + end + + nil + end + private def set_subdir - self.subdir = name.parameterize + self.subdir = name.parameterize_intl end def create_tenant @@ -57,8 +87,8 @@ def create_tenant end def create_defaults - Apartment::Tenant.reset - themes = Theme.all.collect(&:title) + # Apartment::Tenant.reset + # themes = Theme.all.collect(&:title) Apartment::Tenant.switch! subdir Mode.create([ { title: 'BICYCLING', icon: 'bicycle' }, @@ -66,31 +96,58 @@ def create_defaults { title: 'TRANSIT', icon: 'subway' }, { title: 'WALKING', icon: 'walking' } ]) - themes.each do |t| - Theme.create(title: t) - end + # themes.each do |t| + # Theme.create(title: t) + # end end def drop_tenant Apartment::Tenant.drop(subdir) end - def symlink_logo - FileUtils.mkdir "#{Rails.root}/public/uploads/#{self.subdir}" - FileUtils.ln_s "#{Rails.root}/public/otblogo.png", - "#{Rails.root}/public/uploads/#{self.subdir}/otblogo.png" - self.logo = 'otblogo.png' - self.footer_logo = 'otblogo.png' - end + def tmp_file_path + return nil if logo_title.nil? - def uploading? - footer_logo_width.present? && footer_logo_height.present? + Rails.root.join('public', 'storage', 'tmp', logo_title) end - def validate_footer_logo_dimensions - ::Rails.logger.info "Footer logo upload dimensions: #{self.footer_logo_width}x#{self.footer_logo_height}" - if self.footer_logo_width != footer_logo_height - errors.add :footer_logo, 'Footer logo should be square' + # + # Create and attach file from Base64 string. + # + # This should only be called once when a new medium obeject is created via the API + # It assumes + # + # Some code taken from https://github.com/rootstrap/active-storage-base64/blob/v1.2.0/lib/active_storage_support/base64_attach.rb#L17-L32 + # + # + def attach_file + return if base_sixty_four.nil? && !logo.attached? + + + return if !self.will_save_change_to_base_sixty_four? && logo.attached? + + if base_sixty_four.nil? && logo.attached? + logo.purge + else + headers, self.base_sixty_four = base_sixty_four.split(',') + + return if base_sixty_four.nil? + + File.open(tmp_file_path, 'wb') do |f| + f.write(Base64.decode64(base_sixty_four)) + end + + image = MiniMagick::Image.open(tmp_file_path) + + if image[:height] > 80 + image.resize('300x80') + image.write(tmp_file_path) + end + + self.logo.attach( + io: File.open(tmp_file_path), + filename: logo_title + ) end end end diff --git a/app/models/tour_stop.rb b/app/models/tour_stop.rb index 9f4d5769..b47b61c4 100644 --- a/app/models/tour_stop.rb +++ b/app/models/tour_stop.rb @@ -7,8 +7,8 @@ class TourStop < ApplicationRecord validates :position, presence: true + before_save :_ensure_stop_slug before_validation :_set_position - before_destroy :_delete_orphan def slug stop.slug @@ -37,12 +37,14 @@ def previous_slug private def _set_position + return if tour.nil? + self.position = self.position || self.tour.stops.length + 1 end - def _delete_orphan - if self.stop.tours.length == 1 - self.stop.destroy - end + def _ensure_stop_slug + new_slug = StopSlug.find_or_create_by(slug: self.stop.slug, tour: self.tour) + new_slug.stop = self.stop + new_slug.save end end diff --git a/app/models/tour_tag.rb b/app/models/tour_tag.rb deleted file mode 100644 index f69822fd..00000000 --- a/app/models/tour_tag.rb +++ /dev/null @@ -1,6 +0,0 @@ -# frozen_string_literal: true - -# app/models/tour_tag.rb -class TourTag < ApplicationRecord - validates :title, presence: true -end diff --git a/app/models/user.rb b/app/models/user.rb index 581c5825..1ce93c2a 100644 --- a/app/models/user.rb +++ b/app/models/user.rb @@ -1,13 +1,14 @@ # frozen_string_literal: true class User < ActiveRecord::Base - include EcdsRailsAuthEngine::EcdsUser has_many :tour_set_admins has_many :tour_sets, through: :tour_set_admins has_many :tour_authors has_many :tours, through: :tour_authors - scope :search, -> (search) { joins(:login).where("users.display_name ILIKE '%#{search}%' OR logins.identification ILIKE '%#{search}%'")} + validates :email, presence: true + + # scope :search, -> (search) { joins(:login).where("users.display_name ILIKE '%#{search}%' OR logins.identification ILIKE '%#{search}%'")} # # Gets role for current tenant @@ -17,8 +18,28 @@ class User < ActiveRecord::Base def current_tenant_admin? return true if self.super return false if tour_sets.empty? - tour_sets.collect(&:subdir).include? Apartment::Tenant.current + tour_sets.map(&:subdir).include? Apartment::Tenant.current end -end + def provider + return nil if login.nil? + login.provider + end + def all_tours + all = [] + TourSet.all.each do |tour_set| + Apartment::Tenant.switch! tour_set.subdir + next if tours.empty? || current_tenant_admin? + Apartment::Tenant.switch! tour_set.subdir + _tours = TourAuthor.where(user: self) + all.push(_tours.map { |ta| { id: ta.tour.id, tenant: ta.tour.tenant, title: ta.tour.title } }) + end + Apartment::Tenant.reset + all.flatten.uniq + end + + def login + EcdsRailsAuthEngine::Login.find_by(user_id: self.id) + end +end diff --git a/app/models/v3.rb b/app/models/v3.rb deleted file mode 100644 index b31800ab..00000000 --- a/app/models/v3.rb +++ /dev/null @@ -1,5 +0,0 @@ -module V3 - def self.table_name_prefix - 'v3_' - end -end diff --git a/app/serializers/v3/flat_page_serializer.rb b/app/serializers/v3/flat_page_serializer.rb index e386865b..52fb55c0 100644 --- a/app/serializers/v3/flat_page_serializer.rb +++ b/app/serializers/v3/flat_page_serializer.rb @@ -1,3 +1,6 @@ -class V3::FlatPageSerializer < ActiveModel::Serializer - attributes :id, :title, :content +module V3 + class FlatPageSerializer < ActiveModel::Serializer + has_many :tours + attributes :id, :title, :body, :slug, :orphaned + end end diff --git a/app/serializers/v3/map_icon_serializer.rb b/app/serializers/v3/map_icon_serializer.rb new file mode 100644 index 00000000..1fdf2c28 --- /dev/null +++ b/app/serializers/v3/map_icon_serializer.rb @@ -0,0 +1,6 @@ +module V3 + class MapIconSerializer < ActiveModel::Serializer + include Rails.application.routes.url_helpers + attributes :id, :base_sixty_four, :filename, :original_image_url + end +end diff --git a/app/serializers/v3/map_overlay_serializer.rb b/app/serializers/v3/map_overlay_serializer.rb new file mode 100644 index 00000000..94446148 --- /dev/null +++ b/app/serializers/v3/map_overlay_serializer.rb @@ -0,0 +1,6 @@ +module V3 + class MapOverlaySerializer < ActiveModel::Serializer + include Rails.application.routes.url_helpers + attributes :id, :south, :north, :east, :west, :original_image_url, :filename + end +end diff --git a/app/serializers/v3/medium_serializer.rb b/app/serializers/v3/medium_serializer.rb index a6a5283a..7cbc4f87 100644 --- a/app/serializers/v3/medium_serializer.rb +++ b/app/serializers/v3/medium_serializer.rb @@ -1,5 +1,31 @@ # frozen_string_literal: true -class V3::MediumSerializer < ActiveModel::Serializer - attributes :id, :title, :caption, :desktop, :tablet, :mobile, :video, :provider, :original_image, :embed, :srcset, :srcset_sizes, :insecure, :base64 +module V3 + class MediumSerializer < ActiveModel::Serializer + # include Rails.application.routes.url_helpers + attributes :id, + :title, + :caption, + :video, + :provider, + :original_image, + :embed, + :files, + :orphaned, + :filename, + :original_image_url, + :lqip_width, + :mobile_width, + :tablet_width, + :desktop_width + + # def files + # return nil unless object.file.attached? + # { + # mobile: Rails.application.routes.url_helpers.rails_representation_url(object.file.variant(resize: '200x200').processed), + # tablet: Rails.application.routes.url_helpers.rails_representation_url(object.file.variant(resize: '300x300').processed), + # desktop: Rails.application.routes.url_helpers.rails_representation_url(object.file.variant(resize: '750x750').processed) + # } + # end + end end diff --git a/app/serializers/v3/mode_serializer.rb b/app/serializers/v3/mode_serializer.rb index 755721b5..f5e7a842 100644 --- a/app/serializers/v3/mode_serializer.rb +++ b/app/serializers/v3/mode_serializer.rb @@ -1,6 +1,8 @@ # frozen_string_literal: true -class V3::ModeSerializer < ActiveModel::Serializer - has_many :tours - attributes :id, :title, :icon +module V3 + class ModeSerializer < ActiveModel::Serializer + has_many :tours + attributes :id, :title, :icon + end end diff --git a/app/serializers/v3/stop_medium_serializer.rb b/app/serializers/v3/stop_medium_serializer.rb index 866a4bc2..c925d22b 100644 --- a/app/serializers/v3/stop_medium_serializer.rb +++ b/app/serializers/v3/stop_medium_serializer.rb @@ -1,7 +1,9 @@ # frozen_string_literal: true -class V3::StopMediumSerializer < ActiveModel::Serializer - belongs_to :stop - belongs_to :medium - attributes :id, :position +module V3 + class StopMediumSerializer < ActiveModel::Serializer + belongs_to :stop + belongs_to :medium + attributes :id, :position + end end diff --git a/app/serializers/v3/stop_serializer.rb b/app/serializers/v3/stop_serializer.rb index 648c1679..767923ac 100644 --- a/app/serializers/v3/stop_serializer.rb +++ b/app/serializers/v3/stop_serializer.rb @@ -1,8 +1,30 @@ # frozen_string_literal: true -class V3::StopSerializer < ActiveModel::Serializer - has_many :media - has_many :stop_media - has_many :tours - attributes :id, :title, :slug, :description, :sanitized_description, :sanitized_direction_notes, :lat, :lng, :address, :meta_description, :article_link, :video_embed, :video_poster, :parking_lat, :parking_lng, :direction_intro, :direction_notes, :splash, :insecure_splash, :splash_width, :splash_height +module V3 + class StopSerializer < ActiveModel::Serializer + has_many :media + has_many :stop_media + has_many :tours + belongs_to :map_icon + attributes :id, + :title, + :slug, + :description, + :sanitized_description, + :sanitized_direction_notes, + :lat, + :lng, + :address, + :meta_description, + :article_link, + :video_embed, + :video_poster, + :parking_lat, + :parking_lng, + :direction_intro, + :direction_notes, + :splash, + :orphaned, + :icon_color + end end diff --git a/app/serializers/v3/theme_serializer.rb b/app/serializers/v3/theme_serializer.rb index 50afdade..88a37ae3 100644 --- a/app/serializers/v3/theme_serializer.rb +++ b/app/serializers/v3/theme_serializer.rb @@ -1,5 +1,7 @@ # frozen_string_literal: true -class V3::ThemeSerializer < ActiveModel::Serializer - attributes :id, :title +module V3 + class ThemeSerializer < ActiveModel::Serializer + attributes :id, :title + end end diff --git a/app/serializers/v3/tour_author_serializer.rb b/app/serializers/v3/tour_author_serializer.rb new file mode 100644 index 00000000..9f8bff5d --- /dev/null +++ b/app/serializers/v3/tour_author_serializer.rb @@ -0,0 +1,7 @@ +module V3 + class TourAuthorSerializer < ActiveModel::Serializer + belongs_to :tour + belongs_to :user + attributes :id + end +end diff --git a/app/serializers/v3/tour_base_serializer.rb b/app/serializers/v3/tour_base_serializer.rb new file mode 100644 index 00000000..13adde0d --- /dev/null +++ b/app/serializers/v3/tour_base_serializer.rb @@ -0,0 +1,54 @@ +# frozen_string_literal: true + +include ActionView::Helpers::DateHelper + +# app/serializers/tour_serializer.rb +module V3 + class TourBaseSerializer < ActiveModel::Serializer + has_one :map_overlay + attributes :id, + :title, + :slug, + :description, + :is_geo, + :published, + :sanitized_description, + :position, + :theme_title, + :meta_description, + :tenant, + :tenant_title, + :stop_count, + :map_type, + :splash, + :use_directions, + :default_lng, + :stop_count, + :est_time, + :link_address, + :link_text, + :restrict_bounds, + :restrict_bounds_to_overlay, + :blank_map + + def est_time + return nil if object.duration.nil? + + "#{distance_of_time_in_words(object.duration).capitalize} #{object.mode.title.downcase}" + end + + def map_type + object.map_type || 'hybrid' + end + + def bounds + return object.bounds if object.bounds.present? + + if @instance_options[:loc].present? + return @instance_options[:loc] + end + + nil + end + end +end diff --git a/app/serializers/v3/tour_collection_serializer.rb b/app/serializers/v3/tour_collection_serializer.rb deleted file mode 100644 index ddf56b81..00000000 --- a/app/serializers/v3/tour_collection_serializer.rb +++ /dev/null @@ -1,5 +0,0 @@ -# frozen_string_literal: true - -class V3::TourCollectionSerializer < ActiveModel::Serializer - attributes :id -end diff --git a/app/serializers/v3/tour_flat_page_serializer.rb b/app/serializers/v3/tour_flat_page_serializer.rb index 904ccb18..67dfef6f 100644 --- a/app/serializers/v3/tour_flat_page_serializer.rb +++ b/app/serializers/v3/tour_flat_page_serializer.rb @@ -1,5 +1,7 @@ -class V3::TourFlatPageSerializer < ActiveModel::Serializer - belongs_to :tour - belongs_to :flat_page - attributes :id, :position +module V3 + class TourFlatPageSerializer < ActiveModel::Serializer + belongs_to :tour + belongs_to :flat_page + attributes :id, :position + end end diff --git a/app/serializers/v3/tour_medium_serializer.rb b/app/serializers/v3/tour_medium_serializer.rb index 4e67cdf7..465bc51f 100644 --- a/app/serializers/v3/tour_medium_serializer.rb +++ b/app/serializers/v3/tour_medium_serializer.rb @@ -1,7 +1,9 @@ # frozen_string_literal: true -class V3::TourMediumSerializer < ActiveModel::Serializer - belongs_to :tour - belongs_to :medium - attributes :id, :position +module V3 + class TourMediumSerializer < ActiveModel::Serializer + belongs_to :tour + belongs_to :medium + attributes :id, :position + end end diff --git a/app/serializers/v3/tour_mode_serializer.rb b/app/serializers/v3/tour_mode_serializer.rb new file mode 100644 index 00000000..f69200b6 --- /dev/null +++ b/app/serializers/v3/tour_mode_serializer.rb @@ -0,0 +1,9 @@ +# frozen_string_literal: true + +module V3 + class TourModeSerializer < ActiveModel::Serializer + belongs_to :tour + belongs_to :mode + attributes :id + end +end diff --git a/app/serializers/v3/tour_modes_serializer.rb b/app/serializers/v3/tour_modes_serializer.rb deleted file mode 100644 index 05e542bf..00000000 --- a/app/serializers/v3/tour_modes_serializer.rb +++ /dev/null @@ -1,5 +0,0 @@ -# frozen_string_literal: true - -class V3::TourModesSerializer < ActiveModel::Serializer - attributes :id, :tour_id, :mode_id -end diff --git a/app/serializers/v3/tour_serializer.rb b/app/serializers/v3/tour_serializer.rb index a9df2843..052c7f74 100644 --- a/app/serializers/v3/tour_serializer.rb +++ b/app/serializers/v3/tour_serializer.rb @@ -1,15 +1,20 @@ # frozen_string_literal: true # app/serializers/tour_serializer.rb -class V3::TourSerializer < ActiveModel::Serializer - has_many :tour_stops - has_many :stops - belongs_to :mode - belongs_to :theme - has_many :modes - has_many :media - has_many :tour_media - has_many :flat_pages - has_many :tour_flat_pages - attributes :id, :title, :slug, :description, :is_geo, :published, :sanitized_description, :position, :theme_title, :meta_description, :splash, :tenant, :tenant_title, :stop_count, :map_type, :splash_width, :splash_height, :insecure_splash +module V3 + class TourSerializer < V3::TourBaseSerializer + has_many :tour_modes + has_many :tour_stops + has_many :stops + belongs_to :mode + belongs_to :theme + has_many :modes + has_many :media + has_many :tour_media + has_many :flat_pages + has_many :tour_flat_pages + has_many :users + + attributes :bounds + end end diff --git a/app/serializers/v3/tour_set_admin_serializer.rb b/app/serializers/v3/tour_set_admin_serializer.rb index 0efe3f85..4a1508c3 100644 --- a/app/serializers/v3/tour_set_admin_serializer.rb +++ b/app/serializers/v3/tour_set_admin_serializer.rb @@ -1,3 +1,5 @@ -class TourSetAdminSerializer < ActiveModel::Serializer - attributes :id +module V3 + class TourSetAdminSerializer < ActiveModel::Serializer + attributes :id + end end diff --git a/app/serializers/v3/tour_set_serializer.rb b/app/serializers/v3/tour_set_serializer.rb index 5075e24a..497a509d 100644 --- a/app/serializers/v3/tour_set_serializer.rb +++ b/app/serializers/v3/tour_set_serializer.rb @@ -1,11 +1,19 @@ # frozen_string_literal: true -class V3::TourSetSerializer < ActiveModel::Serializer - # attribute :tenant_admins - has_many :admins - attributes :id, :name, :subdir, :published_tours +module V3 + class TourSetSerializer < ActiveModel::Serializer + # attribute :tenant_admins + include Rails.application.routes.url_helpers + has_many :admins + attributes :id, :name, :subdir, :published_tours, :mapable_tours, :logo_url, :logo - def admins - object.admins if current_user.super || current_user.current_tenant_admin? + def admins + begin + object.admins if current_user&.super || current_user&.tour_sets.include?(object) + rescue NameError + # This is a problem when using the serializer directly + nil + end + end end end diff --git a/app/serializers/v3/tour_stop_serializer.rb b/app/serializers/v3/tour_stop_serializer.rb index a0641df9..ad8bef77 100644 --- a/app/serializers/v3/tour_stop_serializer.rb +++ b/app/serializers/v3/tour_stop_serializer.rb @@ -1,8 +1,10 @@ # frozen_string_literal: true # /app/serializers/tour_stop_serializer.rb -class V3::TourStopSerializer < ActiveModel::Serializer - belongs_to :tour - belongs_to :stop - attributes :id, :position, :previous, :slug, :next, :next_slug, :previous_slug +module V3 + class TourStopSerializer < ActiveModel::Serializer + belongs_to :tour + belongs_to :stop + attributes :id, :position, :previous, :slug, :next, :next_slug, :previous_slug + end end diff --git a/app/serializers/v3/user_serializer.rb b/app/serializers/v3/user_serializer.rb index d78c4990..fec6739d 100644 --- a/app/serializers/v3/user_serializer.rb +++ b/app/serializers/v3/user_serializer.rb @@ -1,13 +1,22 @@ # frozen_string_literal: true -# app/serializer/user_serializer.rb -class V3::UserSerializer < ActiveModel::Serializer - has_one :login - has_many :tours - has_many :tour_sets - attributes :id, :display_name, :super, :login, :current_tenant_admin +# app/serializer/v3/user_serializer.rb +module V3 + class UserSerializer < ActiveModel::Serializer + has_many :tours + has_many :tour_authors + has_many :tour_sets + attributes :id, :display_name, :super, :current_tenant_admin, :provider, :email, :all_tours, :terms_accepted - def current_tenant_admin - object.current_tenant_admin? + def current_tenant_admin + object.current_tenant_admin? + end + + def all_tours + if @instance_options[:include_tours] + return object.all_tours + end + [] + end end end diff --git a/app/services/google_directions.rb b/app/services/google_directions.rb new file mode 100644 index 00000000..89547580 --- /dev/null +++ b/app/services/google_directions.rb @@ -0,0 +1,34 @@ +class GoogleDirections + + def initialize(origin, destinations, stops_count, mode) + google_key = ENV['RAILS_ENV'] == 'test' ? 'FAkeFaK-E_fAkeChv-P3nchtQYHoCLfFzn9ylr8' : Rails.application.credentials.dig(:g_maps_key) + @query = { + origins: origin.join(','), + destinations: destinations.map { |d| d.join(',') }.join('|'), + mode: mode, + key: google_key + } + + @stops_count = stops_count + end + + def matrix + response = HTTParty.get( + "https://maps.googleapis.com/maps/api/distancematrix/json?#{@query.to_query}" + ).with_indifferent_access + + return nil if response[:status] && response[:rows].nil? + + return nil if response[:rows].first[:elements].first[:status] == 'ZERO_RESULTS' + + response + end + + def durations + matrix.nil? ? nil : matrix[:rows].first[:elements].map { |e| e[:duration][:value] if e[:duration].present? }.reject { |d| d.nil? } + end + + def duration + durations.nil? ? nil : durations.sum + 600 + (@stops_count * 600) + end +end diff --git a/bin/bundle b/bin/bundle index 58115ecf..f19acf5b 100755 --- a/bin/bundle +++ b/bin/bundle @@ -1,5 +1,3 @@ #!/usr/bin/env ruby -# frozen_string_literal: true - -ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../Gemfile', __FILE__) +ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../Gemfile', __dir__) load Gem.bin_path('bundler', 'bundle') diff --git a/bin/rails b/bin/rails index a3655392..07396602 100755 --- a/bin/rails +++ b/bin/rails @@ -1,11 +1,4 @@ #!/usr/bin/env ruby -# frozen_string_literal: true - -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 index 169c9396..17240489 100755 --- a/bin/rake +++ b/bin/rake @@ -1,11 +1,4 @@ #!/usr/bin/env ruby -# frozen_string_literal: true - -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 index 1a8bfdd8..a334d86a 100755 --- a/bin/setup +++ b/bin/setup @@ -1,12 +1,9 @@ #!/usr/bin/env ruby -# frozen_string_literal: true - -require 'pathname' require 'fileutils' include FileUtils # path to your application root. -APP_ROOT = Pathname.new File.expand_path('../../', __FILE__) +APP_ROOT = File.expand_path('..', __dir__) def system!(*args) system(*args) || abort("\n== Command #{args} failed ==") @@ -20,7 +17,6 @@ chdir APP_ROOT do system! 'gem install bundler --conservative' system('bundle check') || system!('bundle install') - # puts "\n== Copying sample files ==" # unless File.exist?('config/database.yml') # cp 'config/database.yml.sample', 'config/database.yml' diff --git a/bin/update b/bin/update index fdac831b..67d0d496 100755 --- a/bin/update +++ b/bin/update @@ -1,12 +1,9 @@ #!/usr/bin/env ruby -# frozen_string_literal: true - -require 'pathname' require 'fileutils' include FileUtils # path to your application root. -APP_ROOT = Pathname.new File.expand_path('../../', __FILE__) +APP_ROOT = File.expand_path('..', __dir__) def system!(*args) system(*args) || abort("\n== Command #{args} failed ==") diff --git a/build.sh b/build.sh new file mode 100755 index 00000000..ef706fc2 --- /dev/null +++ b/build.sh @@ -0,0 +1,29 @@ +#!/bin/bash +set -e + +TAG=$([ "$BRANCH" == "main" ] && echo "stable" || echo "latest") + +AWS_ECS_CLUSTER=$([ "$BRANCH" == "main" ] && echo "$AWS_ECS_CLUSTER_PROD" || echo "$AWS_ECS_CLUSTER_DEV") + +AWS_ECS_SERVICE=$([ "$BRANCH" == "main" ] && echo "$AWS_ECS_SERVICE_PROD" || echo "$AWS_ECS_SERVICE_DEV") + +echo "Building image for branch: $BRANCH with tag: $TAG" + +docker build \ + --file Dockerfile \ + -t otb \ + . + +echo "Logging in to AWS" +aws ecr get-login-password --region us-east-1 | + docker login --username AWS --password-stdin "${AWS_ECR}" +echo "Logged in successfully" + +echo "Tagging image with $TAG" +docker tag otb "${AWS_ECR}/otb:${TAG}" + +echo "Pushing image" +docker push "${AWS_ECR}/otb:${TAG}" + +echo "Force update service" +aws ecs update-service --cluster ${AWS_ECS_CLUSTER} --service ${AWS_ECS_SERVICE} --force-new-deployment --region ${AWS_REGION} \ No newline at end of file diff --git a/config/application.rb b/config/application.rb index b6e935f9..3933417c 100644 --- a/config/application.rb +++ b/config/application.rb @@ -1,6 +1,5 @@ # frozen_string_literal: true -# /config/application.rb require_relative 'boot' require 'rails' @@ -15,6 +14,8 @@ # require "sprockets/railtie" require 'rails/test_unit/railtie' require 'apartment/elevators/generic' +require 'active_storage/engine' +require 'ipinfo-rails' # Require the gems listed in Gemfile, including any gems # you've limited to :test, :development, or :production. @@ -27,6 +28,10 @@ class DirectoryElevator < Apartment::Elevators::Generic def parse_tenant_name(request) # request is an instance of Rack::Request tenant_name = request.fullpath.split('/')[1] + + if tenant_name == 'auth' || tenant_name == 'rails' + return nil + end tenant_name end end @@ -36,7 +41,11 @@ def parse_tenant_name(request) # 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. - + # config.active_storage.variant_processor = :vips + config.middleware.use(ActionDispatch::Cookies) + config.middleware.use(ActionDispatch::Session::CookieStore) + config.action_dispatch.cookies_serializer = :json + config.middleware.use(IPinfoMiddleware, { token: ENV['IPINFO_TOKEN'] || Rails.application.credentials.dig(:ipinfo) }) # Only loads a smaller set of middleware suitable for API only apps. # Middleware like session, flash, cookies can be added back manually. # Skip views, helpers and assets when generating a new resource. diff --git a/config/cable.yml b/config/cable.yml index ed7b5e43..4a3c0b3c 100644 --- a/config/cable.yml +++ b/config/cable.yml @@ -6,5 +6,5 @@ test: production: adapter: redis - url: redis://localhost:6379/1 + url: <%= ENV.fetch("REDIS_URL") { "redis://localhost:6379/1" } %> channel_prefix: open_tour_api_production diff --git a/config/database.yml b/config/database.yml new file mode 100644 index 00000000..faeb8607 --- /dev/null +++ b/config/database.yml @@ -0,0 +1,22 @@ +default: &default + adapter: <%= ENV['DB_ADAPTER'] || 'postgresql' %> + encoding: utf8 + pool: 50 + schema_search_path: "public,shared_extensions" + username: <%= ENV['DB_USERNAME'] || 'user'%> + host: <%= ENV['DB_HOSTNAME'] || 'localhost' %> + schema_search_path: public, postgis + password: <%= ENV['DB_PASSWORD'] || 'password' %> + database: <%= ENV['DB_NAME'] || 'otb' %> + +development: + <<: *default + database: <%= ENV['DB_NAME'] || 'otb_development' %> + username: <%= ENV['DB_USERNAME'] || 'user' %> + password: <%= ENV['DB_PASSWORD'] || 'password' %> + host: <%= ENV['DB_HOSTNAME'] || 'localhost' %> + +test: + <<: *default + database: <%= ENV['TEST_DB_NAME'] || 'otb_test' %> + \ No newline at end of file diff --git a/config/database.yml.backup b/config/database.yml.backup new file mode 100644 index 00000000..0693ea9e --- /dev/null +++ b/config/database.yml.backup @@ -0,0 +1,45 @@ +default: &default + adapter: <%= ENV['DB_ADAPTER'] || 'postgresql' %> + encoding: utf8 + pool: 50 + schema_search_path: "public,shared_extensions" + username: <%= ENV['DB_USERNAME'] || 'user'%> + host: <%= ENV['DB_HOSTNAME'] || 'localhost' %> + schema_search_path: public, postgis + password: <%= ENV['DB_PASSWORD'] || 'password' %> + database: <%= ENV['DB_NAME'] || 'otb' %> + +# Commented out mysql section that was causing encryption errors +# mysql: &mysql +# <<: *default +# adapter: mysql2 +# database: public +# username: <%= Rails.application.credentials.dig(:dbTest, :mysql, :user) %> +# password: <%= Rails.application.credentials.dig(:dbTest, :mysql, :pw) %> +# host: <%= Rails.application.credentials.dig(:dbTest, :mysql, :host) %> + +development: + <<: *default + database: <%= ENV['DB_NAME'] || 'otb_development' %> + username: <%= ENV['DB_USERNAME'] || 'user' %> + password: <%= ENV['DB_PASSWORD'] || 'password' %> + host: <%= ENV['DB_HOSTNAME'] || 'localhost' %> + +# Commented out staging and production - use environment variables when needed +# staging: +# <<: *default +# database: <%= Rails.application.credentials.dig(:rdsStaging, :db) %> +# username: <%= Rails.application.credentials.dig(:rdsStaging, :user) %> +# password: <%= Rails.application.credentials.dig(:rdsStaging, :pw) %> +# host: <%= Rails.application.credentials.dig(:rdsStaging, :host) %> + +# production: +# <<: *default +# database: <%= Rails.application.credentials.dig(:rdsProduction, :db) %> +# username: <%= Rails.application.credentials.dig(:rdsProduction, :user) %> +# password: <%= Rails.application.credentials.dig(:rdsProduction, :pw) %> +# host: <%= Rails.application.credentials.dig(:rdsProduction, :host) %> + +test: + <<: *default + database: <%= ENV['TEST_DB_NAME'] || 'otb_test' %> \ No newline at end of file diff --git a/config/database.yml.travis b/config/database.yml.travis deleted file mode 100644 index 08b31c12..00000000 --- a/config/database.yml.travis +++ /dev/null @@ -1,13 +0,0 @@ -postgres: &postgres - adapter: postgresql - username: postgres - schema_search_path: "public,shared_extensions" - -mysql: &mysql - adapter: mysql2 - username: root - -test: - database: otb_test - host: localhost - <<: *<%= ENV['DB'] || "postgres" %> diff --git a/config/deploy.rb b/config/deploy.rb index 15f21300..05d670ec 100644 --- a/config/deploy.rb +++ b/config/deploy.rb @@ -1,7 +1,7 @@ # frozen_string_literal: true # config valid for current version and patch releases of Capistrano -lock '~> 3.12.1' +lock '~> 3.17.0' set :application, 'otb-api-server' set :repo_url, 'git@github.com:ecds/otb-api-server.git' @@ -23,13 +23,11 @@ # set :pty, true # Default value for :linked_files is [] -append :linked_files, 'config/database.yml', 'config/initializers/auth.rb', 'config/secrets.yml' +append :linked_files, 'config/master.key' +append :linked_dirs, 'public/storage' # append :linked_files, 'config/database.yml' -# Default value for linked_dirs is [] -append :linked_dirs, 'public/uploads' - # Default value for default_env is {} # set :default_env, { path: '/opt/ruby/bin:$PATH' } diff --git a/config/deploy/production.rb b/config/deploy/production.rb index 66025852..906861df 100644 --- a/config/deploy/production.rb +++ b/config/deploy/production.rb @@ -1,62 +1,13 @@ -# server-based syntax -# ====================== -# Defines a single server with a list of roles and multiple properties. -# You can define all roles on a single server, or split them: set :branch, 'develop' -server "opentour.emory.edu", user: "deploy", roles: %w{app db web}, primary: :my_value -# server "example.com", user: "deploy", roles: %w{app web}, other_property: :other_value -# server "db.example.com", user: "deploy", roles: %w{db} +# server '44.192.30.237', user: 'deploy', roles: %w{app db web}, primary: :my_value +role :app, %w{3.86.138.59 3.236.252.227}, user: 'deploy' +role :web, %w{3.86.138.59 3.236.252.227}, user: 'deploy' +role :db, %w{3.86.138.59 3.236.252.227}, user: 'deploy' +set :deploy_to, '/data/otb-api-server' - -# role-based syntax -# ================== - -# Defines a role with one or multiple servers. The primary server in each -# group is considered to be the first unless any hosts have the primary -# property set. Specify the username and a domain or IP for the server. -# Don't use `:all`, it's a meta role. - -# role :app, %w{deploy@example.com}, my_property: :my_value -# role :web, %w{user1@primary.com user2@additional.com}, other_property: :other_value -# role :db, %w{deploy@example.com} - - - -# Configuration -# ============= -# You can set any configuration variable like in config/deploy.rb -# These variables are then only loaded and set in this stage. -# For available Capistrano configuration variables see the documentation page. -# http://capistranorb.com/documentation/getting-started/configuration/ -# Feel free to add new variables to customise your setup. - - - -# Custom SSH Options -# ================== -# You may pass any option but keep in mind that net/ssh understands a -# limited set of options, consult the Net::SSH documentation. -# http://net-ssh.github.io/net-ssh/classes/Net/SSH.html#method-c-start -# -# Global options -# -------------- -# set :ssh_options, { -# keys: %w(/home/rlisowski/.ssh/id_rsa), -# forward_agent: false, -# auth_methods: %w(password) -# } -# -# The server-based syntax can be used to override options: -# ------------------------------------ -# server "example.com", -# user: "user_name", -# roles: %w{web app}, -# ssh_options: { -# user: "user_name", # overrides user setting above -# keys: %w(/home/user_name/.ssh/id_rsa), -# forward_agent: false, -# auth_methods: %w(publickey password) -# # password: "please use keys" -# } +set :ssh_options, { + forward_agent: false, + auth_methods: %w(publickey) +} diff --git a/config/deploy/staging.rb b/config/deploy/staging.rb index 1a72db53..16ba1d28 100644 --- a/config/deploy/staging.rb +++ b/config/deploy/staging.rb @@ -1,13 +1,13 @@ # frozen_string_literal: true -set :branch, 'develop3.0' +set :branch, 'develop' # server-based syntax # ====================== # Defines a single server with a list of roles and multiple properties. # You can define all roles on a single server, or split them: -server "otb.ecdsdev.org", user: "deploy", roles: %w{app db web}, primary: :my_value +server "3.238.239.164", user: "deploy", roles: %w{app db web}, primary: :my_value # server "otb.ecdsdev.org", user: "deploy", roles: %w{app web}, other_property: :other_value # server "db.otb.ecdsdev.org", user: "deploy", roles: %w{db} @@ -21,9 +21,9 @@ # property set. Specify the username and a domain or IP for the server. # Don't use `:all`, it's a meta role. -# role :app, %w{deploy@otb.ecdsdev.org}, my_property: :my_value -# role :web, %w{user1@primary.com user2@additional.com}, other_property: :other_value -# role :db, %w{deploy@otb.ecdsdev.org} +role :app, %w{deploy@3.238.239.164} +role :web, %w{user1@3.238.239.164} +role :db, %w{deploy@3.238.239.164} @@ -45,11 +45,13 @@ # # Global options # -------------- -# set :ssh_options, { -# keys: %w(/home/rlisowski/.ssh/id_rsa), -# forward_agent: false, -# auth_methods: %w(password) -# } + set :ssh_options, { + forward_agent: false, + auth_methods: %w(publickey) + } + + set :branch, 'develop' + set :deploy_to, '/data/otb-api' # # The server-based syntax can be used to override options: # ------------------------------------ diff --git a/config/environment.rb b/config/environment.rb index 0ea7aadb..f9f287fd 100644 --- a/config/environment.rb +++ b/config/environment.rb @@ -4,7 +4,7 @@ require_relative 'application' # Initialize the Rails application. -Rails.application.initialize! Rails.application.configure do config.force_ssl = true end +Rails.application.initialize! diff --git a/config/environments/development.rb b/config/environments/development.rb index 12ba559e..747b0597 100644 --- a/config/environments/development.rb +++ b/config/environments/development.rb @@ -2,7 +2,8 @@ Rails.application.configure do # Settings specified here will take precedence over those in config/application.rb. - + config.hosts << 'otb.org' + Rails.application.routes.default_url_options[:host] = 'https://otb.org:3000' # 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. @@ -14,6 +15,9 @@ # Show full error reports. config.consider_all_requests_local = true + # Active Storage + config.active_storage.service = :local + # Enable/disable caching. By default caching is disabled. if Rails.root.join('tmp/caching-dev.txt').exist? config.action_controller.perform_caching = true diff --git a/config/environments/mysql.rb b/config/environments/mysql.rb new file mode 120000 index 00000000..c85ef2a0 --- /dev/null +++ b/config/environments/mysql.rb @@ -0,0 +1 @@ +development.rb \ No newline at end of file diff --git a/config/environments/production.rb b/config/environments/production.rb index 7118d532..98b48202 100644 --- a/config/environments/production.rb +++ b/config/environments/production.rb @@ -1,8 +1,13 @@ # frozen_string_literal: true Rails.application.configure do + config.hosts << 'api.opentour.site' + Rails.application.routes.default_url_options[:host] = 'https://api.opentour.site' # Settings specified here will take precedence over those in config/application.rb. + # Store uploaded files on the local file system in a temporary directory. + config.active_storage.service = :production + # Code is not reloaded between requests. config.cache_classes = true @@ -82,7 +87,7 @@ # Do not dump schema after migrations. config.active_record.dump_schema_after_migration = false - ENV['BASE_URL'] = 'https://api.opentour.emory.edu' + ENV['BASE_URL'] = 'https://api.opentour.site' ENV['INSECURE_IMAGE_BASE_URL'] = 'http://otbimages.ecdsdev.org' end diff --git a/config/environments/staging.rb b/config/environments/staging.rb index d79dd7cb..45c49784 100644 --- a/config/environments/staging.rb +++ b/config/environments/staging.rb @@ -1,6 +1,8 @@ # frozen_string_literal: true Rails.application.configure do + config.hosts << 'otb-api.ecdsdev.org' + Rails.application.routes.default_url_options[:host] = 'https://otb-api.ecdsdev.org' # Settings specified here will take precedence over those in config/application.rb. # In the development environment your application's code is reloaded on @@ -49,4 +51,6 @@ config.file_watcher = ActiveSupport::FileUpdateChecker ENV['BASE_URL'] = 'https://otb-api.ecdsdev.org' + + config.active_storage.service = :staging end diff --git a/config/environments/test.rb b/config/environments/test.rb index 5971b4bf..668a85ce 100644 --- a/config/environments/test.rb +++ b/config/environments/test.rb @@ -3,6 +3,9 @@ Rails.application.configure do # Settings specified here will take precedence over those in config/application.rb. + # Store uploaded files on the local file system in a temporary directory. + config.active_storage.service = :test + # 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 @@ -44,5 +47,12 @@ # This is needed for the tests to request tests to pass when subdomain is set. config.action_dispatch.tld_length = 0 + config.force_ssl = false + + # config.active_storage.service = :test + # config.consider_all_requests_local = true + # config.action_controller.perform_caching = false + # config.host = 'localhost:3030' + # config.action_controller.default_url_options = { host: 'localhost:3030' } end diff --git a/config/initializers/apartment.rb b/config/initializers/apartment.rb index 1fef55ec..46ac452a 100644 --- a/config/initializers/apartment.rb +++ b/config/initializers/apartment.rb @@ -3,6 +3,6 @@ # require 'directory_elevator' Apartment.configure do |config| config.tenant_names = -> { TourSet.pluck :subdir } - config.excluded_models = ['Login', 'User', 'Role', 'TourSetAdmin', 'TourSet'] + config.excluded_models = ['User', 'Role', 'TourSetAdmin', 'TourSet', 'EcdsRailsAuthEngine::Login', 'EcdsRailsAuthEngine::Token', 'Theme'] config.persistent_schemas = ['shared_extensions'] end diff --git a/config/initializers/application_controller_renderer.rb b/config/initializers/application_controller_renderer.rb new file mode 100644 index 00000000..89d2efab --- /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/backtrace_silencers.rb b/config/initializers/backtrace_silencers.rb new file mode 100644 index 00000000..59385cdf --- /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/cookie_session.rb b/config/initializers/cookie_session.rb new file mode 100644 index 00000000..53cfc4f4 --- /dev/null +++ b/config/initializers/cookie_session.rb @@ -0,0 +1,3 @@ +# frozen_string_literal: true +Rails.application.config.session_store(:cookie_store, key: '_otb_session') +Rails.application.config.action_dispatch.cookies_serializer = :json diff --git a/config/initializers/cors.rb b/config/initializers/cors.rb index eec10024..f341144d 100644 --- a/config/initializers/cors.rb +++ b/config/initializers/cors.rb @@ -9,10 +9,11 @@ Rails.application.config.middleware.insert_before 0, Rack::Cors do allow do - origins '*' + origins 'https://lvh.me:4200', 'https://otb.ecdsdev.org', 'https://opentour.site', /.*\.opentour.site/, /.*\.lvh.me:4200/, /.*localhost:3000/, /.*\.urbanspatialhistory.org/ resource '*', headers: :any, - methods: [:get, :post, :put, :patch, :delete, :options, :head] + methods: [:get, :post, :put, :patch, :delete, :options, :head], + credentials: true end end diff --git a/config/initializers/disable_ssl_in_development.rb b/config/initializers/disable_ssl_in_development.rb new file mode 100644 index 00000000..13c334c7 --- /dev/null +++ b/config/initializers/disable_ssl_in_development.rb @@ -0,0 +1 @@ +Rails.application.configure { config.force_ssl = false } if Rails.env.development? diff --git a/config/initializers/filter_parameter_logging.rb b/config/initializers/filter_parameter_logging.rb index 7a4f47b4..a18206dc 100644 --- a/config/initializers/filter_parameter_logging.rb +++ b/config/initializers/filter_parameter_logging.rb @@ -3,4 +3,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] +Rails.application.config.filter_parameters += [:password, :base_sixty_four] diff --git a/config/initializers/inflections.rb b/config/initializers/inflections.rb index aa7435fb..ac033bf9 100644 --- a/config/initializers/inflections.rb +++ b/config/initializers/inflections.rb @@ -1,4 +1,3 @@ -# frozen_string_literal: true # Be sure to restart your server when you modify this file. # Add new inflection rules using the following format. Inflections diff --git a/config/initializers/kaminari_config.rb b/config/initializers/kaminari_config.rb new file mode 100644 index 00000000..4ba6ee3e --- /dev/null +++ b/config/initializers/kaminari_config.rb @@ -0,0 +1,14 @@ +# frozen_string_literal: true + +Kaminari.configure do |config| + # config.default_per_page = 25 + # config.max_per_page = nil + # config.window = 4 + # config.outer_window = 0 + # config.left = 0 + # config.right = 0 + # config.page_method_name = :page + # config.param_name = :page + # config.max_pages = nil + # config.params_on_first_page = false +end diff --git a/config/initializers/mime_types.rb b/config/initializers/mime_types.rb new file mode 100644 index 00000000..dc189968 --- /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/new_framework_defaults_5_2.rb b/config/initializers/new_framework_defaults_5_2.rb new file mode 100644 index 00000000..c383d072 --- /dev/null +++ b/config/initializers/new_framework_defaults_5_2.rb @@ -0,0 +1,38 @@ +# Be sure to restart your server when you modify this file. +# +# This file contains migration options to ease your Rails 5.2 upgrade. +# +# Once upgraded flip defaults one by one to migrate to the new default. +# +# Read the Guide for Upgrading Ruby on Rails for more info on each option. + +# Make Active Record use stable #cache_key alongside new #cache_version method. +# This is needed for recyclable cache keys. +# Rails.application.config.active_record.cache_versioning = true + +# Use AES-256-GCM authenticated encryption for encrypted cookies. +# Also, embed cookie expiry in signed or encrypted cookies for increased security. +# +# This option is not backwards compatible with earlier Rails versions. +# It's best enabled when your entire app is migrated and stable on 5.2. +# +# Existing cookies will be converted on read then written with the new scheme. +# Rails.application.config.action_dispatch.use_authenticated_cookie_encryption = true + +# Use AES-256-GCM authenticated encryption as default cipher for encrypting messages +# instead of AES-256-CBC, when use_authenticated_message_encryption is set to true. +# Rails.application.config.active_support.use_authenticated_message_encryption = true + +# Add default protection from forgery to ActionController::Base instead of in +# ApplicationController. +# Rails.application.config.action_controller.default_protect_from_forgery = true + +# Store boolean values are in sqlite3 databases as 1 and 0 instead of 't' and +# 'f' after migrating old data. +# Rails.application.config.active_record.sqlite3.represent_boolean_as_integer = true + +# Use SHA-1 instead of MD5 to generate non-sensitive digests, such as the ETag header. +# Rails.application.config.active_support.use_sha1_digests = true + +# Make `form_with` generate id attributes for any generated HTML tags. +# Rails.application.config.action_view.form_with_generates_ids = true diff --git a/config/initializers/pagy.rb b/config/initializers/pagy.rb new file mode 100644 index 00000000..6149760f --- /dev/null +++ b/config/initializers/pagy.rb @@ -0,0 +1,3 @@ +# require 'pagy/extras/headers' +# Pagy::DEFAULT[:items] = 2 +# Pagy::DEFAULT.freeze diff --git a/config/initializers/s3.rb b/config/initializers/s3.rb new file mode 100644 index 00000000..e3e8b945 --- /dev/null +++ b/config/initializers/s3.rb @@ -0,0 +1,11 @@ +# frozen_string_literal: true + +# Aws.config.update({ +# credentials: Aws::Credentials.new( +# Rails.application.credentials.s3Staging[:access_key_id], +# Rails.application.credentials.s3Staging[:secret_access_key] +# ) +# }) + +# Sometimes the service URL expires too quickly. +Rails.application.config.active_storage.service_urls_expire_in = 1.week diff --git a/config/initializers/string.rb b/config/initializers/string.rb new file mode 100644 index 00000000..d57e10a7 --- /dev/null +++ b/config/initializers/string.rb @@ -0,0 +1,32 @@ +require "active_support/inflector" + +class String + def parameterize_intl(separator: '-', preserve_case: false, locale: nil) + # Replace accented chars with their ASCII equivalents. + transliterated_string = ActiveSupport::Inflector.transliterate(self, replacement = '~', locale: locale) + + parameterized_string = if transliterated_string.include?('~') + self.gsub(/[!@#$%^&*()-=_+|;':",.<>?\s']/, separator) + else + transliterated_string.gsub(/[^a-z0-9\-_]+/i, separator) + end + + unless separator.nil? || separator.empty? + if separator == '_'.freeze + re_duplicate_separator = /-{2,}/ + re_leading_trailing_separator = /^-|-$/ + else + re_sep = Regexp.escape(separator) + re_duplicate_separator = /#{re_sep}{2,}/ + re_leading_trailing_separator = /^#{re_sep}|#{re_sep}$/ + end + # No more than one of the separator in a row. + parameterized_string.gsub!(re_duplicate_separator, separator) + # Remove leading/trailing separator. + parameterized_string.gsub!(re_leading_trailing_separator, ''.freeze) + end + + parameterized_string.downcase! unless preserve_case + parameterized_string + end +end diff --git a/config/initializers/version.rb b/config/initializers/version.rb index 912d5d2d..a0e6e767 100644 --- a/config/initializers/version.rb +++ b/config/initializers/version.rb @@ -1,5 +1,5 @@ module OpenTourApi class Application - VERSION = '3.0.1rc' + VERSION = '3.0' end end \ No newline at end of file diff --git a/config/initializers/wrap_parameters.rb b/config/initializers/wrap_parameters.rb index e7148a36..bbfc3961 100644 --- a/config/initializers/wrap_parameters.rb +++ b/config/initializers/wrap_parameters.rb @@ -1,5 +1,3 @@ -# frozen_string_literal: true - # Be sure to restart your server when you modify this file. # This file contains settings for ActionController::ParamsWrapper which @@ -9,3 +7,8 @@ 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/puma.rb b/config/puma.rb index 6ec461f5..b2102072 100644 --- a/config/puma.rb +++ b/config/puma.rb @@ -1,21 +1,22 @@ -# frozen_string_literal: true - # 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_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 } +port ENV.fetch("PORT") { 3000 } # Specifies the `environment` that Puma will run in. # -environment ENV.fetch('RAILS_ENV') { 'development' } +environment ENV.fetch("RAILS_ENV") { "development" } + +# Specifies the `pidfile` that Puma will use. +pidfile ENV.fetch("PIDFILE") { "tmp/pids/server.pid" } # Specifies the number of `workers` to boot in clustered mode. # Workers are forked webserver processes. If using threads and workers together @@ -28,31 +29,9 @@ # 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. +# process behavior so workers use less memory. # # 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 index aaa110c3..8fbfacdf 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -13,28 +13,29 @@ resources :tour_set_admins scope ':tenant' do - scope module: :v1, constraints: ApiVersion.new('v1') do - resources :tours, only: :index - end scope module: :v3, constraints: ApiVersion.new('v3', true) do + resources :tour_authors, path: 'tour-authors' resources :users - resources :modes, only: [:index] + resources :modes, only: [:index, :show] resources :tour_sets, path: 'tour-sets' resources :tour_set_admins, path: 'tour-set-users' resources :tour_collections, path: 'tour-collections' resources :tour_media, path: 'tour-media' + resources :map_overlays, path: 'map-overlays' + resources :map_icons, path: 'map-icons' resources :themes resources :tours resources :media resources :stops resources :stop_media, path: 'stop-media' resources :tour_media, path: 'tour-media' + resources :tour_modes, path: 'tour-modes' resources :tour_stops, path: 'tour-stops' resources :flat_pages, path: 'flat-pages' resources :tour_flat_pages, path: 'tour-flat-pages' resources :geojson_tours + end - post '/token', to: 'oauth2#create' - post '/revoke', to: 'oauth2#destroy' end + mount EcdsRailsAuthEngine::Engine, at: '/auth' end diff --git a/config/spring.rb b/config/spring.rb index ff5ba06b..9fa7863f 100644 --- a/config/spring.rb +++ b/config/spring.rb @@ -1,8 +1,6 @@ -# frozen_string_literal: true - -%w( +%w[ .ruby-version .rbenv-vars tmp/restart.txt tmp/caching-dev.txt -).each { |path| Spring.watch(path) } +].each { |path| Spring.watch(path) } diff --git a/config/storage.yml b/config/storage.yml new file mode 100644 index 00000000..0153a1c8 --- /dev/null +++ b/config/storage.yml @@ -0,0 +1,51 @@ +test: + service: Disk + root: <%= Rails.root.join("tmp/storage") %> + +local: + service: S3 + access_key_id: <%= Rails.application.credentials.dig(:s3Staging, :access_key_id) %> + secret_access_key: <%= Rails.application.credentials.dig(:s3Staging, :secret_access_key) %> + region: us-east-1 + bucket: otb-dev + +staging: + service: S3 + access_key_id: <%= Rails.application.credentials.dig(:s3Staging, :access_key_id) %> + secret_access_key: <%= Rails.application.credentials.dig(:s3Staging, :secret_access_key) %> + region: us-east-1 + bucket: opentour + +production: + service: S3 + access_key_id: <%= Rails.application.credentials.dig(:s3Staging, :access_key_id) %> + secret_access_key: <%= Rails.application.credentials.dig(:s3Staging, :secret_access_key) %> + region: us-east-1 + bucket: opentour-prod + +# 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/migrate/20200213152142_rename_metadescription.rb b/db/migrate/20200213152142_rename_metadescription.rb index e12ceab6..b9d3725d 100644 --- a/db/migrate/20200213152142_rename_metadescription.rb +++ b/db/migrate/20200213152142_rename_metadescription.rb @@ -1,5 +1,9 @@ class RenameMetadescription < ActiveRecord::Migration[5.2] def change - rename_column :stops, :metadescription, :meta_description + begin + rename_column :stops, :metadescription, :meta_description + rescue + # It's fine + end end end diff --git a/db/migrate/20210518143822_add_email.rb b/db/migrate/20210518143822_add_email.rb new file mode 100644 index 00000000..fcc835e7 --- /dev/null +++ b/db/migrate/20210518143822_add_email.rb @@ -0,0 +1,5 @@ +class AddEmail < ActiveRecord::Migration[5.2] + def change + add_column :users, :email, :string + end +end diff --git a/db/migrate/20210602201922_create_active_storage_tables.active_storage.rb b/db/migrate/20210602201922_create_active_storage_tables.active_storage.rb new file mode 100644 index 00000000..0b2ce257 --- /dev/null +++ b/db/migrate/20210602201922_create_active_storage_tables.active_storage.rb @@ -0,0 +1,27 @@ +# This migration comes from active_storage (originally 20170806125915) +class CreateActiveStorageTables < ActiveRecord::Migration[5.2] + def change + create_table :active_storage_blobs do |t| + t.string :key, null: false + t.string :filename, null: false + t.string :content_type + t.text :metadata + t.bigint :byte_size, null: false + t.string :checksum, null: false + t.datetime :created_at, null: false + + t.index [ :key ], unique: true + end + + create_table :active_storage_attachments do |t| + t.string :name, null: false + t.references :record, null: false, polymorphic: true, index: false + t.references :blob, null: false + + t.datetime :created_at, null: false + + t.index [ :record_type, :record_id, :name, :blob_id ], name: "index_active_storage_attachments_uniqueness", unique: true + t.foreign_key :active_storage_blobs, column: :blob_id + end + end +end diff --git a/db/migrate/20210604131202_add_base64.rb b/db/migrate/20210604131202_add_base64.rb new file mode 100644 index 00000000..11a7dcda --- /dev/null +++ b/db/migrate/20210604131202_add_base64.rb @@ -0,0 +1,5 @@ +class AddBase64 < ActiveRecord::Migration[6.0] + def change + add_column :media, :base64, :text + end +end diff --git a/db/migrate/20210604132602_add_video_provider.rb b/db/migrate/20210604132602_add_video_provider.rb new file mode 100644 index 00000000..1c408042 --- /dev/null +++ b/db/migrate/20210604132602_add_video_provider.rb @@ -0,0 +1,5 @@ +class AddVideoProvider < ActiveRecord::Migration[6.0] + def change + add_column :media, :video_provider, :integer, default: 0 + end +end diff --git a/db/migrate/20210604145226_change_base64.rb b/db/migrate/20210604145226_change_base64.rb new file mode 100644 index 00000000..80142312 --- /dev/null +++ b/db/migrate/20210604145226_change_base64.rb @@ -0,0 +1,5 @@ +class ChangeBase64 < ActiveRecord::Migration[6.0] + def change + rename_column :media, :base64, :base_sixty_four + end +end diff --git a/db/migrate/20210607125915_add_parking_address.rb b/db/migrate/20210607125915_add_parking_address.rb new file mode 100644 index 00000000..fd12fa45 --- /dev/null +++ b/db/migrate/20210607125915_add_parking_address.rb @@ -0,0 +1,5 @@ +class AddParkingAddress < ActiveRecord::Migration[6.0] + def change + add_column :stops, :parking_address, :string + end +end diff --git a/db/migrate/20210607165827_rename_content.rb b/db/migrate/20210607165827_rename_content.rb new file mode 100644 index 00000000..054747b2 --- /dev/null +++ b/db/migrate/20210607165827_rename_content.rb @@ -0,0 +1,5 @@ +class RenameContent < ActiveRecord::Migration[6.0] + def change + rename_column :flat_pages, :content, :body + end +end diff --git a/db/migrate/20210607221303_add_tour_to_stop_slugs.rb b/db/migrate/20210607221303_add_tour_to_stop_slugs.rb new file mode 100644 index 00000000..25244cb0 --- /dev/null +++ b/db/migrate/20210607221303_add_tour_to_stop_slugs.rb @@ -0,0 +1,5 @@ +class AddTourToStopSlugs < ActiveRecord::Migration[6.0] + def change + add_reference :stop_slugs, :tour, foreign_key: true + end +end diff --git a/db/migrate/20210608211705_create_map_overlays.rb b/db/migrate/20210608211705_create_map_overlays.rb new file mode 100644 index 00000000..c4ba6439 --- /dev/null +++ b/db/migrate/20210608211705_create_map_overlays.rb @@ -0,0 +1,14 @@ +class CreateMapOverlays < ActiveRecord::Migration[6.0] + def change + create_table :map_overlays do |t| + t.decimal "south", precision: 100, scale: 8 + t.decimal "north", precision: 100, scale: 8 + t.decimal "east", precision: 100, scale: 8 + t.decimal "west", precision: 100, scale: 8 + t.references :tour, null: true + t.references :stop, null: true + + t.timestamps + end + end +end diff --git a/db/migrate/20210609141823_add_base64_overlay.rb b/db/migrate/20210609141823_add_base64_overlay.rb new file mode 100644 index 00000000..31bb4a60 --- /dev/null +++ b/db/migrate/20210609141823_add_base64_overlay.rb @@ -0,0 +1,5 @@ +class AddBase64Overlay < ActiveRecord::Migration[6.0] + def change + add_column :map_overlays, :base_sixty_four, :text + end +end diff --git a/db/migrate/20210609142132_add_title_overlay.rb b/db/migrate/20210609142132_add_title_overlay.rb new file mode 100644 index 00000000..c531694d --- /dev/null +++ b/db/migrate/20210609142132_add_title_overlay.rb @@ -0,0 +1,5 @@ +class AddTitleOverlay < ActiveRecord::Migration[6.0] + def change + add_column :map_overlays, :title, :text + end +end diff --git a/db/migrate/20210610141706_add_enable_directions.rb b/db/migrate/20210610141706_add_enable_directions.rb new file mode 100644 index 00000000..1913657a --- /dev/null +++ b/db/migrate/20210610141706_add_enable_directions.rb @@ -0,0 +1,5 @@ +class AddEnableDirections < ActiveRecord::Migration[6.0] + def change + add_column :tours, :use_directions, :boolean, default: true + end +end diff --git a/db/migrate/20210610152819_add_default_lang.rb b/db/migrate/20210610152819_add_default_lang.rb new file mode 100644 index 00000000..d6532be1 --- /dev/null +++ b/db/migrate/20210610152819_add_default_lang.rb @@ -0,0 +1,5 @@ +class AddDefaultLang < ActiveRecord::Migration[6.0] + def change + add_column :tours, :default_lng, :integer, default: 0 + end +end diff --git a/db/migrate/20210610180023_add_icon_color.rb b/db/migrate/20210610180023_add_icon_color.rb new file mode 100644 index 00000000..652bbd67 --- /dev/null +++ b/db/migrate/20210610180023_add_icon_color.rb @@ -0,0 +1,5 @@ +class AddIconColor < ActiveRecord::Migration[6.0] + def change + add_column :stops, :icon_color, :string, default: '#D32F2F' + end +end diff --git a/db/migrate/20210610182827_create_map_icons.rb b/db/migrate/20210610182827_create_map_icons.rb new file mode 100644 index 00000000..dc200cea --- /dev/null +++ b/db/migrate/20210610182827_create_map_icons.rb @@ -0,0 +1,9 @@ +class CreateMapIcons < ActiveRecord::Migration[6.0] + def change + create_table :map_icons do |t| + t.text :base_sixty_four + + t.timestamps + end + end +end diff --git a/db/migrate/20210610183825_add_title_to_map_icons.rb b/db/migrate/20210610183825_add_title_to_map_icons.rb new file mode 100644 index 00000000..6aac4c6d --- /dev/null +++ b/db/migrate/20210610183825_add_title_to_map_icons.rb @@ -0,0 +1,6 @@ +class AddTitleToMapIcons < ActiveRecord::Migration[6.0] + def change + add_column :map_icons, :title, :string + add_reference :stops, :map_icon, null: true, foreign_key: true + end +end diff --git a/db/migrate/20210614140848_add_logo64_to_tour_sets.rb b/db/migrate/20210614140848_add_logo64_to_tour_sets.rb new file mode 100644 index 00000000..486a0d34 --- /dev/null +++ b/db/migrate/20210614140848_add_logo64_to_tour_sets.rb @@ -0,0 +1,5 @@ +class AddLogo64ToTourSets < ActiveRecord::Migration[6.0] + def change + add_column :tour_sets, :base_sixty_four, :text + end +end diff --git a/db/migrate/20210614154357_remove_logo_from_tour_sets.rb b/db/migrate/20210614154357_remove_logo_from_tour_sets.rb new file mode 100644 index 00000000..92e49e5f --- /dev/null +++ b/db/migrate/20210614154357_remove_logo_from_tour_sets.rb @@ -0,0 +1,5 @@ +class RemoveLogoFromTourSets < ActiveRecord::Migration[6.0] + def change + remove_column :tour_sets, :logo + end +end diff --git a/db/migrate/20210614154939_add_logo_title_to_tour_sets.rb b/db/migrate/20210614154939_add_logo_title_to_tour_sets.rb new file mode 100644 index 00000000..d034390b --- /dev/null +++ b/db/migrate/20210614154939_add_logo_title_to_tour_sets.rb @@ -0,0 +1,5 @@ +class AddLogoTitleToTourSets < ActiveRecord::Migration[6.0] + def change + add_column :tour_sets, :logo_title, :string + end +end diff --git a/db/migrate/20210702214539_add_file_urls_to_media.rb b/db/migrate/20210702214539_add_file_urls_to_media.rb new file mode 100644 index 00000000..e645d8c4 --- /dev/null +++ b/db/migrate/20210702214539_add_file_urls_to_media.rb @@ -0,0 +1,7 @@ +class AddFileUrlsToMedia < ActiveRecord::Migration[6.0] + def change + add_column :media, :mobile, :string + add_column :media, :tablet, :string + add_column :media, :desktop, :string + end +end diff --git a/db/migrate/20210706150527_add_filename_to_media.rb b/db/migrate/20210706150527_add_filename_to_media.rb new file mode 100644 index 00000000..125fe154 --- /dev/null +++ b/db/migrate/20210706150527_add_filename_to_media.rb @@ -0,0 +1,5 @@ +class AddFilenameToMedia < ActiveRecord::Migration[6.0] + def change + add_column :media, :filename, :string + end +end diff --git a/db/migrate/20210707161803_change_title_to_filename.rb b/db/migrate/20210707161803_change_title_to_filename.rb new file mode 100644 index 00000000..a2eaa124 --- /dev/null +++ b/db/migrate/20210707161803_change_title_to_filename.rb @@ -0,0 +1,6 @@ +class ChangeTitleToFilename < ActiveRecord::Migration[6.0] + def change + rename_column :map_icons, :title, :filename + rename_column :map_overlays, :title, :filename + end +end diff --git a/db/migrate/20210712234206_add_service_name_to_active_storage_blobs.active_storage.rb b/db/migrate/20210712234206_add_service_name_to_active_storage_blobs.active_storage.rb new file mode 100644 index 00000000..9967a132 --- /dev/null +++ b/db/migrate/20210712234206_add_service_name_to_active_storage_blobs.active_storage.rb @@ -0,0 +1,18 @@ +# This migration comes from active_storage (originally 20190112182829) +class AddServiceNameToActiveStorageBlobs < ActiveRecord::Migration[6.0] + def up + unless column_exists?(:active_storage_blobs, :service_name) + add_column :active_storage_blobs, :service_name, :string + + if configured_service = ActiveStorage::Blob.service.name + ActiveStorage::Blob.unscoped.update_all(service_name: configured_service) + end + + change_column :active_storage_blobs, :service_name, :string, null: false + end + end + + def down + remove_column :active_storage_blobs, :service_name + end +end diff --git a/db/migrate/20210712234207_create_active_storage_variant_records.active_storage.rb b/db/migrate/20210712234207_create_active_storage_variant_records.active_storage.rb new file mode 100644 index 00000000..a2862695 --- /dev/null +++ b/db/migrate/20210712234207_create_active_storage_variant_records.active_storage.rb @@ -0,0 +1,12 @@ +# This migration comes from active_storage (originally 20191206030411) +class CreateActiveStorageVariantRecords < ActiveRecord::Migration[6.0] + def change + create_table :active_storage_variant_records do |t| + t.belongs_to :blob, null: false, index: false + t.string :variation_digest, null: false + + t.index %i[ blob_id variation_digest ], name: "index_active_storage_variant_records_uniqueness", unique: true + t.foreign_key :active_storage_blobs, column: :blob_id + end + end +end diff --git a/db/migrate/20210816124019_add_link_to_tour.rb b/db/migrate/20210816124019_add_link_to_tour.rb new file mode 100644 index 00000000..f1b74eb9 --- /dev/null +++ b/db/migrate/20210816124019_add_link_to_tour.rb @@ -0,0 +1,6 @@ +class AddLinkToTour < ActiveRecord::Migration[6.1] + def change + add_column :tours, :link_address, :string + add_column :tours, :link_text, :string + end +end diff --git a/db/migrate/20210831190245_change_id_types.rb b/db/migrate/20210831190245_change_id_types.rb new file mode 100644 index 00000000..8ad84f5f --- /dev/null +++ b/db/migrate/20210831190245_change_id_types.rb @@ -0,0 +1,6 @@ +class ChangeIdTypes < ActiveRecord::Migration[6.1] + def change + change_column :tours, :mode_id, :bigint + change_column :tours, :splash_image_medium_id, :bigint + end +end diff --git a/db/migrate/20210831200851_change_body_type_for_flat_pages.rb b/db/migrate/20210831200851_change_body_type_for_flat_pages.rb new file mode 100644 index 00000000..d26feaa4 --- /dev/null +++ b/db/migrate/20210831200851_change_body_type_for_flat_pages.rb @@ -0,0 +1,5 @@ +class ChangeBodyTypeForFlatPages < ActiveRecord::Migration[6.1] + def change + change_column :flat_pages, :body, :text + end +end diff --git a/db/migrate/20210831202533_drop_tag_tables.rb b/db/migrate/20210831202533_drop_tag_tables.rb new file mode 100644 index 00000000..7ab5a076 --- /dev/null +++ b/db/migrate/20210831202533_drop_tag_tables.rb @@ -0,0 +1,7 @@ +class DropTagTables < ActiveRecord::Migration[6.1] + def change + drop_table :tour_tags + drop_table :stop_tags + drop_table :tags + end +end diff --git a/db/migrate/20210902145305_add_widths_to_medium.rb b/db/migrate/20210902145305_add_widths_to_medium.rb new file mode 100644 index 00000000..f37112ea --- /dev/null +++ b/db/migrate/20210902145305_add_widths_to_medium.rb @@ -0,0 +1,5 @@ +class AddWidthsToMedium < ActiveRecord::Migration[6.1] + def change + add_column :media, :lqip_width, :integer + end +end diff --git a/db/migrate/20210902164843_fix_slug_id.rb b/db/migrate/20210902164843_fix_slug_id.rb new file mode 100644 index 00000000..8e117d92 --- /dev/null +++ b/db/migrate/20210902164843_fix_slug_id.rb @@ -0,0 +1,6 @@ +class FixSlugId < ActiveRecord::Migration[6.1] + def change + remove_column :slugs, :id + add_column :slugs, :id, :primary_key + end +end diff --git a/db/migrate/20210916170901_change_lat_lng_type.rb b/db/migrate/20210916170901_change_lat_lng_type.rb new file mode 100644 index 00000000..2982d935 --- /dev/null +++ b/db/migrate/20210916170901_change_lat_lng_type.rb @@ -0,0 +1,12 @@ +class ChangeLatLngType < ActiveRecord::Migration[6.1] + def change + change_column :map_overlays, :south, :string + change_column :map_overlays, :north, :string + change_column :map_overlays, :east, :string + change_column :map_overlays, :west, :string + change_column :stops, :lat, :string + change_column :stops, :lng, :string + change_column :stops, :parking_lat, :string + change_column :stops, :parking_lng, :string + end +end diff --git a/db/migrate/20211011120714_add_duration.rb b/db/migrate/20211011120714_add_duration.rb new file mode 100644 index 00000000..89f8f8dc --- /dev/null +++ b/db/migrate/20211011120714_add_duration.rb @@ -0,0 +1,6 @@ +class AddDuration < ActiveRecord::Migration[6.1] + def change + add_column :tours, :duration, :integer + add_column :tours, :saved_stop_order, :integer, array: true + end +end diff --git a/db/migrate/20220207133352_add_over_bounds_restrition_to_tour.rb b/db/migrate/20220207133352_add_over_bounds_restrition_to_tour.rb new file mode 100644 index 00000000..3c5d38dc --- /dev/null +++ b/db/migrate/20220207133352_add_over_bounds_restrition_to_tour.rb @@ -0,0 +1,5 @@ +class AddOverBoundsRestritionToTour < ActiveRecord::Migration[6.1] + def change + add_column :tours, :restrict_bounds_to_overlay, :boolean, default: false + end +end diff --git a/db/migrate/20220208172935_add_blank_map_to_tour.rb b/db/migrate/20220208172935_add_blank_map_to_tour.rb new file mode 100644 index 00000000..a3932223 --- /dev/null +++ b/db/migrate/20220208172935_add_blank_map_to_tour.rb @@ -0,0 +1,5 @@ +class AddBlankMapToTour < ActiveRecord::Migration[6.1] + def change + add_column :tours, :blank_map, :boolean, default: false + end +end diff --git a/db/migrate/20220210160507_set_geo_default.rb b/db/migrate/20220210160507_set_geo_default.rb new file mode 100644 index 00000000..6b30c1ed --- /dev/null +++ b/db/migrate/20220210160507_set_geo_default.rb @@ -0,0 +1,5 @@ +class SetGeoDefault < ActiveRecord::Migration[6.1] + def change + change_column :tours, :is_geo, :boolean, default: true + end +end diff --git a/db/migrate/20220211142554_add_restrict_bounds_to_tour.rb b/db/migrate/20220211142554_add_restrict_bounds_to_tour.rb new file mode 100644 index 00000000..2e6eecb9 --- /dev/null +++ b/db/migrate/20220211142554_add_restrict_bounds_to_tour.rb @@ -0,0 +1,5 @@ +class AddRestrictBoundsToTour < ActiveRecord::Migration[6.1] + def change + add_column :tours, :restrict_bounds, :boolean, default: true + end +end diff --git a/db/migrate/20220728131300_add_term_accepted_to_user.rb.rb b/db/migrate/20220728131300_add_term_accepted_to_user.rb.rb new file mode 100644 index 00000000..314a1205 --- /dev/null +++ b/db/migrate/20220728131300_add_term_accepted_to_user.rb.rb @@ -0,0 +1,5 @@ +class AddTermsAcceptedToUsers < ActiveRecord::Migration[6.1] + def change + add_column :users, :terms_accepted, :boolean, default: false + end +end diff --git a/db/schema.rb b/db/schema.rb index df0d600a..9b66af81 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -2,24 +2,68 @@ # 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). +# This file is the source Rails uses to define your schema when running `bin/rails +# db:schema:load`. When creating a new database, `bin/rails db:schema:load` tends to +# be faster and is potentially less error prone than running all of your +# migrations from scratch. Old migrations may fail to apply correctly if those +# migrations use external dependencies or application code. # # It's strongly recommended that you check this file into your version control system. -ActiveRecord::Schema.define(version: 2020_02_13_152142) do +ActiveRecord::Schema.define(version: 2022_08_01_155837) do # These are extensions that must be enabled in order to support this database - # enable_extension "pgcrypto" - # enable_extension "plpgsql" - # enable_extension "uuid-ossp" + enable_extension "pgcrypto" + enable_extension "plpgsql" + enable_extension "uuid-ossp" + + create_table "active_storage_attachments", force: :cascade do |t| + t.string "name", null: false + t.string "record_type", null: false + t.bigint "record_id", null: false + t.bigint "blob_id", null: false + t.datetime "created_at", null: false + t.index ["blob_id"], name: "index_active_storage_attachments_on_blob_id" + t.index ["record_type", "record_id", "name", "blob_id"], name: "index_active_storage_attachments_uniqueness", unique: true + end + + create_table "active_storage_blobs", force: :cascade do |t| + t.string "key", null: false + t.string "filename", null: false + t.string "content_type" + t.text "metadata" + t.bigint "byte_size", null: false + t.string "checksum", null: false + t.datetime "created_at", null: false + t.string "service_name", null: false + t.index ["key"], name: "index_active_storage_blobs_on_key", unique: true + end + + create_table "active_storage_variant_records", force: :cascade do |t| + t.bigint "blob_id", null: false + t.string "variation_digest", null: false + t.index ["blob_id", "variation_digest"], name: "index_active_storage_variant_records_uniqueness", unique: true + end + + create_table "ecds_rails_auth_engine_logins", force: :cascade do |t| + t.string "who" + t.string "provider" + t.bigint "user_id" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + t.index ["user_id"], name: "index_ecds_rails_auth_engine_logins_on_user_id" + end + + create_table "ecds_rails_auth_engine_tokens", force: :cascade do |t| + t.string "token" + t.bigint "login_id" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + end create_table "flat_pages", force: :cascade do |t| t.string "title" - t.text "content" + t.text "body" t.datetime "created_at", null: false t.datetime "updated_at", null: false t.integer "position" @@ -39,6 +83,28 @@ t.index ["user_id"], name: "index_logins_on_user_id" end + create_table "map_icons", force: :cascade do |t| + t.text "base_sixty_four" + t.datetime "created_at", precision: 6, null: false + t.datetime "updated_at", precision: 6, null: false + t.string "filename" + end + + create_table "map_overlays", force: :cascade do |t| + t.string "south" + t.string "north" + t.string "east" + t.string "west" + t.bigint "tour_id" + t.bigint "stop_id" + t.datetime "created_at", precision: 6, null: false + t.datetime "updated_at", precision: 6, null: false + t.text "base_sixty_four" + t.text "filename" + t.index ["stop_id"], name: "index_map_overlays_on_stop_id" + t.index ["tour_id"], name: "index_map_overlays_on_tour_id" + end + create_table "media", force: :cascade do |t| t.string "title" t.text "caption" @@ -54,6 +120,13 @@ t.integer "tablet_height" t.integer "mobile_width" t.integer "mobile_height" + t.text "base_sixty_four" + t.integer "video_provider", default: 0 + t.string "mobile" + t.string "tablet" + t.string "desktop" + t.string "filename" + t.integer "lqip_width" end create_table "modes", force: :cascade do |t| @@ -90,13 +163,9 @@ t.bigint "stop_id" t.datetime "created_at", null: false t.datetime "updated_at", null: false + t.bigint "tour_id" t.index ["stop_id"], name: "index_stop_slugs_on_stop_id" - end - - create_table "stop_tags", force: :cascade do |t| - t.string "title" - t.datetime "created_at", null: false - t.datetime "updated_at", null: false + t.index ["tour_id"], name: "index_stop_slugs_on_tour_id" end create_table "stops", force: :cascade do |t| @@ -106,16 +175,20 @@ t.string "article_link" t.string "video_embed" t.string "video_poster" - t.decimal "lat", precision: 65, scale: 8 - t.decimal "lng", precision: 65, scale: 8 - t.decimal "parking_lat", precision: 65, scale: 8 - t.decimal "parking_lng", precision: 65, scale: 8 + t.string "lat" + t.string "lng" + t.string "parking_lat" + t.string "parking_lng" t.text "direction_intro" t.text "direction_notes" t.datetime "created_at", null: false t.datetime "updated_at", null: false t.string "address" t.bigint "medium_id" + t.string "parking_address" + t.string "icon_color", default: "#D32F2F" + t.bigint "map_icon_id" + t.index ["map_icon_id"], name: "index_stops_on_map_icon_id" t.index ["medium_id"], name: "index_stops_on_medium_id" end @@ -138,12 +211,6 @@ t.index ["tagger_id"], name: "index_taggings_on_tagger_id" end - create_table "tags", id: :serial, force: :cascade do |t| - t.string "name" - t.integer "taggings_count", default: 0 - t.index ["name"], name: "index_tags_on_name", unique: true - end - create_table "themes", force: :cascade do |t| t.string "title" t.datetime "created_at", null: false @@ -210,8 +277,9 @@ t.bigint "tour_id" t.string "external_url" t.text "notes" - t.string "logo" t.string "footer_logo" + t.text "base_sixty_four" + t.string "logo_title" t.index ["tour_id"], name: "index_tour_sets_on_tours_id" end @@ -225,20 +293,12 @@ t.index ["tour_id"], name: "index_tour_stops_on_tour_id" end - create_table "tour_tags", force: :cascade do |t| - t.string "title" - t.decimal "lat", precision: 65, scale: 8 - t.decimal "lng", precision: 65, scale: 8 - t.datetime "created_at", null: false - t.datetime "updated_at", null: false - end - create_table "tours", force: :cascade do |t| t.string "title" t.text "description" t.text "article_link" t.text "google_analytics" - t.boolean "is_geo" + t.boolean "is_geo", default: true t.boolean "published" t.bigint "theme_id" t.datetime "created_at", null: false @@ -249,6 +309,15 @@ t.string "meta_description" t.bigint "medium_id" t.string "map_type" + t.boolean "use_directions", default: true + t.integer "default_lng", default: 0 + t.string "link_address" + t.string "link_text" + t.integer "duration" + t.integer "saved_stop_order", array: true + t.boolean "restrict_bounds_to_overlay", default: false + t.boolean "blank_map", default: false + t.boolean "restrict_bounds", default: true t.index ["medium_id"], name: "index_tours_on_medium_id" t.index ["mode_id"], name: "index_tours_on_mode_id" t.index ["theme_id"], name: "index_tours_on_theme_id" @@ -260,9 +329,15 @@ t.datetime "created_at", null: false t.datetime "updated_at", null: false t.boolean "super", default: false + t.string "email" + t.boolean "terms_accepted", default: false t.index ["login_id"], name: "index_users_on_login_id", unique: true end + add_foreign_key "active_storage_attachments", "active_storage_blobs", column: "blob_id" + add_foreign_key "active_storage_variant_records", "active_storage_blobs", column: "blob_id" + add_foreign_key "stop_slugs", "tours" + add_foreign_key "stops", "map_icons" add_foreign_key "stops", "media" add_foreign_key "tour_set_admins", "roles" add_foreign_key "tour_sets", "tours" diff --git a/db/seeds.rb b/db/seeds.rb index 55e857ab..ac21c9ef 100644 --- a/db/seeds.rb +++ b/db/seeds.rb @@ -47,8 +47,8 @@ 4.times do # User.create(display_name: Faker::Movies::Lebowski) - FactoryBot.create(:login) - Login.last.user.update_attribute(:display_name, Faker::Movies::Lebowski) + FactoryBot.create(:user) + # Login.last.user.update_attribute(:display_name, Faker::Movies::Lebowski) end TourSet.all.each do |ts| @@ -73,6 +73,7 @@ end User.all.each do |u| + FactoryBot.create(:login, who: u.email, user_id: u.id, provider: 'earth') next if u.super next if u.tours.present? # u.tours = Tour.all.order(Arel.sql('random()')).limit(Random.new.rand(2..3)) diff --git a/entrypoint.sh b/entrypoint.sh new file mode 100755 index 00000000..71261527 --- /dev/null +++ b/entrypoint.sh @@ -0,0 +1,4 @@ + +/usr/local/bin/bundle exec rake db:migrate + +/usr/local/bin/bundle exec rails server -b 0.0.0.0 \ No newline at end of file diff --git a/lib/snippets.rb b/lib/snippets.rb new file mode 100644 index 00000000..4cef4160 --- /dev/null +++ b/lib/snippets.rb @@ -0,0 +1,273 @@ +sites = TourSet.all.map(&:subdir) + +sites.each do |ts| + Apartment::Tenant.switch! ts + reload! + ids = Medium.all.map(&:id) + ids.each do |id| + # Apartment::Tenant.switch! ts + m = Medium.find(id) + next if m.video.nil? + case m.provider + when 'youtube' + m.update(embed: "//www.youtube.com/embed/#{m.video}", provider: 'youtube') + when 'vimeo' + m.update(embed: "//player.vimeo.com/video/#{m.video}", video_provider: 'vimeo') + end + end +end + +TourSet.all.each do |ts| + puts ts.subdir + Apartment::Tenant.switch! ts.subdir + Medium.all.each do |medium| + if !medium.file.attached? + # medium.delete + puts "#{medium.id} stops: #{medium.stops.count} tours: #{medium.tours.count}" + end + end +end + +Medium.all.each do |m| + if !m.file.attached? and (!m.tours.empty? or !m.stops.empty?) + print m.id + end +end + +# Medium.all.each do |m| + # if File.exist?(m.original_image.path) && !m.file.attached? + # m.file.attach(io: File.open(m.original_image.path), filename: m.original_image.path.split('/').last) +# else +# m.delete +# end +# end + +sites = TourSet.all.map(&:subdir) + +sites.each do |ts| + Apartment::Tenant.switch! ts + reload! + ids = Medium.all.map(&:id) + ids.each do |id| + Apartment::Tenant.switch! ts + m = Medium.find(id) + next if m.file.attached? + if m.original_image.path && File.exist?(m.original_image.path) + m.file.attach( + io: File.open(m.original_image.path), + filename: m.original_image.path.split('/').last, + content_type: m.original_image.content_type + ) + end + end +end + +# sites = TourSet.all.map(&:subdir) +# sites.each do |ts| +# Apartment::Tenant.switch! ts +# reload! +# ids = Medium.all.map(&:id) +# ids.each do |id| +# m = Medium.find(id) +# next if m.file.attached? +# if m.original_image.path && File.exist?(m.original_image.path) +# m.file.attach( +# io: File.open(m.original_image.path), +# filename: m.original_image.path.split('/').last, +# content_type: m.original_image.content_type +# ) +# end +# end +# end + +# IMPORTANT!!!!!! comment out the hooks on the medium model +require 'open-uri' +sites = TourSet.all.map(&:subdir) +sites.each do |ts| + Apartment::Tenant.switch! ts + ids = Medium.all.map(&:id) + ids.each do |id| + m = Medium.find(id) + next if m.original_image.nil? + next if m.file.attached? + begin + puts "https://api.opentour.emory.edu/uploads/#{ts}/#{m.original_image}" + m.file.purge + m.file.attach( + io: URI.open("https://api.opentour.emory.edu/uploads/#{ts}/#{m.original_image}"), + filename: m.original_image + ) + m.filename = m.original_image + m.save! + puts "https://api.opentour.emory.edu/#{ts}/#{m.original_image}" + rescue OpenURI::HTTPError, URI::InvalidURIError + puts 'dang' + end + end +end + +media = [{"id":1,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16315528664.png"},{"id":2,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/315299464.jpg"},{"id":4,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16315585158.jpeg"},{"id":11,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365689809.jpeg"},{"id":8,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16353452580.png"},{"id":9,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16353452967.png"},{"id":15,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365704670.jpeg"},{"id":19,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365748166.jpeg"},{"id":16,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365706674.jpeg"},{"id":17,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365708608.jpeg"},{"id":3,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16315583885.jpeg"},{"id":5,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16315587373.png"},{"id":6,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16341496140.jpeg"},{"id":14,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365694188.png"},{"id":10,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16353454287.jpeg"},{"id":7,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16341498295.png"},{"id":18,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365709114.jpeg"},{"id":12,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365693051.jpeg"},{"id":13,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365693188.jpeg"},{"id":22,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/671271790.jpg"},{"id":21,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/671271790.jpg"},{"id":23,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16442803933.png"},{"id":20,"url":"https://api.opentour.emory.edu/uploads/middle-passage-markers/16365785848.jpeg"}] +ids = media.map {|m| m[:id]} +ids = Medium.all.map(&:id) +ids.each do |id| + Apartment::Tenant.switch! 'battle-of-atlanta' + m = Medium.find(id) + m.file.purge + # next if m.file.attached? + # if m.original_image.path && File.exist?(m.original_image.path) + puts "https://api.opentour.emory.edu#{m.original_image.url}" + m.file.attach( + io: URI.open("https://api.opentour.emory.edu#{m.original_image.url}"), + filename: m.original_image.url.split('/').last, + content_type: m.original_image.content_type + ) + puts m.original_image.url.split('/').last + m.filename = m.original_image.url.split('/').last + m.save! + puts m.file.attached? + # end +end + + +sites = TourSet.all.map(&:subdir) +sites.each do |ts| + Apartment::Tenant.switch! ts + reload! + ids = Medium.all.map(&:id) + ids.each do |id| + Apartment::Tenant.switch! ts + m = Medium.find(id) + m.file.purge if m.file.attached? + m.delete + end + TourMedium.all.each {|tm| tm.delete} + StopMedium.all.each {|tm| tm.delete} +end + +ActiveStorage::Blob.service.send(:path_for, m.file.key) +Apartment::Tenant.switch! 'july-22nd' +ids = Medium.all.map(&:id) +ids.each do |id| + Apartment::Tenant.switch! 'july-22nd' + m = Medium.find(id) + + next if m.file.attached? + + # next unless m.file.attached? + + next unless File.exists? ActiveStorage::Blob.service.send(:path_for, m.file.key) +Apartment::Tenant.switch! 'july-22nd' + m.file.attach( + io: File.open(ActiveStorage::Blob.service.send(:path_for, m.file.key)), + filename: m.file.filename.to_s, + content_type: m.file.content_type + ) + Apartment::Tenant.switch! 'july-22nd' +end + +Apartment::Tenant.switch! 'july-22nd' +ids = MapIcon.all.map(&:id) +ids.each do |id| + Apartment::Tenant.switch! 'july-22nd' + m = MapIcon.find(id) + next unless m.file.attached? + + next unless File.exists? ActiveStorage::Blob.service.send(:path_for, m.file.key) + + m.file.attach( + io: File.open(ActiveStorage::Blob.service.send(:path_for, m.file.key)), + filename: m.file.filename.to_s, + content_type: m.file.content_type + ) +end + +User.all.each do |u| + login = Login.find_by(user_id: u.id) + u.email = login.identification + u.save +end + +sites = TourSet.all.map(&:subdir) + +sites.each do |ts| + puts ts + Apartment::Tenant.switch! ts + Tour.all.each do |t| + puts t.title + if t.bounds + t.is_geo = true + t.save + end + end +end + +sites = TourSet.all.map(&:subdir) + +sites.each do |ts| + puts ts + Apartment::Tenant.switch! ts + Medium.all.each do |m| + begin + next unless m.public_send("#{ts.underscore}_file").present? + next unless m.public_send("#{ts.underscore}_file").attached? + + m.public_send("#{ts.underscore}_file").purge + rescue NoMethodError => error + end + end +end + + +sites = TourSet.all.map(&:subdir) +sites.each do |ts| + Apartment::Tenant.switch! ts + reload! + ids = Medium.all.map(&:id) + ids.each do |id| + m = Medium.find(id) + m.save + end + end +end + +sites = TourSet.all.map(&:subdir) + +sites.each do |ts| + puts ts + Apartment::Tenant.switch! ts + Stop.all.each do |s| + if s.lat + s.update(lat: s.lat.round(5).to_f, lng: s.lng.round(5).to_f) + end + if s.parking_lat + s.update(parking_lat: s.parking_lat.round(5).to_f, parking_lng: s.parking_lng.round(5).to_f) + end + s.save + end +end + +Medium.all.each do |m| + next unless m.desktop_width.nil? + m.save +end + +require 'open-uri' +sites = TourSet.all.map(&:subdir) +sites.each do |ts| + Apartment::Tenant.switch! ts + Medium.all.each do |m| + m.save + m.files.keys.each { |k| URI.open(m.files[k]) } + end +end + +sites = TourSet.all.map(&:subdir) +sites.each do |ts| + Apartment::Tenant.switch! ts + Tour.all.each { |t| t.update(is_geo: true) } +end + +sites.each do |ts| + Apartment::Tenant.switch! ts + Tour.all.each { |t| t.update(restrict_bounds: false) } +end \ No newline at end of file diff --git a/lib/tasks/db_enhancements.rake.bk b/lib/tasks/db_enhancements.rake similarity index 59% rename from lib/tasks/db_enhancements.rake.bk rename to lib/tasks/db_enhancements.rake index 9e7509ae..e3ac5d84 100644 --- a/lib/tasks/db_enhancements.rake.bk +++ b/lib/tasks/db_enhancements.rake @@ -13,14 +13,14 @@ namespace :db do desc 'Also create shared_extensions Schema' task extensions: :environment do - # Create Schema - ActiveRecord::Base.connection.execute 'CREATE SCHEMA IF NOT EXISTS shared_extensions;' - # Enable pgcrypto - ActiveRecord::Base.connection.execute 'CREATE EXTENSION IF NOT EXISTS "pgcrypto" SCHEMA shared_extensions;' - # Enable UUID-OSSP - ActiveRecord::Base.connection.execute 'CREATE EXTENSION IF NOT EXISTS "uuid-ossp" SCHEMA shared_extensions;' - # Grant usage to public - ActiveRecord::Base.connection.execute 'GRANT usage ON SCHEMA shared_extensions to public;' + # # Create Schema + # ActiveRecord::Base.connection.execute 'CREATE SCHEMA IF NOT EXISTS shared_extensions;' + # # Enable pgcrypto + # ActiveRecord::Base.connection.execute 'CREATE EXTENSION IF NOT EXISTS "pgcrypto" SCHEMA shared_extensions;' + # # Enable UUID-OSSP + # ActiveRecord::Base.connection.execute 'CREATE EXTENSION IF NOT EXISTS "uuid-ossp" SCHEMA shared_extensions;' + # # Grant usage to public + # ActiveRecord::Base.connection.execute 'GRANT usage ON SCHEMA shared_extensions to public;' end end diff --git a/lib/vimeo_props.rb b/lib/vimeo_props.rb deleted file mode 100644 index 220b9b40..00000000 --- a/lib/vimeo_props.rb +++ /dev/null @@ -1,19 +0,0 @@ -class VimeoProps - include HTTParty - include Nokogiri - def initialize(video) - @video = video - end - - def id - if @video.include? 'iframe' - Nokogiri::HTML(@video).xpath('//iframe')[0]['src'].split('/')[-1] - else - /\d{9}/.match(@video)[0] - end - end - - def image - HTTParty.get('https://vimeo.com/api/oembed.json?url=https%3A//vimeo.com/#{self.id}')['thumbnail_url'] - end -end diff --git a/lib/youtube_props.rb b/lib/youtube_props.rb deleted file mode 100644 index 2d9526be..00000000 --- a/lib/youtube_props.rb +++ /dev/null @@ -1,16 +0,0 @@ -class YouTubeProps - include YouTubeRails - include Yt - - def initialize(video) - @video = video - end - - def id - YouTubeRails.extract_video_id(@video) - end - - def image - Yt::Video.new(id:id).thumbnail_url('standard') - end -end diff --git a/old_spcs/requests/map_icons_spec.rb b/old_spcs/requests/map_icons_spec.rb new file mode 100644 index 00000000..9acc699b --- /dev/null +++ b/old_spcs/requests/map_icons_spec.rb @@ -0,0 +1,127 @@ +require 'rails_helper' + +# This spec was generated by rspec-rails when you ran the scaffold generator. +# It demonstrates how one might use RSpec to test the controller code that +# was generated by Rails when you ran the scaffold generator. +# +# It assumes that the implementation code is generated by the rails scaffold +# generator. If you are using any extension libraries to generate different +# controller code, this generated spec may or may not pass. +# +# It only uses APIs available in rails and/or rspec-rails. There are a number +# of tools you can use to make these specs even more expressive, but we're +# sticking to rails and rspec-rails APIs to keep things simple and stable. + +RSpec.describe "/map_icons", type: :request do + # This should return the minimal set of attributes required to create a valid + # MapIcon. As you add validations to MapIcon, be sure to + # adjust the attributes here as well. + let(:valid_attributes) { + skip("Add a hash of attributes valid for your model") + } + + let(:invalid_attributes) { + skip("Add a hash of attributes invalid for your model") + } + + # This should return the minimal set of values that should be in the headers + # in order to pass any filters (e.g. authentication) defined in + # MapIconsController, or in your router and rack + # middleware. Be sure to keep this updated too. + let(:valid_headers) { + {} + } + + describe "GET /index" do + it "renders a successful response" do + MapIcon.create! valid_attributes + get map_icons_url, headers: valid_headers, as: :json + expect(response).to be_successful + end + end + + describe "GET /show" do + it "renders a successful response" do + map_icon = MapIcon.create! valid_attributes + get map_icon_url(map_icon), as: :json + expect(response).to be_successful + end + end + + # describe "POST /create" do + # context "with valid parameters" do + # it "creates a new MapIcon" do + # expect { + # post map_icons_url, + # params: { map_icon: valid_attributes }, headers: valid_headers, as: :json + # }.to change(MapIcon, :count).by(1) + # end + + # it "renders a JSON response with the new map_icon" do + # post map_icons_url, + # params: { map_icon: valid_attributes }, headers: valid_headers, as: :json + # expect(response).to have_http_status(:created) + # expect(response.content_type).to match(a_string_including("application/json")) + # end + # end + + # context "with invalid parameters" do + # it "does not create a new MapIcon" do + # expect { + # post map_icons_url, + # params: { map_icon: invalid_attributes }, as: :json + # }.to change(MapIcon, :count).by(0) + # end + + # it "renders a JSON response with errors for the new map_icon" do + # post map_icons_url, + # params: { map_icon: invalid_attributes }, headers: valid_headers, as: :json + # expect(response).to have_http_status(:unprocessable_entity) + # expect(response.content_type).to eq("application/json") + # end + # end + # end + + describe "PATCH /update" do + context "with valid parameters" do + let(:new_attributes) { + skip("Add a hash of attributes valid for your model") + } + + it "updates the requested map_icon" do + map_icon = MapIcon.create! valid_attributes + patch map_icon_url(map_icon), + params: { map_icon: new_attributes }, headers: valid_headers, as: :json + map_icon.reload + skip("Add assertions for updated state") + end + + it "renders a JSON response with the map_icon" do + map_icon = MapIcon.create! valid_attributes + patch map_icon_url(map_icon), + params: { map_icon: new_attributes }, headers: valid_headers, as: :json + expect(response).to have_http_status(:ok) + expect(response.content_type).to match(a_string_including("application/json")) + end + end + + context "with invalid parameters" do + it "renders a JSON response with errors for the map_icon" do + map_icon = MapIcon.create! valid_attributes + patch map_icon_url(map_icon), + params: { map_icon: invalid_attributes }, headers: valid_headers, as: :json + expect(response).to have_http_status(:unprocessable_entity) + expect(response.content_type).to eq("application/json") + end + end + end + + describe "DELETE /destroy" do + it "destroys the requested map_icon" do + map_icon = MapIcon.create! valid_attributes + expect { + delete map_icon_url(map_icon), headers: valid_headers, as: :json + }.to change(MapIcon, :count).by(-1) + end + end +end diff --git a/old_spcs/requests/tour_authors_spec.fix b/old_spcs/requests/tour_authors_spec.fix new file mode 100644 index 00000000..7eb11eef --- /dev/null +++ b/old_spcs/requests/tour_authors_spec.fix @@ -0,0 +1,127 @@ +require 'rails_helper' + +# This spec was generated by rspec-rails when you ran the scaffold generator. +# It demonstrates how one might use RSpec to test the controller code that +# was generated by Rails when you ran the scaffold generator. +# +# It assumes that the implementation code is generated by the rails scaffold +# generator. If you are using any extension libraries to generate different +# controller code, this generated spec may or may not pass. +# +# It only uses APIs available in rails and/or rspec-rails. There are a number +# of tools you can use to make these specs even more expressive, but we're +# sticking to rails and rspec-rails APIs to keep things simple and stable. + +RSpec.describe "/tour_authors", type: :request do + # This should return the minimal set of attributes required to create a valid + # TourAuthor. As you add validations to TourAuthor, be sure to + # adjust the attributes here as well. + let(:valid_attributes) { + skip("Add a hash of attributes valid for your model") + } + + let(:invalid_attributes) { + skip("Add a hash of attributes invalid for your model") + } + + # This should return the minimal set of values that should be in the headers + # in order to pass any filters (e.g. authentication) defined in + # TourAuthorsController, or in your router and rack + # middleware. Be sure to keep this updated too. + let(:valid_headers) { + {} + } + + describe "GET /index" do + it "renders a successful response" do + TourAuthor.create! valid_attributes + get tour_authors_url, headers: valid_headers, as: :json + expect(response).to be_successful + end + end + + describe "GET /show" do + it "renders a successful response" do + tour_author = TourAuthor.create! valid_attributes + get tour_author_url(tour_author), as: :json + expect(response).to be_successful + end + end + + describe "POST /create" do + context "with valid parameters" do + it "creates a new TourAuthor" do + expect { + post tour_authors_url, + params: { tour_author: valid_attributes }, headers: valid_headers, as: :json + }.to change(TourAuthor, :count).by(1) + end + + it "renders a JSON response with the new tour_author" do + post tour_authors_url, + params: { tour_author: valid_attributes }, headers: valid_headers, as: :json + expect(response).to have_http_status(:created) + expect(response.content_type).to match(a_string_including("application/json")) + end + end + + context "with invalid parameters" do + it "does not create a new TourAuthor" do + expect { + post tour_authors_url, + params: { tour_author: invalid_attributes }, as: :json + }.to change(TourAuthor, :count).by(0) + end + + it "renders a JSON response with errors for the new tour_author" do + post tour_authors_url, + params: { tour_author: invalid_attributes }, headers: valid_headers, as: :json + expect(response).to have_http_status(:unprocessable_entity) + expect(response.content_type).to eq("application/json") + end + end + end + + describe "PATCH /update" do + context "with valid parameters" do + let(:new_attributes) { + skip("Add a hash of attributes valid for your model") + } + + it "updates the requested tour_author" do + tour_author = TourAuthor.create! valid_attributes + patch tour_author_url(tour_author), + params: { tour_author: new_attributes }, headers: valid_headers, as: :json + tour_author.reload + skip("Add assertions for updated state") + end + + it "renders a JSON response with the tour_author" do + tour_author = TourAuthor.create! valid_attributes + patch tour_author_url(tour_author), + params: { tour_author: new_attributes }, headers: valid_headers, as: :json + expect(response).to have_http_status(:ok) + expect(response.content_type).to match(a_string_including("application/json")) + end + end + + context "with invalid parameters" do + it "renders a JSON response with errors for the tour_author" do + tour_author = TourAuthor.create! valid_attributes + patch tour_author_url(tour_author), + params: { tour_author: invalid_attributes }, headers: valid_headers, as: :json + expect(response).to have_http_status(:unprocessable_entity) + expect(response.content_type).to eq("application/json") + end + end + end + + describe "DELETE /destroy" do + it "destroys the requested tour_author" do + tour_author = TourAuthor.create! valid_attributes + expect { + delete tour_author_url(tour_author), headers: valid_headers, as: :json + }.to change(TourAuthor, :count).by(-1) + end + end +end diff --git a/spec/requests/v3/abalities_spec.rb.fix b/old_spcs/requests/v3/abalities_spec.rb.fix similarity index 100% rename from spec/requests/v3/abalities_spec.rb.fix rename to old_spcs/requests/v3/abalities_spec.rb.fix diff --git a/old_spcs/requests/v3/flat_pages_spec.rb b/old_spcs/requests/v3/flat_pages_spec.rb new file mode 100644 index 00000000..a7e7b2cf --- /dev/null +++ b/old_spcs/requests/v3/flat_pages_spec.rb @@ -0,0 +1,145 @@ +# frozen_string_literal: true + +require 'rails_helper' +require 'ecds_rails_auth_engine' + +RSpec.describe 'V3::FlatPages', type: :request do + let!(:theme) { create(:theme) } + let!(:tours) { create_list(:tour_with_flat_pages, 1, theme: theme) } + let!(:flat_page) { tours.first.flat_pages.first } + let(:tour_id) { tours.first.id } + + context 'create tour with flat pages' do + before { + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + user.tours = [] + signed_cookie(user) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: user.id).token + get "/#{Apartment::Tenant.current}/flat-pages" + } + + it 'associates flat_page with tour' do + expect(response).to have_http_status(200) + expect(json.size).to eq(FlatPage.count) + end + end + + # context 'flat page included in tours payload' do + # before { + # tour = create(:tour_with_flat_pages, theme: theme) + # tour.published = true + # tour.save + # get "/#{Apartment::Tenant.current}/tours?slug=#{tour.slug}" + # } + + # it 'creates tour with existing flat page' do + # expect(included.select { |d| d['type'] == 'tour_flat_pages' }.length).to eq(3) + # end + # end + + context 'get specific flat page by id' do + before { + Tour.first.update(published: true) + Tour.first.flat_pages << FlatPage.first + get "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}" + } + + it 'returns requested flat page' do + expect(response).to have_http_status(200) + end + end + + # valid payload + let(:valid_attributes) do + factory_to_json_api(FactoryBot.build(:flat_page)) + end + + describe 'POST /tenant/flat-pages' do + + context 'create page not authenticated' do + before { post "/#{Apartment::Tenant.current}/flat-pages", params: valid_attributes } + it 'returns status code 401' do + expect(response).to have_http_status(401) + end + end + + context 'when created by tour set admin' do + before { + User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + post "/#{Apartment::Tenant.current}/flat-pages", params: valid_attributes + } + it 'creates a tour' do + expect(response).to have_http_status(201) + end + end + + context 'create without valid params' do + before { + User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + post "/#{Apartment::Tenant.current}/flat-pages", params: { foo: 'bar' } + } + it 'returns unprocessable entity' do + expect(response).to have_http_status(422) + end + end + end + + describe 'PUT /tenant/flat-pages/' do + context 'update page not authenticated' do + before { put "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes } + it 'returns status code 401' do + expect(response).to have_http_status(401) + end + end + + context 'when updated by tour set admin' do + before { + User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + put "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes + } + it 'creates a tour' do + expect(response).to have_http_status(200) + end + end + + context 'update without valid params' do + before { + User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + invalid_attributes = { data: { type: 'flat_pages', attributes: { title: nil } } } + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + put "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.last.id}", params: invalid_attributes + } + it 'returns unprocessable entity' do + expect(response).to have_http_status(422) + end + end + + end + + describe 'DELETE /tenant/flat-pages/' do + context 'delete page not authenticated' do + before { + delete "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes + } + it 'returns status code 401' do + expect(response).to have_http_status(401) + end + end + + context 'when deletes by tour set admin' do + before { + User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + delete "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes + } + it 'creates a tour' do + expect(response).to have_http_status(204) + end + end + end +end diff --git a/old_spcs/requests/v3/map_overlays_request_spec.rb b/old_spcs/requests/v3/map_overlays_request_spec.rb new file mode 100644 index 00000000..b183bcd8 --- /dev/null +++ b/old_spcs/requests/v3/map_overlays_request_spec.rb @@ -0,0 +1,5 @@ +require 'rails_helper' + +RSpec.describe "V3::MapOverlays", type: :request do + +end diff --git a/spec/requests/v3/media_spec.rb b/old_spcs/requests/v3/media_spec.rb.fix similarity index 99% rename from spec/requests/v3/media_spec.rb rename to old_spcs/requests/v3/media_spec.rb.fix index c6fc771b..b8922e4c 100644 --- a/spec/requests/v3/media_spec.rb +++ b/old_spcs/requests/v3/media_spec.rb.fix @@ -193,7 +193,7 @@ context 'update with valid data' do before { - valid_attributes[:data][:attributes]['original_image'] = Rack::Test::UploadedFile.new(Rails.root.join('spec/factories/test.jpg'), 'image/jpg') + valid_attributes[:data][:attributes]['original_image'] = Rack::Test::UploadedFile.new(Rails.root.join('spec/factories/images/test.jpg'), 'image/jpg') valid_attributes[:data][:attributes]['id'] = Medium.first.id put "/#{Apartment::Tenant.current}/media/#{Medium.first.id}", params: valid_attributes, headers: { Authorization: "Bearer #{User.last.login.oauth2_token}" } } diff --git a/spec/requests/v3/modes_spec.rb b/old_spcs/requests/v3/modes_spec.rb similarity index 100% rename from spec/requests/v3/modes_spec.rb rename to old_spcs/requests/v3/modes_spec.rb diff --git a/spec/requests/v3/stop_media_spec.rb b/old_spcs/requests/v3/stop_media_spec.rb.fix similarity index 97% rename from spec/requests/v3/stop_media_spec.rb rename to old_spcs/requests/v3/stop_media_spec.rb.fix index 327bf3ae..12cc6903 100644 --- a/spec/requests/v3/stop_media_spec.rb +++ b/old_spcs/requests/v3/stop_media_spec.rb.fix @@ -6,10 +6,7 @@ let!(:stop) { create_list(:stop_with_media, 1) } let!(:medium) { create(:medium) } let!(:new_medium) { create(:medium) } - let!(:user) { create(:user) } - let!(:login) { create(:login, user: user) } - let(:headers) { { Authorization: "Bearer #{login.oauth2_token}" } } - + let!(:user) { User.first } describe 'GET /stop-media' do context 'gets all stop media' do diff --git a/spec/requests/v3/stops_spec.rb b/old_spcs/requests/v3/stops_spec.rb similarity index 60% rename from spec/requests/v3/stops_spec.rb rename to old_spcs/requests/v3/stops_spec.rb index dfd46919..339f6723 100644 --- a/spec/requests/v3/stops_spec.rb +++ b/old_spcs/requests/v3/stops_spec.rb @@ -5,14 +5,22 @@ RSpec.describe 'V3::Stops API' do # Initialize the test data - let!(:login) { User.find_by(super: true).login } + let!(:user) { User.find_by(super: true) } # Test suite for GET /stops describe 'GET /stops' do context 'when stops exist' do - - before { get "/#{Apartment::Tenant.current}/stops" } - + + before { + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + user.tours = [] + signed_cookie(user) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: user.id).token + get "/#{Apartment::Tenant.current}/stops" + } + it 'returns status code 200' do expect(response).to have_http_status(200) end @@ -22,19 +30,21 @@ end end - context 'returns stops not associated with given tour' do - before { - get "/#{Apartment::Tenant.current}/stops?tour_id=#{Tour.last.id}" - } + # context 'returns stops not associated with given tour' do + # before { + # get "/#{Apartment::Tenant.current}/stops?tour_id=#{Tour.last.id}" + # } - it 'returns stops not part of given tour' do - expect(json.size).to eq(Stop.not_in_tour(Tour.last.id).count) - end - end + # it 'returns stops not part of given tour' do + # expect(json.size).to eq(Stop.not_in_tour(Tour.last.id).count) + # end + # end context 'get stop by slug and tour' do before { - get "/#{Apartment::Tenant.current}/tour-stops?slug=#{Stop.first.slug}&tour=#{Stop.first.tours.first.id}" + Tour.first.update(published: true) + Tour.first.stops << Stop.first + get "/#{Apartment::Tenant.current}/tour-stops?slug=#{Stop.first.slug}&tour=#{Tour.first.id}" } it 'returns stop in tour with slug' do @@ -46,36 +56,42 @@ # Test suite for GET /stops/:id describe 'GET /stops/:id' do - before { get "/#{Apartment::Tenant.current}/stops/#{Stop.first.id}" } + before { + Tour.first.update(published: true) + Tour.first.stops << Stop.first + get "/#{Apartment::Tenant.current}/stops/#{Stop.first.id}" + } context 'when tour stop exists' do it 'returns status code 200' do expect(response).to have_http_status(200) end - it 'returns the stop' do - expect(json['id']).to eq(Stop.first.id.to_s) - end + # For now, access to stop is through /tour-stop?slug=XX&tour=Y + # it 'returns the stop' do + # expect(json['id']).to eq(Stop.first.id.to_s) + # end - it 'has a meta_description based on description truncated and sanitized' do - expect(attributes['meta_description']).not_to include('

') - end + # it 'has a meta_description based on description truncated and sanitized' do + # expect(attributes['meta_description']).not_to include('

') + # end end - context 'when tour stop does not exist' do + context 'when stop does not exist' do before { get "/#{Apartment::Tenant.current}/stops/0" } it 'returns status code 404' do - expect(response).to have_http_status(404) + expect(response).to have_http_status(200) end - it 'returns a not found message' do - expect(response.body).to match(/Couldn't find Stop/) + it 'returns dummy stop' do + expect(json[:id]).to match('0') + expect(attributes[:title]).to be_nil end end end - describe 'GET /:tenant/stops?slug=:stop_slug' do + describe 'GET /:tenant/tour-stops?slug=:stop_slug' do let!(:stop) { Stop.second } let!(:tour) { stop.tours.first } let!(:original_slug) { stop.slug } @@ -84,8 +100,8 @@ context 'get stop after title change' do before { - stop.title = new_title - stop.save + tour.update(published: true) + stop.update(title: new_title) } before { get "/#{Apartment::Tenant.current}/tour-stops?slug=#{new_title.parameterize}&tour=#{tour.id}" } @@ -114,8 +130,11 @@ end context 'when request attributes are valid' do - before { User.first.update_attribute(:super, true) } - before { post "/#{Apartment::Tenant.current}/stops", params: valid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } + before { + User.first.update_attribute(:super, true) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + post "/#{Apartment::Tenant.current}/stops", params: valid_attributes + } it 'returns status code 201' do expect(response).to have_http_status(201) @@ -141,7 +160,10 @@ factory_to_json_api(FactoryBot.build(:stop, title: '3 Stacks')) end - before { put "/#{Apartment::Tenant.current}/stops/#{Stop.second.id}", params: valid_attributes, headers: { Authorization: "Bearer #{login.oauth2_token}" } } + before { + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: user.id).token + put "/#{Apartment::Tenant.current}/stops/#{Stop.second.id}", params: valid_attributes + } context 'when stop exists' do it 'returns status code 204' do @@ -155,21 +177,27 @@ end context 'when the stop does not exist' do - before { put "/#{Apartment::Tenant.current}/stops/0", params: valid_attributes, headers: { Authorization: "Bearer #{login.oauth2_token}" } } + before { + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: user.id).token + put "/#{Apartment::Tenant.current}/stops/0", params: valid_attributes + } it 'returns status code 404' do expect(response).to have_http_status(404) end it 'returns a not found message' do - expect(response.body).to match(/Couldn't find Stop/) + expect(response.body).to match(/Record not found/) end end end # Test suite for DELETE /stops/:id describe 'DELETE /stops/:id' do - before { delete "/#{Apartment::Tenant.current}/stops/#{Stop.last.id}", headers: { Authorization: "Bearer #{login.oauth2_token}" } } + before { + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: user.id).token + delete "/#{Apartment::Tenant.current}/stops/#{Stop.last.id}" + } it 'returns status code 204' do expect(response).to have_http_status(204) diff --git a/spec/requests/v3/themes_spec.rb b/old_spcs/requests/v3/themes_spec.rb similarity index 100% rename from spec/requests/v3/themes_spec.rb rename to old_spcs/requests/v3/themes_spec.rb diff --git a/spec/requests/v3/tour_set_users_spec.rb b/old_spcs/requests/v3/tour_set_users_spec.rb similarity index 100% rename from spec/requests/v3/tour_set_users_spec.rb rename to old_spcs/requests/v3/tour_set_users_spec.rb diff --git a/spec/requests/v3/tour_sets_spec.rb.fix b/old_spcs/requests/v3/tour_sets_spec.rb.fix similarity index 100% rename from spec/requests/v3/tour_sets_spec.rb.fix rename to old_spcs/requests/v3/tour_sets_spec.rb.fix diff --git a/spec/requests/v3/tour_stops_spec.rb b/old_spcs/requests/v3/tour_stops_spec.rb similarity index 57% rename from spec/requests/v3/tour_stops_spec.rb rename to old_spcs/requests/v3/tour_stops_spec.rb index f1900815..01d39771 100644 --- a/spec/requests/v3/tour_stops_spec.rb +++ b/old_spcs/requests/v3/tour_stops_spec.rb @@ -8,9 +8,11 @@ # Test suite for GET /stops describe 'GET /tour-stops' do - before { Apartment::Tenant.switch! TourSet.second.subdir } - # before { Tour.first.stops << Stop.last(5) } - before { get "/#{Apartment::Tenant.current}/tour-stops" } + before { + Apartment::Tenant.switch! TourSet.second.subdir + Tour.all.each { |tour| tour.update(published: true) } + get "/#{Apartment::Tenant.current}/tour-stops" + } context 'when stops exist' do it 'returns status code 200' do @@ -34,13 +36,16 @@ end describe 'GET /tour-stops?slug=slug&tour=X' do - before { Apartment::Tenant.switch! TourSet.second.subdir } - before { get "/#{Apartment::Tenant.current}/tour-stops?slug=#{Tour.first.stops.first.stop_slugs.first.slug}&tour=#{Tour.first.id}" } - - context 'get tour stop by slug and tour'do - it 'responds with the tour stop' do - expect(json['id'].to_i).to eq(Tour.first.stops.first.id) - end + before { + Apartment::Tenant.switch! TourSet.second.subdir + Tour.first.update(published: true) + get "/#{Apartment::Tenant.current}/tour-stops?slug=#{Tour.first.stops.first.stop_slugs.first.slug}&tour=#{Tour.first.id}" + } + + context 'get tour stop by slug and tour' do + it 'responds with the tour stop' do + expect(json['id'].to_i).to eq(Tour.first.stops.first.id) + end end end @@ -52,15 +57,17 @@ let!(:tour2) { Tour.last } let!(:stop2) { tour2.stops.last } let!(:new_title) { "#{Faker::Movies::Lebowski.character}" } - - before{ + + before { tour1.stops = [Stop.create(title: new_title)] + tour1.update(published: true) tour1.save tour2.stops = [Stop.create(title: new_title)] + tour2.update(published: true) tour2.save } - - + + context 'get stop with duplicate title/slug in correct tour' do before { get "/#{Apartment::Tenant.current}/tour-stops?slug=#{new_title.parameterize}&tour=#{tour1.id}" } it 'is true' do @@ -69,7 +76,6 @@ end end - context 'get stop with duplicate title/slug in correct tour' do before { get "/#{Apartment::Tenant.current}/tour-stops?slug=#{new_title.parameterize}&tour=#{tour2.id}" } it 'is true' do @@ -77,28 +83,5 @@ expect(json['relationships']['tour']['data']['id'].to_i).to eq(tour2.id) end end - - end - - describe 'DELETE /tour-stops' do - before { User.last.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) } - before { @stop = Stop.last } - before { @stop_count = Stop.count } - before { delete "/#{Apartment::Tenant.current}/tour-stops/#{TourStop.find_by(stop: @stop).id}", headers: { Authorization: "Bearer #{User.last.login.oauth2_token}" } } - - context 'when a tour-stop is deleted and the stop no longers belogs to a tour, the stop is deleted' do - it 'deletes the associated stop' do - expect(Stop.count).to eq @stop_count - 1 - end - end - - before { Tour.all.each { |t| t.stops << Stop.first } } - before { delete "/#{Apartment::Tenant.current}/tour-stops/#{TourStop.find_by(stop: Stop.first).id}", headers: { Authorization: "Bearer #{User.last.login.oauth2_token}" } } - - context 'when a tour-stop is deleted, the stop is not deleted if it belongs to other tours.' do - it 'deletes tour-stop but not the stop' do - expect(Stop.first.title).to eq(Stop.first.title) - end - end end end diff --git a/spec/requests/v3/tours_spec.rb b/old_spcs/requests/v3/tours_spec.rb similarity index 76% rename from spec/requests/v3/tours_spec.rb rename to old_spcs/requests/v3/tours_spec.rb index 7778c6b6..ac13555d 100644 --- a/spec/requests/v3/tours_spec.rb +++ b/old_spcs/requests/v3/tours_spec.rb @@ -108,7 +108,7 @@ } it 'returns nothing' do - expect(response).to have_http_status(404) + expect(response).to have_http_status(200) end end end @@ -122,9 +122,12 @@ before { Apartment::Tenant.switch! TourSet.find(TourSet.pluck(:id).sample).subdir } context 'when the post is valid and authenticated as non-tour set admin' do - before { User.last.update_attribute(:super, false) } - before { User.last.update_attribute(:tour_sets, []) } - before { post "/#{Apartment::Tenant.current}/tours", params: valid_attributes, headers: { Authorization: "Bearer #{User.last.login.oauth2_token}" } } + before { + User.last.update_attribute(:super, false) + User.last.update_attribute(:tour_sets, []) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.last.id).token + post "/#{Apartment::Tenant.current}/tours", params: valid_attributes + } it 'returns status code 401' do expect(response).to have_http_status(401) @@ -132,8 +135,11 @@ end context 'when created by tour set admin' do - before { User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) } - before { post "/#{Apartment::Tenant.current}/tours", params: valid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } + before { + User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + post "/#{Apartment::Tenant.current}/tours", params: valid_attributes + } it 'creates a tour' do expect(attributes['title']).to eq('Learn Elm') end @@ -150,7 +156,8 @@ before { # Tour.create!(published: true) User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) - post "/#{Apartment::Tenant.current}/tours", params: invalid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + post "/#{Apartment::Tenant.current}/tours", params: invalid_attributes } it 'returns status code 201' do @@ -170,7 +177,14 @@ end context 'when the record exists' do - before { put "/#{Apartment::Tenant.current}/tours/#{Tour.last.id}", params: valid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } + before { + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: user.id).token + put "/#{Apartment::Tenant.current}/tours/#{Tour.last.id}", params: valid_attributes + } it 'updates the record' do expect(json).not_to be_empty @@ -185,9 +199,12 @@ # Test suite for DELETE /atlanta/tours/:id describe 'DELETE /atlanta/tours/:id' do - before { Tour.create! } - before { User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) } - before { delete "/#{Apartment::Tenant.current}/tours/#{Tour.last.id}", headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } + before { + Tour.create! + User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.first.id).token + delete "/#{Apartment::Tenant.current}/tours/#{Tour.last.id}" + } it 'returns status code 204' do expect(response).to have_http_status(204) @@ -197,9 +214,9 @@ describe 'Get //tours authenticated' do context 'tour set adim gets all the tours for that set' do before { - user = User.last - user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) - get "/#{Apartment::Tenant.current}/tours", headers: { Authorization: "Bearer #{user.login.oauth2_token}" } + User.last.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + cookies['auth'] = EcdsRailsAuthEngine::Login.find_by(user_id: User.last.id).token + get "/#{Apartment::Tenant.current}/tours" } it 'returns all the tours in the set' do @@ -210,16 +227,19 @@ context 'get tours as tour author' do before { user = User.first - user.super = false - user.tour_sets = [] - user.tours = [] + login = EcdsRailsAuthEngine::Login.find_by(user_id: user.id) + user.update(super: false, tour_sets: [], tours: []) + # user.super = false + # user.tour_sets = [] + # user.tours = [] user.save user.tours << Tour.first Tour.all.each do |t| t.published = false t.save end - get "/#{Apartment::Tenant.current}/tours", headers: { Authorization: "Bearer #{user.login.oauth2_token}" } + cookies['auth'] = login.token + get "/#{Apartment::Tenant.current}/tours" } it 'only returns tours user can edit' do diff --git a/spec/requests/v3/users_spec.fix b/old_spcs/requests/v3/users_spec.fix similarity index 100% rename from spec/requests/v3/users_spec.fix rename to old_spcs/requests/v3/users_spec.fix diff --git a/public/soundcloud.jpg b/public/soundcloud.jpg new file mode 100644 index 00000000..a18731df Binary files /dev/null and b/public/soundcloud.jpg differ diff --git a/spec/controllers/v1/tours_controller_spec.rb b/spec/controllers/v1/tours_controller_spec.rb deleted file mode 100644 index 721b5dda..00000000 --- a/spec/controllers/v1/tours_controller_spec.rb +++ /dev/null @@ -1,7 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -RSpec.describe V1::ToursController, type: :controller do - -end diff --git a/spec/controllers/v3/authenticated_controller_spec.rb b/spec/controllers/v3/authenticated_controller_spec.rb deleted file mode 100644 index f2935b2e..00000000 --- a/spec/controllers/v3/authenticated_controller_spec.rb +++ /dev/null @@ -1,7 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -RSpec.describe V3::AuthenticatedController, type: :controller do - -end diff --git a/spec/controllers/v3/flat_pages_controller_spec.rb b/spec/controllers/v3/flat_pages_controller_spec.rb index ec7f46b2..d425c993 100644 --- a/spec/controllers/v3/flat_pages_controller_spec.rb +++ b/spec/controllers/v3/flat_pages_controller_spec.rb @@ -1,129 +1,360 @@ require 'rails_helper' -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - RSpec.describe V3::FlatPagesController, type: :controller do + describe 'GET #index' do + it 'returns a 200 response with flat_pages connected to published tours' do + create_list(:tour_with_flat_pages, 5, theme: create(:theme), mode: create(:mode)) + Tour.first.update(published: true) if Tour.published.empty? + Tour.last.update(published: false) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(Tour.count).to be > Tour.published.count + json.each do |flat_page| + expect(FlatPage.find(flat_page[:id]).tours.any? { |tour| tour.published }) + end + expect(json.count).to be < FlatPage.count + end - # This should return the minimal set of attributes required to create a valid - # V3::FlatPage. As you add validations to V3::FlatPage, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip('Add a hash of attributes valid for your model') - } - - let(:invalid_attributes) { - skip('Add a hash of attributes invalid for your model') - } + it 'returns a 200 response with no flat_pages when request is authenticated by person with no access' do + create_list(:tour_with_flat_pages, 5, theme: create(:theme), mode: create(:mode)) + Tour.first.update(published: true) if Tour.published.empty? + Tour.last.update(published: false) if Tour.published.count == Tour.count + user = create(:user) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(Tour.count).to be > Tour.published.count + json.each do |flat_page| + expect(FlatPage.find(flat_page[:id]).tours.any? { |tour| tour.published }) + end + expect(json.count).to be == FlatPage.all.reject {|fp| !fp.published}.count + end - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # V3::FlatPagesController. Be sure to keep this updated too. - let(:valid_session) { {} } + it 'returns a 200 response with flat_pages when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour_with_flat_pages, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: tour.tenant } + expect(FlatPage.count).to be > 1 + expect(response.status).to eq(200) + expect(json.count).to eq(FlatPage.count) + end - describe 'GET #index' do - it 'returns a success response' do - flat_page = FlatPage.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + tour = create(:tour_with_flat_pages) + tour.update(published: false) + create_list(:flat_page, 7) + user = create(:user) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(tour.flat_pages.count) + expect(json.count).to be < FlatPage.count end end describe 'GET #show' do - it 'returns a success response' do - flat_page = FlatPage.create! valid_attributes - get :show, params: { id: flat_page.to_param }, session: valid_session - expect(response).to be_success + it 'returns a 200 response that is empty stop' do + tour = create(:tour) + tour.update(published: false) + create_list(:flat_page, 3) + FlatPage.all.each { |flat_page| tour.flat_pages << flat_page } + # Make sure the flat page is only associated with the newly created tour + tour.flat_pages.last.update(tours: [tour]) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.flat_pages.last.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(tour.flat_pages.last.id.to_s) + expect(attributes[:title]).to be_nil + end + + it 'returns a 200 response and stop when stop is part of published tour' do + tour = create(:tour) + tour.update(published: true) + create_list(:flat_page, 3) + FlatPage.all.each { |flat_page| tour.flat_pages << flat_page } + tour.flat_pages.last.update(tours: [tour]) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.flat_pages.last.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.flat_pages.last.title) + end + + it 'returns a 200 response that is empty stop when request is authenticated by someone w/o permission' do + tour = create(:tour) + tour.update(published: false) + create_list(:flat_page, 3) + FlatPage.all.each { |flat_page| tour.flat_pages << flat_page } + tour.flat_pages.last.update(tours: [tour]) + user = create(:user) + user.update(super: false) + user.tours = [] + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.flat_pages.last.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(tour.flat_pages.last.id.to_s) + expect(attributes[:title]).to be_nil + end + + it 'returns a 200 response that is a stop when request is authenticated by a tour author' do + tour = create(:tour) + tour.update(published: false) + create_list(:flat_page, 3) + FlatPage.all.each { |flat_page| tour.flat_pages << flat_page } + tour.flat_pages.first.update(tours: [tour]) + user = create(:user) + user.update(super: false) + user.tours << tour + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.flat_pages.first.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.flat_pages.first.title) + end + + it 'returns a 200 response that is a stop when request is authenticated by a tenant admin' do + tour = create(:tour) + tour.update(published: false) + create_list(:flat_page, 3) + FlatPage.all.each { |flat_page| tour.flat_pages << flat_page } + tour.flat_pages.first.update(tours: [tour]) + user = create(:user) + user.update(super: false) + user.tours = [] + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.flat_pages.first.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.flat_pages.first.title) + end + + it 'returns a 200 response that is a stop when request is authenticated by a super user' do + tour = create(:tour) + tour.update(published: false) + create_list(:flat_page, 3) + FlatPage.all.each { |flat_page| tour.flat_pages << flat_page } + tour.flat_pages.first.update(tours: [tour]) + user = create(:user) + user.update(super: true) + user.tours = [] + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.flat_pages.first.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.flat_pages.first.title) end end describe 'POST #create' do context 'with valid params' do - it 'creates a new V3::FlatPage' do - expect { - post :create, params: { v3_flat_page: valid_attributes }, session: valid_session - }.to change(FlatPage, :count).by(1) + it 'return 401 when unauthenciated' do + post :create, params: { data: { type: 'flat_pages', attributes: { title: 'Burrito FlatPage' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + + it 'return 401 when authenciated but not an admin for current tenant' do + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + post :create, params: { data: { type: 'flat_pages', attributes: { title: 'Burrito FlatPage' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) end - it 'renders a JSON response with the new v3_flat_page' do + it 'return 201 when authenciated but an admin for current tenant' do + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + original_flat_page_count = FlatPage.count + post :create, params: { data: { type: 'flat_pages', attributes: { title: 'Burrito FlatPage' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(attributes[:title]).to eq('Burrito FlatPage') + expect(FlatPage.count).to eq(original_flat_page_count + 1) + end - post :create, params: { v3_flat_page: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(v3_flat_page_url(FlatPage.last)) + it 'return 201 when authenciated by super' do + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + original_flat_page_count = FlatPage.count + post :create, params: { data: { type: 'flat_pages', attributes: { title: 'Taco FlatPage' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(attributes[:title]).to eq('Taco FlatPage') + expect(FlatPage.count).to eq(original_flat_page_count + 1) end - end - context 'with invalid params' do - it 'renders a JSON response with errors for the new v3_flat_page' do + it 'return 201 when authenciated by a tour author' do + user = create(:user) + user.tour_sets = [] + user.tours << Tour.last + user.update(super: false) + signed_cookie(user) + original_flat_page_count = FlatPage.count + post :create, params: { data: { type: 'flat_pages', attributes: { title: 'Elmyr' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(attributes[:title]).to eq('Elmyr') + expect(FlatPage.count).to eq(original_flat_page_count + 1) + end - post :create, params: { v3_flat_page: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'return 422 when missing title' do + user = create(:user, super: true) + signed_cookie(user) + original_flat_page_count = FlatPage.count + post :create, params: { data: { type: 'flat_pages', attributes: {} }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(422) + expect(errors).to include('Title can\'t be blank') + expect(FlatPage.count).to eq(original_flat_page_count) end end end describe 'PUT #update' do context 'with valid params' do - let(:new_attributes) { - skip('Add a hash of attributes valid for your model') - } - - it 'updates the requested v3_flat_page' do - flat_page = FlatPage.create! valid_attributes - put :update, params: { id: flat_page.to_param, v3_flat_page: new_attributes }, session: valid_session - flat_page.reload - skip('Add assertions for updated state') + it 'return 401 when unauthenciated' do + tour = create(:tour_with_flat_pages) + post :update, params: { id: tour.flat_pages.last.id, data: { type: 'flat_pages', attributes: { title: 'Burrito FlatPage' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) end - it 'renders a JSON response with the v3_flat_page' do - flat_page = FlatPage.create! valid_attributes + it 'return 401 when authenciated but not an admin for current tenant' do + tour = create(:tour_with_flat_pages, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + post :update, params: { id: tour.flat_pages.first.id, data: { type: 'flat_pages', attributes: { title: 'Burrito FlatPage' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end - put :update, params: { id: flat_page.to_param, v3_flat_page: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') + it 'return 200 and updated tour when authenciated but an admin for current tenant' do + tour = create(:tour_with_flat_pages, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + original_flat_page_title = FlatPage.find(tour.flat_pages.last.id).title + new_title = Faker::Name.unique.name + post :update, params: { id: tour.flat_pages.first.id, data: { type: 'flat_pages', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(original_flat_page_title) + expect(attributes[:title]).to eq(new_title) + expect(FlatPage.find(tour.flat_pages.first.id).title).to eq(new_title) end - end - context 'with invalid params' do - it 'renders a JSON response with errors for the v3_flat_page' do - flat_page = FlatPage.create! valid_attributes + it 'return 200 and updated tour when authenciated by super' do + tour = create(:tour_with_flat_pages) + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + original_flat_page_title = FlatPage.find(tour.flat_pages.last.id).title + new_title = Faker::Name.unique.name + post :update, params: { id: tour.flat_pages.last.id, data: { type: 'flat_pages', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(original_flat_page_title) + expect(attributes[:title]).to eq(new_title) + expect(FlatPage.find(tour.flat_pages.last.id).title).to eq(new_title) + end + + it 'return 200 and updated tour when authenciated by tour author' do + tour = create(:tour_with_flat_pages) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + original_flat_page_title = FlatPage.find(tour.flat_pages.last.id).title + new_title = Faker::Name.unique.name + post :update, params: { id: tour.flat_pages.first.id, data: { type: 'flat_pages', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(original_flat_page_title) + expect(attributes[:title]).to eq(new_title) + expect(FlatPage.find(tour.flat_pages.first.id).title).to eq(new_title) + end - put :update, params: { id: flat_page.to_param, v3_flat_page: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'returns 422 when title in nil' do + flat_page = create(:flat_page) + user = create(:user, super: true) + signed_cookie(user) + post :update, params: { id: flat_page.id, data: { type: 'flat_pages', attributes: { title: nil } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(422) + expect(errors).to include('Title can\'t be blank') end end + + # context 'with invalid params' do + # it 'renders a JSON response with errors for the tour' do + # tour = FlatPage.create! valid_attributes + + # put :update, params: { id: tour.to_param, tour: invalid_attributes } + # expect(response).to have_http_status(:unprocessable_entity) + # expect(response.content_type).to eq('application/json') + # end + # end end describe 'DELETE #destroy' do - it 'destroys the requested v3_flat_page' do - flat_page = FlatPage.create! valid_attributes - expect { - delete :destroy, params: { id: flat_page.to_param }, session: valid_session - }.to change(FlatPage, :count).by(-1) + it 'return 401 when unauthenciated' do + tour = create(:tour_with_flat_pages) + post :destroy, params: { id: Tour.find(tour.id).flat_pages.first.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + + it 'return 401 when authenciated but not an admin for current tenant' do + tour = create(:tour_with_flat_pages) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + signed_cookie(user) + post :destroy, params: { id: tour.flat_pages.first.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + + it 'return 204 and one less tour when authenciated but an admin for current tenant' do + tour = create(:tour_with_flat_pages) + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + flat_page_count = FlatPage.count + post :destroy, params: { id: tour.flat_pages.last.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(FlatPage.count).to eq(flat_page_count - 1) end - end + it 'return 204 and one less tour when authenciated by super' do + tour = create(:tour_with_flat_pages) + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + flat_page_count = FlatPage.count + post :destroy, params: { id: tour.flat_pages.first.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(FlatPage.count).to eq(flat_page_count - 1) + end + + it 'return 204 and one less tour when authenciated by tour author' do + tour = create(:tour_with_flat_pages) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + new_title = Faker::Name.unique.name + flat_page_count = FlatPage.count + post :destroy, params: { id: tour.flat_pages.last.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(FlatPage.count).to eq(flat_page_count - 1) + end + end end diff --git a/spec/controllers/v3/map_icons_controller_spec.rb b/spec/controllers/v3/map_icons_controller_spec.rb new file mode 100644 index 00000000..35d607a5 --- /dev/null +++ b/spec/controllers/v3/map_icons_controller_spec.rb @@ -0,0 +1,185 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe V3::MapIconsController, type: :controller do + describe 'GET #index' do + it 'returns all map icons' do + create_list(:map_icon, rand(2..6)) + get :index, params: { tenant: Apartment::Tenant.current } + expect(json.count).to eq(MapIcon.count) + end + end + + describe 'GET #show' do + context 'unauthenticated and unauthorized' do + it 'returns empty MapIcon when not unauthenticated' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_icon = create(:map_icon, stop: tour.stops.last) + get :show, params: { tenant: TourSet.first.subdir, id: map_icon.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_icon.id.to_s) + end + + it 'returns empty MapIcon when not unauthenticated but unauthorized' do + initial_tenant = Apartment::Tenant.current + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_icon = create(:map_icon, stop: tour.stops.last) + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + expect(user.tour_sets).not_to include TourSet.find_by(subdir: initial_tenant) + signed_cookie(user) + Apartment::Tenant.switch! initial_tenant + get :show, params: { tenant: initial_tenant, id: map_icon } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_icon.id.to_s) + end + end + + context 'authorized' do + it 'responds with 200 and a MapIcon when tour is published' do + tour = create(:tour, published: true, stops: create_list(:stop, 3)) + map_icon = create(:map_icon, stop: tour.stops.last) + get :show, params: { tenant: TourSet.first.subdir, id: map_icon.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_icon.id.to_s) + end + + it 'responds with 200 and a MapIcon when requested by a tour author' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_icon = create(:map_icon, stop: tour.stops.last) + user = create(:user, super: false) + user.tours << tour + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: map_icon } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_icon.id.to_s) + end + + it 'responds with 200 and a list of MapIcon when requested by tenant admin' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_icon = create(:map_icon, stop: tour.stops.last) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: map_icon } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_icon.id.to_s) + end + + it 'responds with 200 and a MapIcon when requested by super' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_icon = create(:map_icon, stop: tour.stops.last) + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: map_icon } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_icon.id.to_s) + end + end + end + + describe 'POST #create' do + let(:tour) { create(:tour, stops: create_list(:stop, 2)) } + let(:valid_params) { + { + data: { + type: 'map_icon', + attributes: { + filename: Faker::File.file_name(dir: '', ext: 'png', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/images/icon_base64.txt')), + tour_id: tour.id + } + }, + tenant: Apartment::Tenant.current + } + } + + context 'unauthorized' do + it 'returns 401 when unauthenticated' do + initial_map_icon_count = MapIcon.count + post :create, params: valid_params + expect(response.status).to eq(401) + expect(MapIcon.count).to eq(initial_map_icon_count) + end + + it 'returns 401 when authenticated but unauthorized tenant admin' do + initial_map_icon_count = MapIcon.count + initial_tenant = Apartment::Tenant.current + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + expect(user.tour_sets).not_to include TourSet.find_by(subdir: initial_tenant) + signed_cookie(user) + Apartment::Tenant.switch! initial_tenant + valid_params[:tenant] = initial_tenant + post :create, params: valid_params + expect(response.status).to eq(401) + expect(MapIcon.count).to eq(initial_map_icon_count) + end + end + + context 'authorized' do + it 'creates when request by super' do + user = create(:user, super: true) + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(MapIcon, :count).by(1) + end + + it 'creates when request by current tenant admin' do + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(MapIcon, :count).by(1) + end + + it 'creates when request by tour author' do + user = create(:user, super: false) + user.tours << tour + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(MapIcon, :count).by(1) + end + end + + context 'invalid params' do + let(:tour) { create(:tour, stops: create_list(:stop, 2)) } + let(:invalid_params) { + { + data: { + type: 'map_icon', + attributes: { + filename: Faker::File.file_name(dir: '', ext: 'png', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/images/atl_base64.txt')), + tour_id: tour.id + } + }, + tenant: Apartment::Tenant.current + } + } + + it 'returns 422 with invalid params' do + initial_map_icon_count = MapIcon.count + user = create(:user, super: true) + signed_cookie(user) + post :create, params: { tenant: Apartment::Tenant.current, data: {} } + expect(response).to have_http_status(:unprocessable_entity) + expect(MapIcon.count).to eq(initial_map_icon_count) + end + + it 'returns 422 and size error message' do + initial_map_icon_count = MapIcon.count + user = create(:user, super: true) + signed_cookie(user) + post :create, params: invalid_params + expect(response).to have_http_status(:unprocessable_entity) + expect(MapIcon.count).to eq(initial_map_icon_count) + expect(errors).to include('Icons should be no bigger that 80 by 80 pixels') + end + end + end +end diff --git a/spec/controllers/v3/map_overlays_controller_spec.rb b/spec/controllers/v3/map_overlays_controller_spec.rb new file mode 100755 index 00000000..e8175c3c --- /dev/null +++ b/spec/controllers/v3/map_overlays_controller_spec.rb @@ -0,0 +1,160 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe V3::MapOverlaysController, type: :controller do + describe 'GET #index' do + context 'unauthenticated and unauthorized' do + it 'returns empty MapOverlay when not unauthenticated' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_overlay = create(:map_overlay, tour: tour) + get :show, params: { tenant: TourSet.first.subdir, id: map_overlay.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_overlay.id.to_s) + expect(attributes[:south]).not_to eq(map_overlay.south.to_f.to_s) + expect(attributes[:east]).to be nil + end + + it 'returns empty MapOverlay when not unauthenticated but unauthorized' do + initial_tenant = Apartment::Tenant.current + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_overlay = create(:map_overlay, tour: tour) + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + expect(user.tour_sets).not_to include TourSet.find_by(subdir: initial_tenant) + signed_cookie(user) + Apartment::Tenant.switch! initial_tenant + get :show, params: { tenant: initial_tenant, id: map_overlay } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_overlay.id.to_s) + expect(attributes[:north]).not_to eq(map_overlay.north.to_f.to_s) + expect(attributes[:west]).to be nil + end + end + + context 'authorized' do + it 'responds with 200 and a MapOverlay when tour is published' do + tour = create(:tour, published: true, stops: create_list(:stop, 3)) + map_overlay = create(:map_overlay, tour: tour) + get :show, params: { tenant: TourSet.first.subdir, id: map_overlay.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_overlay.id.to_s) + expect(attributes[:south]).to eq(map_overlay.south.to_f.to_s) + end + + it 'responds with 200 and a MapOverlay when requested by a tour author' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_overlay = create(:map_overlay, tour: tour) + user = create(:user, super: false) + user.tours << tour + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: map_overlay } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_overlay.id.to_s) + expect(attributes[:north]).to eq(map_overlay.north.to_f.to_s) + end + + it 'responds with 200 and a list of MapOverlay when requested by tenant admin' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_overlay = create(:map_overlay, tour: tour) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: map_overlay } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_overlay.id.to_s) + expect(attributes[:north]).to eq(map_overlay.north.to_f.to_s) + end + + it 'responds with 200 and a MapOverlay when requested by super' do + tour = create(:tour, published: false, stops: create_list(:stop, 3)) + map_overlay = create(:map_overlay, tour: tour) + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: map_overlay } + expect(response.status).to eq(200) + expect(json[:id]).to eq(map_overlay.id.to_s) + expect(attributes[:north]).to eq(map_overlay.north.to_f.to_s) + end + end + end + + describe 'POST #create' do + let(:tour) { create(:tour, stops: create_list(:stop, 2)) } + let(:valid_params) { + { + data: { + type: 'map_overlay', + attributes: { + filename: Faker::File.file_name(dir: '', ext: 'png', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/base64_image.txt')), + tour_id: tour.id + } + }, + tenant: Apartment::Tenant.current + } + } + + context 'unauthorized' do + it 'returns 401 when unauthenticated' do + initial_map_overlay_count = MapOverlay.count + post :create, params: valid_params + expect(response.status).to eq(401) + expect(MapOverlay.count).to eq(initial_map_overlay_count) + end + + it 'returns 401 when authenticated but unauthorized tenant admin' do + initial_map_overlay_count = MapOverlay.count + initial_tenant = Apartment::Tenant.current + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + expect(user.tour_sets).not_to include TourSet.find_by(subdir: initial_tenant) + signed_cookie(user) + Apartment::Tenant.switch! initial_tenant + valid_params[:tenant] = initial_tenant + post :create, params: valid_params + expect(response.status).to eq(401) + expect(MapOverlay.count).to eq(initial_map_overlay_count) + end + end + + context 'authorized' do + it 'creates when request by super' do + user = create(:user, super: true) + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(MapOverlay, :count).by(1) + end + + it 'creates when request by current tenant admin' do + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(MapOverlay, :count).by(1) + end + + it 'creates when request by tour author' do + user = create(:user, super: false) + user.tours << tour + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(MapOverlay, :count).by(1) + end + end + + context 'invalid params' do + it 'returns 422 with invalid params' do + initial_map_overlay_count = MapOverlay.count + user = create(:user, super: true) + signed_cookie(user) + post :create, params: { tenant: Apartment::Tenant.current, data: {} } + expect(response).to have_http_status(:unprocessable_entity) + expect(MapOverlay.count).to eq(initial_map_overlay_count) + end + end + end +end diff --git a/spec/controllers/v3/media_controller_spec.rb b/spec/controllers/v3/media_controller_spec.rb index 8c370f78..0e2116af 100644 --- a/spec/controllers/v3/media_controller_spec.rb +++ b/spec/controllers/v3/media_controller_spec.rb @@ -2,130 +2,345 @@ require 'rails_helper' -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - RSpec.describe V3::MediaController, type: :controller do - # This should return the minimal set of attributes required to create a valid - # Medium. As you add validations to Medium, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip('Add a hash of attributes valid for your model') + let(:valid_params) { + { + data: { + type: 'media', + attributes: { + base_sixty_four: File.read(Rails.root.join('spec/factories/base64_image.txt')), + filename: Faker::File.file_name(dir: '', ext: 'png', directory_separator: '') + } + }, + tenant: Apartment::Tenant.current + } } - let(:invalid_attributes) { - skip('Add a hash of attributes invalid for your model') - } + if ENV['DB_ADAPTER'] == 'mysql2' + skip('Fix this spec for MySQL. Something to do with it being transactional') + else + describe 'GET #index' do - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # MediaController. Be sure to keep this updated too. - let(:valid_session) { {} } + it 'returns a success response' do + create_list(:medium, 5) + tour = create(:tour, published: true) + Medium.all.each { |m| tour.media << m } + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(tour.media.count) + end - describe 'GET #index' do - it 'returns a success response' do - medium = Medium.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success - end - end + it 'returns only media associated with a public tour' do + published_tour = create(:tour, published: true) + create_list(:medium, rand(1..8)).each { |m| published_tour.media << m } + unpublished_tour = create(:tour, published: false) + create_list(:medium, rand(1..8)).each { |m| unpublished_tour.media << m } + get :index, params: { tenant: Apartment::Tenant.current } + expect(json.count).to eq(Tour.published.map { |t| t.media.count }.sum) + expect(json.count).to be < Medium.count + end + + it 'returns all media when requested by tenant admin' do + published_tour = create(:tour, published: true) + create_list(:medium, rand(1..8)).each { |m| published_tour.media << m } + unpublished_tour = create(:tour, published: false) + create_list(:medium, rand(1..8)).each { |m| unpublished_tour.media << m } + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(Medium.count) + expect(Medium.count).to be > Tour.published.map { |t| t.media.count }.sum + end - describe 'GET #show' do - it 'returns a success response' do - medium = Medium.create! valid_attributes - get :show, params: { id: medium.to_param }, session: valid_session - expect(response).to be_success + it 'returns a paginated list when page parameter is persent' do + user = create(:user, super: true) + signed_cookie(user) + published_tour = create(:tour, published: true) + create_list(:medium, 35).each { |m| published_tour.media << m } + get :index, params: { tenant: Apartment::Tenant.current, page: '2' } + expect(json.count).to eq(10) + end end - end - describe 'POST #create' do - context 'with valid params' do - it 'creates a new Medium' do - expect { - post :create, params: { medium: valid_attributes }, session: valid_session - }.to change(Medium, :count).by(1) + describe 'GET #show' do + it 'returns 401 when medium is not published by a tour or stop' do + medium = create(:medium) + get :show, params: { tenant: Apartment::Tenant.current, id: medium.id } + expect(response.status).to eq(200) + expect(medium.published).to be false + expect(attributes[:title]).to eq('....') + end + + it 'returns the medium when associated with published stop' do + tour = create(:tour, published: true) + stop = create(:stop) + medium = create(:medium) + tour.stops << stop + stop.media << medium + get :show, params: { tenant: Apartment::Tenant.current, id: medium.id } + expect(medium.published).to be true + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(medium.title) end - it 'renders a JSON response with the new medium' do + it 'returns the medium when unpublished but requested is authorized' do + medium = create(:medium) + medium.save + expect(medium.file.attached?).to be true + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: medium.id } + expect(medium.published).to be false + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(medium.title) + end - post :create, params: { medium: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(medium_url(Medium.last)) + it 'returns the medium that if a gif and requested is authorized' do + medium = create(:medium, base_sixty_four: File.read(Rails.root.join('spec/factories/images/gif_base64.txt')), filename: Faker::File.file_name(dir: '', ext: 'gif', directory_separator: '')) + medium.save + expect(medium.file.attached?).to be true + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: medium.id } + expect(response.status).to eq(200) + expect(attributes[:files][:mobile]).to end_with 'gif' end end - context 'with invalid params' do - it 'renders a JSON response with errors for the new medium' do + describe 'POST #create' do + context 'with valid params and request is authorized' do + it 'creates a new Medium' do + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(Medium, :count).by(1) + end - post :create, params: { medium: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'renders a JSON response with the new medium when super' do + user = create(:user, super: true) + signed_cookie(user) + post :create, params: valid_params + expect(response).to have_http_status(:created) + expect(response.content_type).to eq('application/json; charset=utf-8') + expect(json[:id]).to eq(Medium.last.id.to_s) + end + + it 'renders a JSON response with the new medium when tenant admin' do + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + post :create, params: valid_params + expect(response).to have_http_status(:created) + expect(response.content_type).to eq('application/json; charset=utf-8') + expect(json[:id]).to eq(Medium.last.id.to_s) + end + + it 'renders a JSON response with the new medium when tour author' do + user = create(:user) + tour = create(:tour) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + post :create, params: valid_params + expect(response).to have_http_status(:created) + expect(response.content_type).to eq('application/json; charset=utf-8') + expect(json[:id]).to eq(Medium.last.id.to_s) + end end - end - end - describe 'PUT #update' do - context 'with valid params' do - let(:new_attributes) { - skip('Add a hash of attributes valid for your model') - } + context 'with invalid params' do + it 'renders a JSON response with errors for the new medium' do + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + post :create, params: { medium: 'invalid_params', tenant: Apartment::Tenant.current } + expect(response).to have_http_status(:unprocessable_entity) + expect(response.content_type).to eq('application/json; charset=utf-8') + end - it 'updates the requested medium' do - medium = Medium.create! valid_attributes - put :update, params: { id: medium.to_param, medium: new_attributes }, session: valid_session - medium.reload - skip('Add assertions for updated state') + it 'renders a JSON response with errors for the new medium when file is jp2' do + user = create(:user, super: true) + signed_cookie(user) + jp2_params = valid_params.clone + jp2_params[:data][:attributes] = { + base_sixty_four: File.read(Rails.root.join('spec/factories/images/jp2_base64.txt')), + filename: Faker::File.file_name(dir: '', ext: 'jp2', directory_separator: '') + } + post :create, params: jp2_params + expect(response).to have_http_status(:unprocessable_entity) + expect(errors).to include('JPEG 2000 fils are not supported. Plese convert the image to a reqular JPEG or WebP format.') + end end - it 'renders a JSON response with the medium' do - medium = Medium.create! valid_attributes + context 'with unauthenticated request' do + it 'responds unauthorized when unauthenticated' do + post :create, params: valid_params + expect(response).to have_http_status(:unauthorized) + end - put :update, params: { id: medium.to_param, medium: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') + it 'responds unauthorized when authenticated but not authorized' do + initial_tenant = Apartment::Tenant.current + user = create(:user) + user.tour_sets << create(:tour_set) + signed_cookie(user) + Apartment::Tenant.switch! initial_tenant + post :create, params: valid_params + expect(response).to have_http_status(:unauthorized) + end end end - context 'with invalid params' do - it 'renders a JSON response with errors for the medium' do - medium = Medium.create! valid_attributes + describe 'PUT #update' do + context 'with valid params and request is authorized' do - put :update, params: { id: medium.to_param, medium: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'renders a JSON response with the new medium when super' do + medium = create(:medium) + update_params = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::MediumSerializer.new(medium)).to_json).with_indifferent_access + update_params[:tenant] = Apartment::Tenant.current + update_params[:id] = update_params[:data][:id] + user = create(:user, super: true) + initial_title = update_params[:data][:attributes][:title] + update_params[:data][:attributes][:title] = Faker::Movies::HitchhikersGuideToTheGalaxy.location + signed_cookie(user) + post :update, params: update_params + expect(response).to have_http_status(:ok) + expect(attributes[:title]).not_to eq(initial_title) + end + + it 'renders a JSON response with the new medium when tenant admin' do + medium = create(:medium) + update_params = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::MediumSerializer.new(medium)).to_json).with_indifferent_access + update_params[:tenant] = Apartment::Tenant.current + update_params[:id] = update_params[:data][:id] + user = create(:user, super: false) + user.tours = [] + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + initial_title = update_params[:data][:attributes][:title] + update_params[:data][:attributes][:title] = Faker::Movies::HitchhikersGuideToTheGalaxy.location + signed_cookie(user) + post :update, params: update_params + expect(response).to have_http_status(:ok) + expect(attributes[:title]).not_to eq(initial_title) + end + + it 'renders a JSON response with the new medium when tour author' do + medium = create(:medium) + update_params = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::MediumSerializer.new(medium)).to_json).with_indifferent_access + update_params[:tenant] = Apartment::Tenant.current + update_params[:id] = update_params[:data][:id] + user = create(:user, super: false) + user.tour_sets = [] + user.tours << create(:tour) + initial_title = update_params[:data][:attributes][:title] + update_params[:data][:attributes][:title] = Faker::Movies::HitchhikersGuideToTheGalaxy.location + signed_cookie(user) + post :update, params: update_params + expect(response).to have_http_status(:ok) + expect(attributes[:title]).not_to eq(initial_title) + end + end + + context 'with invalid params' do + it 'renders a JSON response with errors for the new medium' do + user = create(:user, super: true) + signed_cookie(user) + post :update, params: { id: create(:medium).id, data: { type: 'medium', attributes: { filename: nil } }, tenant: Apartment::Tenant.current } + expect(response).to have_http_status(:unprocessable_entity) + expect(response.content_type).to eq('application/json; charset=utf-8') + end + end + + context 'with unauthenticated request' do + it 'responds unauthorized when unauthenticated' do + medium = create(:medium) + update_params = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::MediumSerializer.new(medium)).to_json).with_indifferent_access + update_params[:tenant] = Apartment::Tenant.current + update_params[:id] = update_params[:data][:id] + post :update, params: update_params + expect(response).to have_http_status(:unauthorized) + end + + it 'responds unauthorized when authenticated but not authorized' do + initial_tenant = Apartment::Tenant.current + medium = create(:medium) + update_params = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::MediumSerializer.new(medium)).to_json).with_indifferent_access + update_params[:tenant] = Apartment::Tenant.current + update_params[:id] = update_params[:data][:id] + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + signed_cookie(user) + Apartment::Tenant.switch! initial_tenant + post :update, params: update_params + expect(response).to have_http_status(:unauthorized) + end end end - end - describe 'DELETE #destroy' do - it 'destroys the requested medium' do - medium = Medium.create! valid_attributes - expect { - delete :destroy, params: { id: medium.to_param }, session: valid_session - }.to change(Medium, :count).by(-1) + describe 'DELETE #destroy' do + context 'authenticated and authorized' do + it 'destroys the requested medium when request from super' do + medium = create(:medium) + user = create(:user, super: true) + signed_cookie(user) + expect { + delete :destroy, params: { id: medium.to_param, tenant: Apartment::Tenant.current } + }.to change(Medium, :count).by(-1) + end + + it 'destroys the requested medium when request from tenant admin' do + medium = create(:medium) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + expect { + delete :destroy, params: { id: medium.to_param, tenant: Apartment::Tenant.current } + }.to change(Medium, :count).by(-1) + end + + it 'destroys the requested medium when request from tour author' do + medium = create(:medium) + user = create(:user, super: false) + user.tour_sets = [] + user.tours << create(:tour) + signed_cookie(user) + expect { + delete :destroy, params: { id: medium.to_param, tenant: Apartment::Tenant.current } + }.to change(Medium, :count).by(-1) + end + end + + context 'authenticated but not authorized' do + it 'returns unauthorized' do + initial_tenant = Apartment::Tenant.current + medium = create(:medium) + user = create(:user, super: false) + Apartment::Tenant.reset + tour_set = create(:tour_set) + user.tour_sets << tour_set + signed_cookie(user) + Apartment::Tenant.switch! initial_tenant + initial_media_count = Medium.count + delete :destroy, params: { id: medium.to_param, tenant: Apartment::Tenant.current } + expect(response).to have_http_status(:unauthorized) + expect(Medium.count).to eq(initial_media_count) + end + end + + context 'unauthenticated' do + it 'returns unauthorized' do + medium = create(:medium) + initial_media_count = Medium.count + delete :destroy, params: { id: medium.to_param, tenant: Apartment::Tenant.current } + expect(response).to have_http_status(:unauthorized) + expect(Medium.count).to eq(initial_media_count) + end + end end end - end diff --git a/spec/controllers/v3/modes_controller_spec.rb b/spec/controllers/v3/modes_controller_spec.rb index 633e7aa2..6a103de3 100644 --- a/spec/controllers/v3/modes_controller_spec.rb +++ b/spec/controllers/v3/modes_controller_spec.rb @@ -2,130 +2,25 @@ require 'rails_helper' -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - RSpec.describe V3::ModesController, type: :controller do - # This should return the minimal set of attributes required to create a valid - # Mode. As you add validations to Mode, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip('Add a hash of attributes valid for your model') - } - - let(:invalid_attributes) { - skip('Add a hash of attributes invalid for your model') - } - - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # ModesController. Be sure to keep this updated too. - let(:valid_session) { {} } describe 'GET #index' do it 'returns a success response' do - mode = Mode.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success + tour_set = create(:tour_set) + get :index, params: { tenant: tour_set.subdir } + expect(response.status).to eq(200) + expect(json.count).to eq(4) end end describe 'GET #show' do it 'returns a success response' do - mode = Mode.create! valid_attributes - get :show, params: { id: mode.to_param }, session: valid_session - expect(response).to be_success - end - end - - describe 'POST #create' do - context 'with valid params' do - it 'creates a new Mode' do - expect { - post :create, params: { mode: valid_attributes }, session: valid_session - }.to change(Mode, :count).by(1) - end - - it 'renders a JSON response with the new mode' do - - post :create, params: { mode: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(mode_url(Mode.last)) - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the new mode' do - - post :create, params: { mode: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe 'PUT #update' do - context 'with valid params' do - let(:new_attributes) { - skip('Add a hash of attributes valid for your model') - } - - it 'updates the requested mode' do - mode = Mode.create! valid_attributes - put :update, params: { id: mode.to_param, mode: new_attributes }, session: valid_session - mode.reload - skip('Add assertions for updated state') - end - - it 'renders a JSON response with the mode' do - mode = Mode.create! valid_attributes - - put :update, params: { id: mode.to_param, mode: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the mode' do - mode = Mode.create! valid_attributes - - put :update, params: { id: mode.to_param, mode: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe 'DELETE #destroy' do - it 'destroys the requested mode' do - mode = Mode.create! valid_attributes - expect { - delete :destroy, params: { id: mode.to_param }, session: valid_session - }.to change(Mode, :count).by(-1) + tour_set = create(:tour_set) + Apartment::Tenant.switch! tour_set.subdir + get :show, params: { id: Mode.first.to_param, tenant: tour_set.subdir } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(Mode.first.title) end end - end diff --git a/spec/controllers/v3/stops_controller_spec.rb b/spec/controllers/v3/stops_controller_spec.rb index 155cc5ad..929cd1f9 100644 --- a/spec/controllers/v3/stops_controller_spec.rb +++ b/spec/controllers/v3/stops_controller_spec.rb @@ -2,130 +2,354 @@ require 'rails_helper' -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - RSpec.describe V3::StopsController, type: :controller do + describe 'GET #index' do + it 'returns a 200 response with stops connected to published tours' do + create_list(:tour_with_stops, 5, theme: create(:theme), mode: create(:mode)) + Tour.first.update(published: true) + Tour.last.update(published: false) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(Tour.count).to be > Tour.published.count + json.each do |stop| + expect(Stop.find(stop[:id]).tours.any? { |tour| tour.published }) + end + expect(json.count).to be < Stop.count + end - # This should return the minimal set of attributes required to create a valid - # Stop. As you add validations to Stop, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip('Add a hash of attributes valid for your model') - } - - let(:invalid_attributes) { - skip('Add a hash of attributes invalid for your model') - } + it 'returns a 200 response with no stops when request is authenticated by person with no access' do + create_list(:tour_with_stops, 5, theme: create(:theme), mode: create(:mode)) + Tour.first.update(published: true) if Tour.published.empty? + Tour.last.update(published: false) if Tour.published.count == Tour.count + Tour.last.stops.tours = [] if Tour.last.stops.count > 1 + if Stop.all.all? { |s| s.published } + Stop.last.update(tours: []) + end + user = create(:user) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(Tour.count).to be > Tour.published.count + json.each do |stop| + expect(Stop.find(stop[:id]).tours.any? { |tour| tour.published }) + end + expect(json.count).to be < Stop.count + end - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # StopsController. Be sure to keep this updated too. - let(:valid_session) { {} } + it 'returns a 200 response with stops when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: tour.tenant } + expect(Stop.count).to be > 1 + expect(response.status).to eq(200) + expect(json.count).to eq(Stop.count) + end - describe 'GET #index' do - it 'returns a success response' do - stop = Stop.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + create_list(:tour_with_stops, 5, theme: create(:theme), mode: create(:mode)) + user = create(:user) + user.tour_sets = [] + user.tours << Tour.first + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(Tour.first.stops.count) + expect(json.count).to be < Stop.count end end describe 'GET #show' do - it 'returns a success response' do - stop = Stop.create! valid_attributes - get :show, params: { id: stop.to_param }, session: valid_session - expect(response).to be_success + it 'returns a 200 response that is empty stop' do + tour = create(:tour) + tour.update(published: false) + Stop.all.each { |stop| tour.stops << stop } + # Make sure the stop is only associated with the newly created tour + tour.stops.last.update(tours: [tour]) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.stops.last.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(tour.stops.last.id.to_s) + expect(attributes[:title]).to be_nil + end + + it 'returns a 200 response and stop when stop is part of published tour' do + tour = create(:tour) + tour.update(published: true) + Stop.all.each { |stop| tour.stops << stop } + tour.stops.last.update(tours: [tour]) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.stops.last.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.stops.last.title) + end + + it 'returns a 200 response that is empty stop when request is authenticated by someone w/o permission' do + tour = create(:tour) + tour.update(published: false) + Stop.all.each { |stop| tour.stops << stop } + tour.stops.last.update(tours: [tour]) + user = create(:user) + user.update(super: false) + user.tours = [] + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.stops.last.id } + expect(response.status).to eq(200) + expect(json[:id]).to eq(tour.stops.last.id.to_s) + expect(attributes[:title]).to be_nil + end + + it 'returns a 200 response that is a stop when request is authenticated by a tour author' do + tour = create(:tour) + tour.update(published: false) + Stop.all.each { |stop| tour.stops << stop } + tour.stops.first.update(tours: [tour]) + user = create(:user) + user.update(super: false) + user.tours << tour + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.stops.first.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.stops.first.title) + end + + it 'returns a 200 response that is a stop when request is authenticated by a tenant admin' do + tour = create(:tour) + tour.update(published: false) + Stop.all.each { |stop| tour.stops << stop } + tour.stops.first.update(tours: [tour]) + user = create(:user) + user.update(super: false) + user.tours = [] + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.stops.first.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.stops.first.title) + end + + it 'returns a 200 response that is a stop when request is authenticated by a super user' do + tour = create(:tour) + tour.update(published: false) + Stop.all.each { |stop| tour.stops << stop } + tour.stops.first.update(tours: [tour]) + user = create(:user) + user.update(super: true) + user.tours = [] + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.stops.first.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.stops.first.title) end end describe 'POST #create' do context 'with valid params' do - it 'creates a new Stop' do - expect { - post :create, params: { stop: valid_attributes }, session: valid_session - }.to change(Stop, :count).by(1) - end + it 'return 401 when unauthenciated' do + post :create, params: { data: { type: 'stops', attributes: { title: 'Burrito Stop' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end - it 'renders a JSON response with the new stop' do + it 'return 401 when authenciated but not an admin for current tenant' do + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + post :create, params: { data: { type: 'stops', attributes: { title: 'Burrito Stop' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end - post :create, params: { stop: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - # expect(response.location).to eq(stop_url(Stop.last)) + it 'return 201 when authenciated but an admin for current tenant' do + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + original_stop_count = Stop.count + post :create, params: { data: { type: 'stops', attributes: { title: 'Burrito Stop' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(attributes[:title]).to eq('Burrito Stop') + expect(Stop.count).to eq(original_stop_count + 1) end - end - context 'with invalid params' do - it 'renders a JSON response with errors for the new stop' do + it 'return 201 when authenciated by super' do + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + original_stop_count = Stop.count + post :create, params: { data: { type: 'stops', attributes: { title: 'Taco Stop' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(attributes[:title]).to eq('Taco Stop') + expect(Stop.count).to eq(original_stop_count + 1) + end - post :create, params: { stop: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'return 201 when authenciated by a tour author' do + user = create(:user) + user.tour_sets = [] + user.tours << Tour.last + user.update(super: false) + signed_cookie(user) + original_stop_count = Stop.count + post :create, params: { data: { type: 'stops', attributes: { title: 'Elmyr' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(attributes[:title]).to eq('Elmyr') + expect(Stop.count).to eq(original_stop_count + 1) end end end describe 'PUT #update' do context 'with valid params' do - let(:new_attributes) { - skip('Add a hash of attributes valid for your model') - } - - it 'updates the requested stop' do - stop = Stop.create! valid_attributes - put :update, params: { id: stop.to_param, stop: new_attributes }, session: valid_session - stop.reload - skip('Add assertions for updated state') + it 'return 401 when unauthenciated' do + create(:tour) + post :update, params: { id: Stop.last.id, data: { type: 'stops', attributes: { title: 'Burrito Stop' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) end - it 'renders a JSON response with the stop' do - stop = Stop.create! valid_attributes + it 'return 401 when authenciated but not an admin for current tenant' do + create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + post :update, params: { id: Stop.first.id, data: { type: 'stops', attributes: { title: 'Burrito Stop' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end - put :update, params: { id: stop.to_param, stop: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') + it 'return 200 and updated tour when authenciated but an admin for current tenant' do + create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + original_stop_title = Stop.last.title + new_title = Faker::Name.unique.name + post :update, params: { id: Stop.first.id, data: { type: 'stops', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(original_stop_title) + expect(attributes[:title]).to eq(new_title) + expect(Stop.first.title).to eq(new_title) end - end - context 'with invalid params' do - it 'renders a JSON response with errors for the stop' do - stop = Stop.create! valid_attributes + it 'return 200 and updated tour when authenciated by super' do + tour = create(:tour) + tour.stops << create_list(:stop, 4) + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + original_stop_title = Stop.last.title + new_title = Faker::Name.unique.name + post :update, params: { id: Stop.last.id, data: { type: 'stops', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(original_stop_title) + expect(attributes[:title]).to eq(new_title) + expect(Stop.last.title).to eq(new_title) + end - put :update, params: { id: stop.to_param, stop: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'return 200 and updated tour when authenciated by tour author' do + tour = create(:tour) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + original_stop_title = Stop.last.title + new_title = Faker::Name.unique.name + post :update, params: { id: Stop.first.id, data: { type: 'stops', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(original_stop_title) + expect(attributes[:title]).to eq(new_title) + expect(Stop.first.title).to eq(new_title) end end + + # context 'with invalid params' do + # it 'renders a JSON response with errors for the tour' do + # tour = Stop.create! valid_attributes + + # put :update, params: { id: tour.to_param, tour: invalid_attributes } + # expect(response).to have_http_status(:unprocessable_entity) + # expect(response.content_type).to eq('application/json') + # end + # end end describe 'DELETE #destroy' do - it 'destroys the requested stop' do - stop = Stop.create! valid_attributes - expect { - delete :destroy, params: { id: stop.to_param }, session: valid_session - }.to change(Stop, :count).by(-1) + it 'return 401 when unauthenciated' do + create(:tour) + post :destroy, params: { id: Stop.first.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) end - end + it 'return 401 when authenciated but not an admin for current tenant' do + tour = create(:tour) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + signed_cookie(user) + post :destroy, params: { id: Stop.first.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + + it 'return 204 and one less tour when authenciated but an admin for current tenant' do + tour = create(:tour) + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + stop_count = Stop.count + Stop.last.update(tours: []) + post :destroy, params: { id: Stop.last.id, tenant: Apartment::Tenant.current } + Stop.last.update(tours: []) + expect(response.status).to eq(204) + expect(Stop.count).to eq(stop_count - 1) + end + + it 'return 204 and one less tour when authenciated by super' do + tour = create(:tour) + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + Stop.first.update(tours: []) + stop_count = Stop.count + post :destroy, params: { id: Stop.first.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(Stop.count).to eq(stop_count - 1) + end + + it 'return 204 and one less stop when authenciated by tour author and Stop does not belong to a Tour' do + tour = create(:tour) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + Stop.last.update(tours: []) + stop_count = Stop.count + post :destroy, params: { id: Stop.last.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(Stop.count).to eq(stop_count - 1) + end + + it 'return 405 and does not delete Stop when Stop belongs to a Tour and requested by super' do + tour = create(:tour) + user = create(:user) + user.update(super: true) + Stop.last.tours << tour if Stop.last.tours.empty? + signed_cookie(user) + stop_count = Stop.count + post :destroy, params: { id: Stop.last.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(Stop.count).to eq(stop_count) + end + end end diff --git a/spec/controllers/v3/stops_media_controller_spec-fix.rb b/spec/controllers/v3/stops_media_controller_spec-fix.rb deleted file mode 100644 index 52ee1c5f..00000000 --- a/spec/controllers/v3/stops_media_controller_spec-fix.rb +++ /dev/null @@ -1,6 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' -RSpec.describe V3::StopsMediaController, type: :controller do - -end diff --git a/spec/controllers/v3/stops_media_controller_spec.rb b/spec/controllers/v3/stops_media_controller_spec.rb new file mode 100644 index 00000000..932c344d --- /dev/null +++ b/spec/controllers/v3/stops_media_controller_spec.rb @@ -0,0 +1,189 @@ +require 'rails_helper' + +RSpec.describe V3::StopMediaController, type: :controller do + + describe 'GET #index' do + before(:each) { Tour.all.each { |tour| tour.update(published: false) } } + + context 'unauthenticated' do + it 'returns a success response but zero StopMedium objects' do + create(:stop_medium, stop: create(:stop)) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(0) + expect(StopMedium.count).to be > 0 + end + end + + context 'authenticated unauthorized' do + it 'returns zero StopMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + stop_medium = create(:stop_medium, stop: create(:stop)) + tour_set = create(:tour_set) + user = create(:user, super: false) + user.tour_sets << tour_set + signed_cookie(user) + get :index, params: { tenant: original_tenant } + Apartment::Tenant.switch! original_tenant + expect(response.status).to eq(200) + expect(json.count).to eq(0) + expect(StopMedium.count).to be > 0 + end + + it 'returns zero StopMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + stop_medium = create(:stop_medium, stop: create(:stop)) + tour_set = create(:tour_set) + Apartment::Tenant.switch! tour_set.subdir + user = create(:user, super: false) + user.tours << create(:tour) + signed_cookie(user) + Apartment::Tenant.switch! original_tenant + get :index, params: { tenant: original_tenant } + expect(response.status).to eq(200) + expect(json.count).to eq(0) + expect(StopMedium.count).to be > 0 + end + end + + context 'authenticated and authorized' do + it 'returns all StopMedium objects to super' do + create_list(:stop_medium, 4) + user = create(:user, super: true) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(StopMedium.count) + end + + it 'returns all StopMedium objects to tenant admin' do + create_list(:stop_medium, 4) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(StopMedium.count) + end + end + end + + describe 'GET #show' do + context 'unauthenticated' do + it 'returns a success response but empty StopMedium objects' do + + stop_medium = create(:stop_medium, stop: create(:stop)) + get :show, params: { id: stop_medium.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(relationships[:medium][:data]).to be nil + expect(relationships[:stop][:data]).to be nil + expect(stop_medium.stop).not_to be nil + expect(stop_medium.medium).not_to be nil + expect(StopMedium.count).to be > 0 + end + + # it 'returns a success response but empty StopMedium objects' do + # stop_medium = create(:stop_medium, stop: create(:stop)) + # get :index, params: { id: stop_medium.id, tenant: Apartment::Tenant.current } + # expect(response.status).to eq(200) + # expect(json.count).to eq(1) + # end + end + + context 'authenticated unauthorized' do + + it 'returns empty StopMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + stop_medium = create(:stop_medium, stop: create(:stop)) + tour_set = create(:tour_set) + user = create(:user, super: false) + user.tour_sets << tour_set + signed_cookie(user) + Apartment::Tenant.switch! original_tenant + get :show, params: { id: stop_medium.id, tenant: original_tenant } + expect(response.status).to eq(200) + expect(relationships[:medium][:data]).to be nil + expect(relationships[:stop][:data]).to be nil + expect(stop_medium.stop).not_to be nil + expect(stop_medium.medium).not_to be nil + expect(StopMedium.count).to be > 0 + end + + it 'returns empty StopMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + stop_medium = create(:stop_medium, stop: create(:stop)) + tour_set = create(:tour_set) + Apartment::Tenant.switch! tour_set.subdir + user = create(:user, super: false) + user.tours << create(:tour) + signed_cookie(user) + Apartment::Tenant.switch! original_tenant + get :show, params: { id: stop_medium.id, tenant: original_tenant } + expect(response.status).to eq(200) + expect(relationships[:medium][:data]).to be nil + expect(relationships[:stop][:data]).to be nil + expect(StopMedium.count).to be > 0 + end + end + + context 'authenticated and authorized' do + it 'returns all StopMedium objects to super' do + create_list(:stop_medium, 4) + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { id: StopMedium.last.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(StopMedium.count) + end + end + end + + describe 'POST #create' do + it 'returns does not create a new StopMedium' do + expect { + post :create, params: { tenant: Apartment::Tenant.current } + }.to change(StopMedium, :count).by(0) + end + + it 'returns 401' do + user = create(:user, super: true) + signed_cookie(user) + post :create, params: { data: { type: 'stop_media', attributes: { stop_id: 1, medium_id: 1, position: 1 } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + end + + describe 'PUT #update' do + context 'with valid params' do + it 'renders a JSON response with the v3_stop_medium' do + stop_medium = create(:stop_medium, position: 1) + expect(stop_medium.position).not_to eq(100) + user = create(:user, super: true) + signed_cookie(user) + put :update, params: { id: stop_medium.id, data: { type: 'stop_media', attributes: { stop_id: stop_medium.stop.id, medium_id: stop_medium.medium.id, position: 100 } }, tenant: Apartment::Tenant.current } + expect(response).to have_http_status(:ok) + expect(attributes[:position]).to eq(100) + expect(StopMedium.find(stop_medium.id).position).to eq(100) + end + end + end + + describe 'DELETE #destroy' do + it 'does not destroy the requested v3_stop_medium' do + stop_medium = create(:stop_medium) + user = create(:user, super: true) + signed_cookie(user) + expect { + delete :destroy, params: { id: stop_medium.to_param, tenant: Apartment::Tenant.current } + }.to change(StopMedium, :count).by(0) + end + + it 'responds with 405' do + stop_medium = create(:stop_medium) + user = create(:user, super: true) + signed_cookie(user) + delete :destroy, params: { id: stop_medium.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + end + end +end diff --git a/spec/controllers/v3/themes_controller_spec.rb b/spec/controllers/v3/themes_controller_spec.rb index 7d06fb67..80557559 100644 --- a/spec/controllers/v3/themes_controller_spec.rb +++ b/spec/controllers/v3/themes_controller_spec.rb @@ -2,129 +2,42 @@ require 'rails_helper' -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - RSpec.describe V3::ThemesController, type: :controller do - # This should return the minimal set of attributes required to create a valid - # Theme. As you add validations to Theme, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip('Add a hash of attributes valid for your model') - } - - let(:invalid_attributes) { - skip('Add a hash of attributes invalid for your model') - } - - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # ThemesController. Be sure to keep this updated too. - let(:valid_session) { {} } + before(:each) { create_list(:theme, rand(3..6)) } describe 'GET #index' do it 'returns a success response' do - theme = Theme.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) end end describe 'GET #show' do it 'returns a success response' do - theme = Theme.create! valid_attributes - get :show, params: { id: theme.to_param }, session: valid_session - expect(response).to be_success + get :show, params: { id: Theme.last.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) end end describe 'POST #create' do - context 'with valid params' do - it 'creates a new Theme' do - expect { - post :create, params: { theme: valid_attributes }, session: valid_session - }.to change(Theme, :count).by(1) - end - - it 'renders a JSON response with the new theme' do - - post :create, params: { theme: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(theme_url(Theme.last)) - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the new theme' do - - post :create, params: { theme: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end + it 'returns 405' do + post :create, params: { tenant: Apartment::Tenant.current, data: { type: 'theme', attributes: {} } } + expect(response.status).to eq(405) end end describe 'PUT #update' do - context 'with valid params' do - let(:new_attributes) { - skip('Add a hash of attributes valid for your model') - } - - it 'updates the requested theme' do - theme = Theme.create! valid_attributes - put :update, params: { id: theme.to_param, theme: new_attributes }, session: valid_session - theme.reload - skip('Add assertions for updated state') - end - - it 'renders a JSON response with the theme' do - theme = Theme.create! valid_attributes - - put :update, params: { id: theme.to_param, theme: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the theme' do - theme = Theme.create! valid_attributes - - put :update, params: { id: theme.to_param, theme: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end + it 'renders a JSON response with errors for the theme' do + put :update, params: { id: Theme.first.to_param, tenant: Apartment::Tenant.current, data: { type: 'theme', attributes: {} } } + expect(response.status).to eq(405) end end describe 'DELETE #destroy' do it 'destroys the requested theme' do - theme = Theme.create! valid_attributes - expect { - delete :destroy, params: { id: theme.to_param }, session: valid_session - }.to change(Theme, :count).by(-1) + delete :destroy, params: { id: Theme.first.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) end end diff --git a/spec/controllers/v3/tour_authors_controller_spec.rb b/spec/controllers/v3/tour_authors_controller_spec.rb new file mode 100755 index 00000000..c0931e9f --- /dev/null +++ b/spec/controllers/v3/tour_authors_controller_spec.rb @@ -0,0 +1,123 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe V3::TourAuthorsController, type: :controller do + before(:each) { + create_list(:tour_set, rand(2..5)) + TourSet.all.each { |tour_set| tour_set.update(admins: create_list(:user, rand(2..5))) } + } + + describe 'GET #index' do + context 'unauthenticated and unauthorized' do + it 'returns 401 when not unauthenticated' do + get :index, params: { tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'returns 401 when authenticated but unauthorized' do + initial_tour_set = TourSet.find_by(subdir: Apartment::Tenant.current) + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + expect(user.tour_sets).not_to include initial_tour_set + signed_cookie(user) + Apartment::Tenant.switch! initial_tour_set.subdir + get :index, params: { tenant: initial_tour_set.subdir } + expect(response.status).to eq(401) + end + end + + context 'authorized' do + it 'responds with 200 and a list of TourAuthors when requested by tenant admin' do + create_list(:tour_author, rand(3..4)) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.first[:type]).to eq('tour_authors') + expect(TourAuthor.count).to be > 1 + expect(json.count).to eq(TourAuthor.count) + end + + it 'responds with 200 and a list of TourAuthors when requested by super' do + create_list(:tour_author, rand(4..6)) + user = create(:user, super: true) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.first[:type]).to eq('tour_authors') + expect(TourAuthor.count).to be > 1 + expect(json.count).to eq(TourAuthor.count) + end + end + end + + describe 'GET #show' do + context 'unauthorized' do + it 'returns 401 when unauthenticated' do + tour_author = create(:tour_author) + get :show, params: { tenant: Apartment::Tenant.current, id: tour_author.id } + expect(response.status).to eq(401) + end + + it 'returns 401 when authenticated but not authorized' do + tour_author = create(:tour_author) + initial_tour_set = TourSet.find_by(subdir: Apartment::Tenant.current) + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + expect(user.tour_sets).not_to include initial_tour_set + signed_cookie(user) + Apartment::Tenant.switch! initial_tour_set.subdir + get :show, params: { tenant: Apartment::Tenant.current, id: tour_author.id } + expect(response.status).to eq(401) + end + end + + context 'authorized' do + it 'responds with 200 and a list of TourAuthors when requested by tenant admin' do + tour_author = create(:tour_author) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour_author.id } + expect(response.status).to eq(200) + expect(json[:type]).to eq('tour_authors') + expect(json[:id]).to eq(tour_author.id.to_s) + end + + it 'responds with 200 and a list of TourAuthors when requested by super' do + tour_author = create(:tour_author) + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour_author.id } + expect(response.status).to eq(200) + expect(json[:type]).to eq('tour_authors') + expect(json[:id]).to eq(tour_author.id.to_s) + end + end + end + + describe 'POST #create' do + it 'returns 405' do + post :create, params: { tenant: Apartment::Tenant.current, data: {} } + expect(response.status).to eq(405) + end + end + + describe 'PUT #update' do + it 'returns 405' do + tour_author = create(:tour_author) + put :update, params: { tenant: Apartment::Tenant.current, id: tour_author.id } + expect(response.status).to eq(405) + end + end + + describe 'DELETE #destroy' do + it 'returns 405' do + tour_author = create(:tour_author) + delete :destroy, params: { tenant: Apartment::Tenant.current, id: tour_author.id } + expect(response.status).to eq(405) + end + end +end diff --git a/spec/controllers/v3/tour_collections_controller_spec.rb b/spec/controllers/v3/tour_collections_controller_spec.rb deleted file mode 100644 index 6d53c22a..00000000 --- a/spec/controllers/v3/tour_collections_controller_spec.rb +++ /dev/null @@ -1,131 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - -RSpec.describe V3::TourCollectionsController, type: :controller do - - # This should return the minimal set of attributes required to create a valid - # TourCollection. As you add validations to TourCollection, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip('Add a hash of attributes valid for your model') - } - - let(:invalid_attributes) { - skip('Add a hash of attributes invalid for your model') - } - - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # TourCollectionsController. Be sure to keep this updated too. - let(:valid_session) { {} } - - describe 'GET #index' do - it 'returns a success response' do - tour_collection = TourCollection.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success - end - end - - describe 'GET #show' do - it 'returns a success response' do - tour_collection = TourCollection.create! valid_attributes - get :show, params: { id: tour_collection.to_param }, session: valid_session - expect(response).to be_success - end - end - - describe 'POST #create' do - context 'with valid params' do - it 'creates a new TourCollection' do - expect { - post :create, params: { v3_tour_collection: valid_attributes }, session: valid_session - }.to change(TourCollection, :count).by(1) - end - - it 'renders a JSON response with the new v3_tour_collection' do - - post :create, params: { v3_tour_collection: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(v3_tour_collection_url(TourCollection.last)) - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the new v3_tour_collection' do - - post :create, params: { v3_tour_collection: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe 'PUT #update' do - context 'with valid params' do - let(:new_attributes) { - skip('Add a hash of attributes valid for your model') - } - - it 'updates the requested v3_tour_collection' do - tour_collection = TourCollection.create! valid_attributes - put :update, params: { id: tour_collection.to_param, v3_tour_collection: new_attributes }, session: valid_session - tour_collection.reload - skip('Add assertions for updated state') - end - - it 'renders a JSON response with the v3_tour_collection' do - tour_collection = TourCollection.create! valid_attributes - - put :update, params: { id: tour_collection.to_param, v3_tour_collection: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the v3_tour_collection' do - tour_collection = TourCollection.create! valid_attributes - - put :update, params: { id: tour_collection.to_param, v3_tour_collection: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe 'DELETE #destroy' do - it 'destroys the requested v3_tour_collection' do - tour_collection = TourCollection.create! valid_attributes - expect { - delete :destroy, params: { id: tour_collection.to_param }, session: valid_session - }.to change(TourCollection, :count).by(-1) - end - end - -end diff --git a/spec/controllers/v3/tour_flat_pages_controller_spec.rb b/spec/controllers/v3/tour_flat_pages_controller_spec.rb new file mode 100755 index 00000000..5043ef0d --- /dev/null +++ b/spec/controllers/v3/tour_flat_pages_controller_spec.rb @@ -0,0 +1,365 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe V3::TourFlatPagesController, type: :controller do + def data(tour, flat_page, position = 1) + { + type: 'tour_flat_pages', + attributes: { position: position }, + relationships: { + tour: { data: { type: 'tours', id: tour.id } }, + flat_page: { data: { type: 'flat_pages', id: flat_page.id } } + } + } + end + + describe 'GET #index' do + it 'returns a 200 response and empty tour when none are part of a published tour' do + Tour.all.each { |tour| tour.update(published: false) } + get :index, params: { tenant: Apartment::Tenant.current } + expect(json).to be_empty + expect(response.status).to eq(200) + end + + it 'returns a 200 response and only tour flat_pages that are part of a published tour' do + create_list(:tour_with_flat_pages, 5, theme: create(:theme), mode: create(:mode)) + Tour.first.update(published: true) if Tour.published.empty? + Tour.last.update(published: false) if Tour.published.count == Tour.count + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(Tour.published.map { |tour| tour.tour_flat_pages.count }.sum) + end + + it 'returns a 200 response when requeted by slug' do + tour = create(:tour_with_flat_pages) + tour.update(published: true) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(FlatPage.count) + end + + it 'returns a 200 response when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour_with_flat_pages, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(FlatPage.count) + end + + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + tour = create(:tour_with_flat_pages, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(FlatPage.count) + end + end + + describe 'GET #show' do + it 'returns a 200 response' do + tour = create(:tour_with_flat_pages) + tour.update(published: true) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_flat_pages.first.id } + expect(response.status).to eq(200) + expect(relationships[:tour][:data][:id]).to eq(tour.id.to_s) + end + + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + tour = create(:tour_with_flat_pages) + tour.update(published: false) + user = create(:user) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_flat_pages.first.id } + expect(response.status).to eq(200) + expect(relationships[:tour][:data][:id]).to eq(tour.id.to_s) + end + + it 'returns a 200 response when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour_with_flat_pages) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_flat_pages.first.id } + expect(response.status).to eq(200) + expect(relationships[:tour][:data][:id]).to eq(tour.id.to_s) + end + + it 'returns a 200 response and empty json when tour is unpublished and request is not authenticated' do + tour = create(:tour_with_flat_pages) + tour.update(published: false) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_flat_pages.first.id } + expect(response.status).to eq(200) + expect(json).to be_empty + end + + it 'returns a 200 response and empty json when tour is unpublished and request is authenticated by someone who is nither a tenant admin or tour author' do + tour = create(:tour_with_flat_pages) + tour.update(published: false) + user = create(:user) + user.tours = [] + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_flat_pages.first.id } + expect(response.status).to eq(200) + expect(json).to be_empty + end + end + + # TourFlatPage objects are NOT created via tha API. Every test should return 401 + describe 'POST #create' do + context 'with valid params' do + it 'return 405 when unauthenciated' do + tour = create(:tour) + flat_page = create(:flat_page) + post :create, params: { data: data(tour, flat_page), tenant: TourSet.first.subdir } + expect(response.status).to eq(405) + end + + it 'return 405 when authenciated but not an admin for current tenant' do + tour = create(:tour) + flat_page = create(:flat_page) + original_tour_flat_page_count = TourFlatPage.count + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + post :create, params: { data: data(tour, flat_page), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_flat_page_count).to eq(TourFlatPage.count) + end + + it 'return 405 when authenciated but an admin for current tenant' do + tour = create(:tour) + flat_page = create(:flat_page) + original_tour_flat_page_count = TourFlatPage.count + user = create(:user) + user.update(super: false) + user.tours = [] + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + post :create, params: { data: data(tour, flat_page), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_flat_page_count).to eq(TourFlatPage.count) + end + + it 'return 405 when authenciated by super' do + tour = create(:tour) + flat_page = create(:flat_page) + original_tour_flat_page_count = TourFlatPage.count + user = create(:user) + user.tours = [] + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + post :create, params: { data: data(tour, flat_page), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_flat_page_count).to eq(TourFlatPage.count) + end + + it 'return 405 when authenciated by tour author' do + tour = create(:tour) + flat_page = create(:flat_page) + original_tour_flat_page_count = TourFlatPage.count + user = create(:user) + user.tours << tour + user.tour_sets = [] + user.update(super: false) + signed_cookie(user) + post :create, params: { data: data(tour, flat_page), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_flat_page_count).to eq(TourFlatPage.count) + end + end + end + + describe 'PUT #update' do + context 'with valid params' do + it 'return 401 when unauthenciated' do + tour = create(:tour) + flat_page = create(:flat_page) + tour.flat_pages << flat_page + request_data = data(tour, flat_page, 4) + request_data[:id] = TourFlatPage.find_by(tour: tour, flat_page: flat_page).id + post :update, params: { id: request_data[:id], data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'return 401 when authenciated but not an admin for current tenant' do + tour = create(:tour) + flat_page = create(:flat_page) + tour.flat_pages << flat_page + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + request_data = data(tour, flat_page, 5) + request_data[:id] = TourFlatPage.find_by(tour: tour, flat_page: flat_page).id + post :update, params: { id: request_data[:id], data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'return 200 and updated tour when authenciated but an admin for current tenant' do + tour = create(:tour) + flat_pages = create_list(:flat_page, 5) + flat_pages.each { |flat_page| tour.flat_pages << flat_page } + tour.save + flat_page = FlatPage.find(flat_pages.first.id) + tour.flat_pages << flat_page + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + user.tours = [] + signed_cookie(user) + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + tour_flat_page.update(position: 2) + expect(TourFlatPage.find(tour_flat_page.id).position).to eq(2) + request_data = data(tour, flat_page, 5) + request_data[:id] = tour_flat_page.id + post :update, params: { id: tour_flat_page.id, data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(200) + expect(attributes[:position]).not_to eq('5') + expect(TourFlatPage.find(tour_flat_page.id).position).to eq(5) + end + + it 'return 200 and updated tour when authenciated by super' do + tour = create(:tour) + flat_pages = create_list(:flat_page, 5) + flat_pages.each { |flat_page| tour.flat_pages << flat_page } + tour.save + flat_page = FlatPage.find(flat_pages.first.id) + tour.flat_pages << flat_page + user = create(:user) + user.update(super: true) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + tour_flat_page.update(position: 3) + expect(TourFlatPage.find(tour_flat_page.id).position).to eq(3) + request_data = data(tour, flat_page, 4) + request_data[:id] = tour_flat_page.id + post :update, params: { id: tour_flat_page.id, data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(200) + expect(attributes[:position]).not_to eq('4') + expect(TourFlatPage.find(tour_flat_page.id).position).to eq(4) + end + + it 'return 200 and updated tour when authenciated by tour author' do + tour = create(:tour) + flat_pages = create_list(:flat_page, 5) + flat_pages.each { |flat_page| tour.flat_pages << flat_page } + tour.save + flat_page = FlatPage.find(flat_pages.first.id) + tour.flat_pages << flat_page + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + tour_flat_page.update(position: 6) + expect(TourFlatPage.find(tour_flat_page.id).position).to eq(6) + request_data = data(tour, flat_page, 1) + request_data[:id] = tour_flat_page.id + post :update, params: { id: tour_flat_page.id, data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(200) + expect(attributes[:position]).not_to eq('1') + expect(TourFlatPage.find(tour_flat_page.id).position).to eq(1) + end + + it 'returns 422 when params are invalid' do + tour_flat_page = create(:tour_flat_page) + user = create(:user, super: true) + invalid_params = { type: 'tour_flat_pages', attributes: {}, relationships: { tour: { data: nil }, flat_page: { data: nil } } } + signed_cookie(user) + post :update, params: { id: tour_flat_page.id, data: invalid_params, tenant: TourSet.first.subdir } + expect(response.status).to eq(422) + expect(errors).to include('Tour must exist') + end + end + end + + describe 'DELETE #destroy' do + it 'return 405 when unauthenciated' do + tour = create(:tour) + flat_page = create(:flat_page) + tour.flat_pages << flat_page + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + post :destroy, params: { id: tour_flat_page.id, tenant: TourSet.first.subdir } + expect(response.status).to eq(405) + end + + it 'return 405 when authenciated but not an admin for current tenant' do + tour = create(:tour) + flat_page = create(:flat_page) + tour.flat_pages << flat_page + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + signed_cookie(user) + post :destroy, params: { id: tour_flat_page.id, tenant: TourSet.first.subdir } + expect(response.status).to eq(405) + end + + it 'return 405 and one less tour when authenciated but an admin for current tenant' do + tour = create(:tour) + flat_page = create(:flat_page) + tour.flat_pages << flat_page + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + tour_count = Tour.count + post :destroy, params: { id: tour_flat_page.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(Tour.count).to eq(tour_count) + end + + it 'return 405 and one less tour when authenciated by super' do + tour = create(:tour) + flat_page = create(:flat_page) + tour.flat_pages << flat_page + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + tour_count = Tour.count + post :destroy, params: { id: tour_flat_page.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(Tour.count).to eq(tour_count) + end + + it 'return 405 and one less tour when authenciated by tour author' do + tour = create(:tour) + flat_page = create(:flat_page) + tour.flat_pages << flat_page + tour_flat_page = TourFlatPage.find_by(tour: tour, flat_page: flat_page) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + new_title = Faker::Name.unique.name + tour_count = Tour.count + post :destroy, params: { id: tour_flat_page.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(Tour.count).to eq(tour_count) + end + end +end diff --git a/spec/controllers/v3/tour_geojson_controller_spec.rb b/spec/controllers/v3/tour_geojson_controller_spec.rb index af122ccc..b3193615 100644 --- a/spec/controllers/v3/tour_geojson_controller_spec.rb +++ b/spec/controllers/v3/tour_geojson_controller_spec.rb @@ -2,4 +2,25 @@ RSpec.describe V3::GeojsonToursController, type: :controller do + describe 'GET #show' do + it 'returns a geojosn representation of a tour when tour published' do + tour = create(:tour, published: true, media: create_list(:medium, rand(1..3)), stops: create_list(:stop, rand(4..7))) + tour.stops.each { |stop| stop.media << create_list(:medium, rand(1..3)) } + get :show, params: { id: tour.to_param, tenant: Apartment::Tenant.current } + geojson = JSON.parse(response.body).with_indifferent_access + first_stop = Stop.find_by(title: geojson[:features].first[:properties][:title]) + expect(geojson[:type]).to eq('FeatureCollection') + expect(geojson[:features].count).to eq(tour.stops.count) + expect(geojson[:features].first[:geometry][:coordinates]).to eq([first_stop.lng.to_f, first_stop.lat.to_f]) + expect(first_stop.media.map(&:caption)).to include geojson[:features].first[:properties][:images].first[:caption] + end + + # TODO: Renable after OpenWorld stuff + # it 'returns 401 when tour is unpublished' do + # tour = create(:tour, published: false) + # get :show, params: { id: tour.to_param, tenant: Apartment::Tenant.current } + # expect(response.status).to eq(401) + # end + end + end diff --git a/spec/controllers/v3/tour_media_controller_spec.rb b/spec/controllers/v3/tour_media_controller_spec.rb index 35b29e2d..ac53ccde 100644 --- a/spec/controllers/v3/tour_media_controller_spec.rb +++ b/spec/controllers/v3/tour_media_controller_spec.rb @@ -25,105 +25,218 @@ RSpec.describe V3::TourMediaController, type: :controller do - # This should return the minimal set of attributes required to create a valid - # TourMedium. As you add validations to TourMedium, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip("Add a hash of attributes valid for your model") - } - - let(:invalid_attributes) { - skip("Add a hash of attributes invalid for your model") - } - - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # TourMediaController. Be sure to keep this updated too. - let(:valid_session) { {} } - - describe "GET #index" do - it "returns a success response" do - tour_medium = TourMedium.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success - end - end + describe 'GET #index' do + before(:each) { Tour.all.each { |tour| tour.update(published: false) } } - describe "GET #show" do - it "returns a success response" do - tour_medium = TourMedium.create! valid_attributes - get :show, params: {id: tour_medium.to_param}, session: valid_session - expect(response).to be_success - end - end + context 'unauthenticated' do + it 'returns a success response but zero TourMedium objects' do - describe "POST #create" do - context "with valid params" do - it "creates a new TourMedium" do - expect { - post :create, params: {v3_tour_medium: valid_attributes}, session: valid_session - }.to change(TourMedium, :count).by(1) + tour_medium = create(:tour_medium, tour: create(:tour, published: false)) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(0) + expect(TourMedium.count).to be > 0 end - it "renders a JSON response with the new v3_tour_medium" do + it 'returns a success response but zero TourMedium objects' do + tour_medium = create(:tour_medium, tour: create(:tour, published: true)) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(1) + end + end + + context 'authenticated unauthorized' do + + it 'returns zero TourMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + tour_medium = create(:tour_medium, tour: create(:tour, published: false)) + tour_set = create(:tour_set) + user = create(:user, super: false) + user.tour_sets << tour_set + signed_cookie(user) + get :index, params: { tenant: original_tenant } + Apartment::Tenant.switch! original_tenant + expect(response.status).to eq(200) + expect(json.count).to eq(0) + expect(TourMedium.count).to be > 0 + end - post :create, params: {v3_tour_medium: valid_attributes}, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(v3_tour_medium_url(TourMedium.last)) + it 'returns zero TourMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + tour_medium = create(:tour_medium, tour: create(:tour, published: false)) + tour_set = create(:tour_set) + Apartment::Tenant.switch! tour_set.subdir + user = create(:user, super: false) + user.tours << create(:tour) + signed_cookie(user) + Apartment::Tenant.switch! original_tenant + get :index, params: { tenant: original_tenant } + expect(response.status).to eq(200) + expect(json.count).to eq(0) + expect(TourMedium.count).to be > 0 end end - context "with invalid params" do - it "renders a JSON response with errors for the new v3_tour_medium" do + context 'authenticated and authorized' do + it 'returns all TourMedium objects to super' do + create_list(:tour_medium, 4) + user = create(:user, super: true) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(TourMedium.count) + end - post :create, params: {v3_tour_medium: invalid_attributes}, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'returns all TourMedium objects to tenant admin' do + create_list(:tour_medium, 4) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(TourMedium.count) end end end - describe "PUT #update" do - context "with valid params" do - let(:new_attributes) { - skip("Add a hash of attributes valid for your model") - } - - it "updates the requested v3_tour_medium" do - tour_medium = TourMedium.create! valid_attributes - put :update, params: {id: tour_medium.to_param, v3_tour_medium: new_attributes}, session: valid_session - tour_medium.reload - skip("Add assertions for updated state") + describe 'GET #show' do + context 'unauthenticated' do + it 'returns a success response but empty TourMedium objects' do + + tour_medium = create(:tour_medium, tour: create(:tour, published: false)) + get :show, params: { id: tour_medium.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(relationships[:medium][:data]).to be nil + expect(relationships[:tour][:data]).to be nil + expect(tour_medium.tour).not_to be nil + expect(tour_medium.medium).not_to be nil + expect(TourMedium.count).to be > 0 end - it "renders a JSON response with the v3_tour_medium" do - tour_medium = TourMedium.create! valid_attributes + # it 'returns a success response but empty TourMedium objects' do + # tour_medium = create(:tour_medium, tour: create(:tour, published: true)) + # get :index, params: { id: tour_medium.id, tenant: Apartment::Tenant.current } + # expect(response.status).to eq(200) + # expect(json.count).to eq(1) + # end + end - put :update, params: {id: tour_medium.to_param, v3_tour_medium: valid_attributes}, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') + context 'authenticated unauthorized' do + + it 'returns empty TourMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + tour_medium = create(:tour_medium, tour: create(:tour, published: false)) + tour_set = create(:tour_set) + user = create(:user, super: false) + user.tour_sets << tour_set + signed_cookie(user) + Apartment::Tenant.switch! original_tenant + get :show, params: { id: tour_medium.id, tenant: original_tenant } + expect(response.status).to eq(200) + expect(relationships[:medium][:data]).to be nil + expect(relationships[:tour][:data]).to be nil + expect(tour_medium.tour).not_to be nil + expect(tour_medium.medium).not_to be nil + expect(TourMedium.count).to be > 0 end - end - context "with invalid params" do - it "renders a JSON response with errors for the v3_tour_medium" do - tour_medium = TourMedium.create! valid_attributes + it 'returns empty TourMedium objects, not current tenant admin, non tour author' do + original_tenant = Apartment::Tenant.current + tour_medium = create(:tour_medium, tour: create(:tour, published: false)) + tour_set = create(:tour_set) + Apartment::Tenant.switch! tour_set.subdir + user = create(:user, super: false) + user.tours << create(:tour) + signed_cookie(user) + Apartment::Tenant.switch! original_tenant + get :show, params: { id: tour_medium.id, tenant: original_tenant } + expect(response.status).to eq(200) + expect(relationships[:medium][:data]).to be nil + expect(relationships[:tour][:data]).to be nil + expect(TourMedium.count).to be > 0 + end + end - put :update, params: {id: tour_medium.to_param, v3_tour_medium: invalid_attributes}, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + context 'authenticated and authorized' do + it 'returns all TourMedium objects to super' do + create_list(:tour_medium, 4) + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { id: TourMedium.last.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(TourMedium.count) end end end - describe "DELETE #destroy" do - it "destroys the requested v3_tour_medium" do - tour_medium = TourMedium.create! valid_attributes + describe 'POST #create' do + it 'returns does not create a new TourMedium' do expect { - delete :destroy, params: {id: tour_medium.to_param}, session: valid_session - }.to change(TourMedium, :count).by(-1) + post :create, params: { tenant: Apartment::Tenant.current } + }.to change(TourMedium, :count).by(0) end + + it 'returns 401' do + user = create(:user, super: true) + signed_cookie(user) + post :create, params: { data: { type: 'tour_media', attributes: { tour_id: 1, medium_id: 1, position: 1 } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + end + + describe 'PUT #update' do + context 'with valid params' do + # let(:new_attributes) { + # skip('Add a hash of attributes valid for your model') + # } + + # it 'updates the requested v3_tour_medium' do + # tour_medium = TourMedium.create! valid_attributes + # put :update, params: {id: tour_medium.to_param, v3_tour_medium: new_attributes}, session: valid_session + # tour_medium.reload + # skip('Add assertions for updated state') + # end + + it 'renders a JSON response with the v3_tour_medium' do + tour_medium = create(:tour_medium, position: 1) + expect(tour_medium.position).not_to eq(100) + user = create(:user, super: true) + signed_cookie(user) + put :update, params: { id: tour_medium.id, data: { type: 'tour_media', attributes: { tour_id: tour_medium.tour.id, medium_id: tour_medium.medium.id, position: 100 } }, tenant: Apartment::Tenant.current } + expect(response).to have_http_status(:ok) + expect(attributes[:position]).to eq(100) + expect(TourMedium.find(tour_medium.id).position).to eq(100) + end + end + + # context 'with invalid params' do + # it 'renders a JSON response with errors for the v3_tour_medium' do + # tour_medium = TourMedium.create! valid_attributes + + # put :update, params: {id: tour_medium.to_param, v3_tour_medium: invalid_attributes}, session: valid_session + # expect(response).to have_http_status(:unprocessable_entity) + # expect(response.content_type).to eq('application/json') + # end + # end end + describe 'DELETE #destroy' do + it 'does not destroy the requested v3_tour_medium' do + tour_medium = create(:tour_medium) + user = create(:user, super: true) + signed_cookie(user) + expect { + delete :destroy, params: { id: tour_medium.to_param, tenant: Apartment::Tenant.current } + }.to change(TourMedium, :count).by(0) + end + + it 'responds with 405' do + tour_medium = create(:tour_medium) + user = create(:user, super: true) + signed_cookie(user) + delete :destroy, params: { id: tour_medium.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + end + end end diff --git a/spec/controllers/v3/tour_modes_controller_spec.rb b/spec/controllers/v3/tour_modes_controller_spec.rb deleted file mode 100644 index aa2e7eac..00000000 --- a/spec/controllers/v3/tour_modes_controller_spec.rb +++ /dev/null @@ -1,7 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -RSpec.describe V3::TourModesController, type: :controller do - -end diff --git a/spec/controllers/v3/tour_set_admins_controller_spec.rb b/spec/controllers/v3/tour_set_admins_controller_spec.rb new file mode 100644 index 00000000..196290ee --- /dev/null +++ b/spec/controllers/v3/tour_set_admins_controller_spec.rb @@ -0,0 +1,81 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe V3::TourSetAdminsController, type: :controller do + before(:each) { + create_list(:tour_set, rand(2..5)) + TourSet.all.each { |tour_set| tour_set.update(admins: create_list(:user, rand(2..5))) } + } + + describe 'GET #index' do + context 'unauthenticated and unauthorized' do + it 'returns 401 when not unauthenticated' do + get :index, params: { tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'returns 401 when not unauthenticated but unauthorized' do + initial_tour_set = TourSet.first + user = create(:user, super: false) + user.tour_sets << create(:tour_set) + expect(user.tour_sets).not_to include initial_tour_set + signed_cookie(user) + Apartment::Tenant.switch! initial_tour_set.subdir + get :index, params: { tenant: initial_tour_set.subdir } + expect(response.status).to eq(401) + end + end + + context 'authorized' do + it 'responds with 200 and a list of TourSetAdmins when requested by tenant admin' do + user = create(:user, super: false) + user.tour_sets << TourSet.last + signed_cookie(user) + Apartment::Tenant.switch! TourSet.last.subdir + get :index, params: { tenant: TourSet.last.subdir } + expect(response.status).to eq(200) + expect(json.first[:type]).to eq('tour_set_admins') + expect(json.count).to eq(TourSetAdmin.count) + end + + it 'responds with 200 and a list of TourSetAdmins when requested by super' do + user = create(:user, super: true) + signed_cookie(user) + Apartment::Tenant.switch! TourSet.first.subdir + get :index, params: { tenant: TourSet.first.subdir } + expect(response.status).to eq(200) + expect(json.first[:type]).to eq('tour_set_admins') + expect(json.count).to eq(TourSetAdmin.count) + end + end + end + + describe 'GET #show' do + it 'returns 405' do + get :show, params: { tenant: Apartment::Tenant.current, id: TourSetAdmin.first.id } + expect(response.status).to eq(405) + end + end + + describe 'POST #create' do + it 'returns 405' do + post :create, params: { tenant: Apartment::Tenant.current, id: TourSetAdmin.first.id } + expect(response.status).to eq(405) + end + end + + describe 'PUT #update' do + it 'returns 405' do + put :update, params: { tenant: Apartment::Tenant.current, id: TourSetAdmin.first.id } + expect(response.status).to eq(405) + end + end + + describe 'DELETE #destroy' do + it 'returns 405' do + delete :destroy, params: { tenant: Apartment::Tenant.current, id: TourSetAdmin.first.id } + expect(response.status).to eq(405) + end + end +end diff --git a/spec/controllers/v3/tour_set_users_controller_spec.rb b/spec/controllers/v3/tour_set_users_controller_spec.rb deleted file mode 100644 index 204ad3f3..00000000 --- a/spec/controllers/v3/tour_set_users_controller_spec.rb +++ /dev/null @@ -1,129 +0,0 @@ -require 'rails_helper' - -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - -RSpec.describe V3::TourSetAdminsController, type: :controller do - - # This should return the minimal set of attributes required to create a valid - # TourSetAdmin. As you add validations to TourSetAdmin, be sure to - # adjust the attributes here as well. - let(:valid_attributes) { - skip("Add a hash of attributes valid for your model") - } - - let(:invalid_attributes) { - skip("Add a hash of attributes invalid for your model") - } - - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # TourSetAdminsController. Be sure to keep this updated too. - let(:valid_session) { {} } - - describe "GET #index" do - it "returns a success response" do - tour_set_admin = TourSetAdmin.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success - end - end - - describe "GET #show" do - it "returns a success response" do - tour_set_admin = TourSetAdmin.create! valid_attributes - get :show, params: {id: tour_set_admin.to_param}, session: valid_session - expect(response).to be_success - end - end - - describe "POST #create" do - context "with valid params" do - it "creates a new TourSetAdmin" do - expect { - post :create, params: {tour_set_admin: valid_attributes}, session: valid_session - }.to change(TourSetAdmin, :count).by(1) - end - - it "renders a JSON response with the new tour_set_admin" do - - post :create, params: {tour_set_admin: valid_attributes}, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(tour_set_admin_url(TourSetAdmin.last)) - end - end - - context "with invalid params" do - it "renders a JSON response with errors for the new tour_set_admin" do - - post :create, params: {tour_set_admin: invalid_attributes}, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe "PUT #update" do - context "with valid params" do - let(:new_attributes) { - skip("Add a hash of attributes valid for your model") - } - - it "updates the requested tour_set_admin" do - tour_set_admin = TourSetAdmin.create! valid_attributes - put :update, params: {id: tour_set_admin.to_param, tour_set_admin: new_attributes}, session: valid_session - tour_set_admin.reload - skip("Add assertions for updated state") - end - - it "renders a JSON response with the tour_set_admin" do - tour_set_admin = TourSetAdmin.create! valid_attributes - - put :update, params: {id: tour_set_admin.to_param, tour_set_admin: valid_attributes}, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') - end - end - - context "with invalid params" do - it "renders a JSON response with errors for the tour_set_admin" do - tour_set_admin = TourSetAdmin.create! valid_attributes - - put :update, params: {id: tour_set_admin.to_param, tour_set_admin: invalid_attributes}, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe "DELETE #destroy" do - it "destroys the requested tour_set_admin" do - tour_set_admin = TourSetAdmin.create! valid_attributes - expect { - delete :destroy, params: {id: tour_set_admin.to_param}, session: valid_session - }.to change(TourSetAdmin, :count).by(-1) - end - end - -end diff --git a/spec/controllers/v3/tour_sets_controller_spec.rb b/spec/controllers/v3/tour_sets_controller_spec.rb index 5727837a..5c383f5a 100644 --- a/spec/controllers/v3/tour_sets_controller_spec.rb +++ b/spec/controllers/v3/tour_sets_controller_spec.rb @@ -2,40 +2,15 @@ require 'rails_helper' -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - RSpec.describe V3::TourSetsController, type: :controller do # This should return the minimal set of attributes required to create a valid # TourSet. As you add validations to TourSet, be sure to # adjust the attributes here as well. let(:valid_attributes) { - skip('Add a hash of attributes valid for your model') } let(:invalid_attributes) { - skip('Add a hash of attributes invalid for your model') } # This should return the minimal set of values that should be in the session @@ -43,89 +18,354 @@ # TourSetsController. Be sure to keep this updated too. let(:valid_session) { {} } - describe 'GET #index' do - it 'returns a success response' do - tour_set = TourSet.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success - end - end + describe 'TourSetsController' do + let(:valid_params) { { data: { type: 'tour_sets', attributes: { name: Faker::Music::Hiphop.artist } }, tenant: 'public' } } + let(:invalid_params) { { data: { type: 'tour_sets', attributes: { name: nil } }, tenant: 'public' } } - describe 'GET #show' do - it 'returns a success response' do - tour_set = TourSet.create! valid_attributes - get :show, params: { id: tour_set.to_param }, session: valid_session - expect(response).to be_success + before(:each) do + Apartment::Tenant.reset + TourSet.all.each { |ts| ts.delete } + create_list(:tour_set, rand(3..5)) end - end - describe 'POST #create' do - context 'with valid params' do - it 'creates a new TourSet' do - expect { - post :create, params: { tour_set: valid_attributes }, session: valid_session - }.to change(TourSet, :count).by(1) + describe 'GET #index' do + it 'returns a success response but return no TourSet objects' do + get :index, params: { tenant: 'public' } + expect(response.status).to eq(200) + expect(json.count).to eq(0) end - it 'renders a JSON response with the new tour_set' do + it 'returns a success response and returns TourSet objects with tours that are published and have stops' do + Apartment::Tenant.switch! TourSet.second.subdir + tour = create(:tour, published: true) + tour.stops << create(:stop) + get :index, params: { tenant: 'public' } + expect(response.status).to eq(200) + expect(json.count).to eq(1) + end - post :create, params: { tour_set: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(tour_set_url(TourSet.last)) + it 'returns a success response by subdir but returns no TourSet objects when no published tours and not authorized' do + get :index, params: { tenant: 'public', subdir: TourSet.last.subdir } + expect(response.status).to eq(200) + expect(json.count).to eq(0) + end + + it 'returns a success response and only TourSet object by subdir when tour set has a published tour and not authorized' do + TourSet.all.each do |ts| + Apartment::Tenant.switch! ts.subdir + tour = create(:tour, published: true) + tour.stops << create(:stop) + end + Apartment::Tenant.reset + expect(TourSet.all.reject { |ts| ts.published_tours.empty? }.count).to be > 1 + get :index, params: { tenant: 'public', subdir: TourSet.second.subdir } + expect(response.status).to eq(200) + expect(json.count).to eq(1) + expect(attributes.first[:name]).to eq(TourSet.second.name) + end + + it 'returns a success response and only one TourSet object by subdir when tour set has a published tour and not authorized' do + Apartment::Tenant.switch! TourSet.second.subdir + tour = create(:tour, published: true) + tour.stops << create(:stop) + get :index, params: { tenant: 'public', subdir: TourSet.second.subdir } + expect(response.status).to eq(200) + expect(json.count).to eq(1) + end + + it 'returns all TourSet objects when requested by super' do + user = create(:user, super: true) + signed_cookie(user) + get :index, params: { tenant: 'public' } + expect(response.status).to eq(200) + expect(json.count).to eq(TourSet.count) + end + + it 'returns TourSet objects when requested by admin' do + user = create(:user, super: false) + user.tour_sets << [TourSet.first, TourSet.last] + + # Make sure a set doesn't slip in because of published tours. + [TourSet.first.subdir, TourSet.last.subdir].each do |ts| + Apartment::Tenant.switch! ts + Tour.all.update(published: false) + end + + # Make a new set with published tour to make sure it's included. + published_set = create(:tour_set) + Apartment::Tenant.switch! published_set.subdir + create(:tour, published: true, stops: create_list(:stop, 2)) + + Apartment::Tenant.reset + signed_cookie(user) + get :index, params: { tenant: 'public' } + expect(response.status).to eq(200) + expect(json.count).to be > 2 + end + + it 'returns no TourSet objects when requested by non admin' do + user = create(:user, super: false) + user.tour_sets = [] + signed_cookie(user) + get :index, params: { tenant: 'public' } + expect(response.status).to eq(200) + expect(json.count).to eq(0) + end + + it 'returns one unpublished TourSet object when requested by subdir and by super' do + user = create(:user, super: true) + signed_cookie(user) + Apartment::Tenant.switch! TourSet.second.subdir + Tour.all.each { |t| t.update(published: false) } + Apartment::Tenant.reset + get :index, params: { tenant: 'public', subdir: TourSet.second.subdir } + expect(response.status).to eq(200) + expect(json.count).to eq(1) + expect(attributes.first[:name]).to eq(TourSet.second.name) + end + + it 'returns one unpublished TourSet object when requested by subdir and by super' do + user = create(:user, super: false) + user.tour_sets << TourSet.last + signed_cookie(user) + Apartment::Tenant.switch! TourSet.last.subdir + Tour.all.each { |t| t.update(published: false) } + Apartment::Tenant.reset + get :index, params: { tenant: 'public', subdir: TourSet.last.subdir } + expect(response.status).to eq(200) + expect(json.count).to eq(1) + expect(attributes.first[:name]).to eq(TourSet.last.name) end end - context 'with invalid params' do - it 'renders a JSON response with errors for the new tour_set' do + describe 'GET #show' do + context 'unauthenticated' do + it 'returns a success response and dummy TourSet when no tour is published' do + get :show, params: { tenant: 'public', id: TourSet.first.to_param } + expect(response.status).to eq(200) + expect(attributes[:name]).to eq('....') + end + + it 'returns a success response and TourSet when TourSet has published tour' do + Apartment::Tenant.switch! TourSet.last.subdir + tour = create(:tour, published: true) + tour.stops << create(:stop) + get :show, params: { tenant: 'public', id: TourSet.last.to_param } + expect(response.status).to eq(200) + expect(attributes[:name]).to eq(TourSet.last.name) + expect(relationships[:admins][:data]).to be_empty + end + end + + context 'authenticated unauthorized' do + it 'returns a success response and dummy TourSet when no tour is published' do + user = create(:user, super: false) + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: 'public', id: TourSet.first.to_param } + expect(response.status).to eq(200) + expect(attributes[:name]).to eq('....') + end + + it 'returns a success response and TourSet when TourSet has published tour' do + user = create(:user, super: false) + user.tour_sets << TourSet.first + signed_cookie(user) + get :show, params: { tenant: 'public', id: TourSet.last.to_param } + expect(response.status).to eq(200) + expect(attributes[:name]).to eq('....') + end + end + + context 'authenticated authorized' do + it 'returns a success response and TourSet when requested by tenant adamin' do + user = create(:user, super: false) + user.tour_sets << TourSet.last + signed_cookie(user) + get :show, params: { tenant: 'public', id: TourSet.last.to_param } + expect(response.status).to eq(200) + expect(attributes[:name]).to eq(TourSet.last.name) + expect(relationships[:admins][:data].map { |admin| admin[:id] }).to include(user.id.to_s) + end - post :create, params: { tour_set: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'returns a success response and TourSet when requested by super' do + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { tenant: 'public', id: TourSet.last.to_param } + expect(response.status).to eq(200) + expect(attributes[:name]).to eq(TourSet.last.name) + end end end - end - describe 'PUT #update' do - context 'with valid params' do - let(:new_attributes) { - skip('Add a hash of attributes valid for your model') - } + describe 'POST #create' do + context 'when unauthenticated and unauthorized' do + it 'does not create a new TourSet when not super' do + expect { + post :create, params: valid_params + }.to change(TourSet, :count).by(0) + end - it 'updates the requested tour_set' do - tour_set = TourSet.create! valid_attributes - put :update, params: { id: tour_set.to_param, tour_set: new_attributes }, session: valid_session - tour_set.reload - skip('Add assertions for updated state') + it 'returns 401' do + post :create, params: valid_params + expect(response).to have_http_status(401) + end end - it 'renders a JSON response with the tour_set' do - tour_set = TourSet.create! valid_attributes + context 'when authenticated but unauthorized' do + it 'does not create a new TourSet when not super' do + user = create(:user, super: false) + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(TourSet, :count).by(0) + end + + it 'returns 401 when not super' do + user = create(:user, super: false) + signed_cookie(user) + post :create, params: valid_params + expect(response).to have_http_status(401) + end + + it 'does not create a new TourSet when not super but is a tenant admin' do + user = create(:user, super: false) + user.tour_sets << TourSet.second + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(TourSet, :count).by(0) + end - put :update, params: { id: tour_set.to_param, tour_set: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') + it 'returns 401 when not super but is a tenant admin' do + user = create(:user, super: false) + user.tour_sets << TourSet.first + signed_cookie(user) + post :create, params: valid_params + expect(response).to have_http_status(401) + end + end + + context 'when authenticated and authorized' do + context 'valid params' do + it 'creates a new TourSet' do + user = create(:user, super: true) + signed_cookie(user) + expect { + post :create, params: valid_params + }.to change(TourSet, :count).by(1) + end + + it 'renders a JSON response with the new tour_set' do + user = create(:user, super: true) + signed_cookie(user) + post :create, params: valid_params + expect(response).to have_http_status(:created) + expect(response.content_type).to eq('application/json; charset=utf-8') + end + end + end + + context 'with invalid params' do + it 'renders a JSON response with errors for the new tour_set' do + user = create(:user, super: true) + signed_cookie(user) + post :create, params: invalid_params + expect(response).to have_http_status(:unprocessable_entity) + end end end - context 'with invalid params' do - it 'renders a JSON response with errors for the tour_set' do - tour_set = TourSet.create! valid_attributes + describe 'PUT #update' do + context 'when unauthenticated and unauthorized' do + it 'returns 401' do + put :update, params: valid_params.merge({ id: TourSet.last.to_param }) + expect(response).to have_http_status(401) + end + end - put :update, params: { id: tour_set.to_param, tour_set: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + context 'when authenticated but unauthorized' do + it 'does not update a TourSet when not super' do + user = create(:user, super: false) + signed_cookie(user) + put :update, params: valid_params.merge({ id: TourSet.first.to_param }) + expect(response).to have_http_status(401) + end + + it 'allows update TourSet when not super but is a tenant admin' do + user = create(:user, super: false) + user.tour_sets << TourSet.second + signed_cookie(user) + put :update, params: valid_params.merge({ id: TourSet.second.to_param }) + expect(response).to have_http_status(200) + end + + it 'does not update TourSet when not super not a tenant admin but is a tour author' do + Apartment::Tenant.switch! TourSet.second.subdir + tour = create(:tour) + user = create(:user, super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + put :update, params: valid_params.merge({ id: TourSet.second.to_param }) + expect(response).to have_http_status(401) + end + + end + + context 'when authenticated and authorized' do + context 'valid params' do + it 'renders a JSON response with the new tour_set' do + new_name = Faker::Music::Hiphop.artist + valid_params[:data][:attributes][:name] = new_name + user = create(:user, super: true) + signed_cookie(user) + put :update, params: valid_params.merge({ id: TourSet.last.to_param }) + expect(response).to have_http_status(200) + expect(attributes[:name]).to eq(new_name) + end + end + + context 'valid params' do + it 'renders a JSON response with the new tour_set and purges icon' do + tour_set = create(:tour_set) + tour_set.update( + logo_title: Faker::File.file_name(dir: '', ext: 'png', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/base64_image.txt')) + ) + expect(TourSet.find(tour_set.id).logo.attached?).to be true + serialized_tour_set = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::TourSetSerializer.new(tour_set)).to_json).with_indifferent_access + serialized_tour_set[:data][:attributes][:base_sixty_four] = nil + serialized_tour_set[:data][:attributes][:logo] = nil + user = create(:user, super: true) + signed_cookie(user) + put :update, params: { data: serialized_tour_set[:data], id: tour_set.id, tenant: 'public' } + expect(response).to have_http_status(200) + expect(TourSet.find(tour_set.id).logo.attached?).to be false + end + end + end + + context 'with invalid params' do + it 'renders a JSON response with errors for the new tour_set' do + user = create(:user, super: true) + signed_cookie(user) + put :update, params: invalid_params.merge({ id: TourSet.first.to_param }) + expect(response).to have_http_status(:unprocessable_entity) + end end end - end - describe 'DELETE #destroy' do - it 'destroys the requested tour_set' do - tour_set = TourSet.create! valid_attributes - expect { - delete :destroy, params: { id: tour_set.to_param }, session: valid_session - }.to change(TourSet, :count).by(-1) + describe 'DELETE #destroy' do + it 'destroys the requested tour_set' do + tour_set = create(:tour_set) + user = create(:user, super: true) + signed_cookie(user) + Apartment::Tenant.reset + expect { + delete :destroy, params: { id: tour_set.to_param, tenant: Apartment::Tenant.current } + }.to change(TourSet, :count).by(-1) + end end end - end diff --git a/spec/controllers/v3/tour_stops_controller_spec.rb b/spec/controllers/v3/tour_stops_controller_spec.rb index 98b7db9f..4d26035c 100644 --- a/spec/controllers/v3/tour_stops_controller_spec.rb +++ b/spec/controllers/v3/tour_stops_controller_spec.rb @@ -3,5 +3,387 @@ require 'rails_helper' RSpec.describe V3::TourStopsController, type: :controller do + def data(tour, stop, position = 1) + { + type: 'tour_stops', + attributes: { position: position }, + relationships: { + tour: { data: { type: 'tours', id: tour.id } }, + stop: { data: { type: 'stops', id: stop.id } } + } + } + end + describe 'GET #index' do + it 'returns a 200 response and empty tour when none are part of a published tour' do + Tour.all.each { |tour| tour.update(published: false) } + get :index, params: { tenant: Apartment::Tenant.current } + expect(json).to be_empty + expect(response.status).to eq(200) + end + + it 'returns a 200 response and only tour stops that are part of a published tour' do + create_list(:tour_with_stops, 5, theme: create(:theme), mode: create(:mode)) + Tour.first.update(published: true) if Tour.published.empty? + Tour.last.update(published: false) if Tour.published.count == Tour.count + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json.count).to eq(Tour.published.map { |tour| tour.tour_stops.count }.sum) + end + + it 'returns a 200 response when requeted by slug' do + tour = create(:tour_with_stops) + tour.update(published: true) + get :index, params: { tenant: Apartment::Tenant.current, slug: tour.tour_stops.first.stop.slug, tour: tour.id } + expect(response.status).to eq(200) + expect(included.first[:attributes][:title]).to eq(tour.tour_stops.first.stop.title) + end + + it 'returns a 200 response when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour_with_stops, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current, slug: tour.tour_stops.first.stop.slug, tour: tour.id } + expect(response.status).to eq(200) + expect(included.first[:attributes][:title]).to eq(tour.tour_stops.first.stop.title) + end + + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + tour = create(:tour_with_stops, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current, slug: tour.tour_stops.first.stop.slug, tour: tour.id } + expect(response.status).to eq(200) + expect(included.first[:attributes][:title]).to eq(tour.tour_stops.first.stop.title) + end + + it 'returns empty TourStop when `fastboo` in params' do + create(:tour_with_stops, published: true) + get :index, params: { tenant: Apartment::Tenant.current, fastboot: 'true' } + expect(response.status).to eq(200) + expect(json[:id]).to eq(0) + end + + it 'returns empty json when unauthenticated but tour is not published' do + tour = create(:tour_with_stops, published: false) + get :index, params: { tenant: Apartment::Tenant.current, slug: tour.tour_stops.first.stop.slug, tour: tour.id } + expect(json).to be nil + end + + it 'returns all TourStops when requested by tenant admin' do + create_list(:tour_with_stops, rand(4..7)) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(json.count).to eq(TourStop.count) + end + end + + describe 'GET #show' do + it 'returns a 200 response' do + tours = create(:tour_with_stops) + tour = Tour.last + tour.update(published: true) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_stops.last.id } + expect(response.status).to eq(200) + expect(relationships[:tour][:data][:id]).to eq(tour.id.to_s) + end + + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + tour = create(:tour_with_stops) + tour.update(published: false) + user = create(:user) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_stops.last.id } + expect(response.status).to eq(200) + expect(relationships[:tour][:data][:id]).to eq(tour.id.to_s) + end + + it 'returns a 200 response when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour_with_stops) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_stops.last.id } + expect(response.status).to eq(200) + expect(relationships[:tour][:data][:id]).to eq(tour.id.to_s) + end + + it 'returns a 200 response and empty json when tour is unpublished and request is not authenticated' do + tour = create(:tour_with_stops) + tour.update(published: false) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_stops.last.id } + expect(response.status).to eq(200) + expect(json).to be_empty + end + + it 'returns a 200 response and empty json when tour is unpublished and request is authenticated by someone who is nither a tenant admin or tour author' do + tour = create(:tour_with_stops) + tour.update(published: false) + user = create(:user) + user.tours = [] + user.tour_sets = [] + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.tour_stops.last.id } + expect(response.status).to eq(200) + expect(json).to be_empty + end + end + + # TourStop objects are NOT created via tha API. Every test should return 401 + describe 'POST #create' do + context 'with valid params' do + it 'return 405 when unauthenciated' do + tour = create(:tour) + stop = create(:stop) + post :create, params: { data: data(tour, stop), tenant: TourSet.first.subdir } + expect(response.status).to eq(405) + end + + it 'return 405 when authenciated but not an admin for current tenant' do + tour = create(:tour) + stop = create(:stop) + original_tour_stop_count = TourStop.count + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + post :create, params: { data: data(tour, stop), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_stop_count).to eq(TourStop.count) + end + + it 'return 405 when authenciated but an admin for current tenant' do + tour = create(:tour) + stop = create(:stop) + original_tour_stop_count = TourStop.count + user = create(:user) + user.update(super: false) + user.tours = [] + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + post :create, params: { data: data(tour, stop), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_stop_count).to eq(TourStop.count) + end + + it 'return 405 when authenciated by super' do + tour = create(:tour) + stop = create(:stop) + original_tour_stop_count = TourStop.count + user = create(:user) + user.tours = [] + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + post :create, params: { data: data(tour, stop), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_stop_count).to eq(TourStop.count) + end + + it 'return 405 when authenciated by tour author' do + tour = create(:tour) + stop = create(:stop) + original_tour_stop_count = TourStop.count + user = create(:user) + user.tours << tour + user.tour_sets = [] + user.update(super: false) + signed_cookie(user) + post :create, params: { data: data(tour, stop), tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(original_tour_stop_count).to eq(TourStop.count) + end + end + end + + describe 'PUT #update' do + context 'with valid params' do + it 'return 401 when unauthenciated' do + tour = create(:tour) + stop = create(:stop) + tour.stops << stop + request_data = data(tour, stop, 4) + request_data[:id] = TourStop.find_by(tour: tour, stop: stop).id + post :update, params: { id: request_data[:id], data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'return 401 when authenciated but not an admin for current tenant' do + tour = create(:tour) + stop = create(:stop) + tour.stops << stop + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + request_data = data(tour, stop, 5) + request_data[:id] = TourStop.find_by(tour: tour, stop: stop).id + post :update, params: { id: request_data[:id], data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'return 200 and updated tour when authenciated but an admin for current tenant' do + tour = create(:tour) + stops = create_list(:stop, 5) + stops.each { |stop| tour.stops << stop } + tour.save + stop = Stop.find(stops.first.id) + tour.stops << stop + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + user.tours = [] + signed_cookie(user) + tour_stop = TourStop.find_by(tour: tour, stop: stop) + tour_stop.update(position: 2) + expect(TourStop.find(tour_stop.id).position).to eq(2) + request_data = data(tour, stop, 5) + request_data[:id] = tour_stop.id + post :update, params: { id: tour_stop.id, data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(200) + expect(attributes[:position]).not_to eq('5') + expect(TourStop.find(tour_stop.id).position).to eq(5) + end + + it 'return 200 and updated tour when authenciated by super' do + tour = create(:tour) + stops = create_list(:stop, 5) + stops.each { |stop| tour.stops << stop } + tour.save + stop = Stop.find(stops.first.id) + tour.stops << stop + user = create(:user) + user.update(super: true) + user.tour_sets = [] + user.tours = [] + signed_cookie(user) + tour_stop = TourStop.find_by(tour: tour, stop: stop) + tour_stop.update(position: 3) + expect(TourStop.find(tour_stop.id).position).to eq(3) + request_data = data(tour, stop, 4) + request_data[:id] = tour_stop.id + post :update, params: { id: tour_stop.id, data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(200) + expect(attributes[:position]).not_to eq('4') + expect(TourStop.find(tour_stop.id).position).to eq(4) + end + + it 'return 200 and updated tour when authenciated by tour author' do + tour = create(:tour) + stops = create_list(:stop, 5) + stops.each { |stop| tour.stops << stop } + tour.save + stop = Stop.find(stops.first.id) + tour.stops << stop + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + tour_stop = TourStop.find_by(tour: tour, stop: stop) + tour_stop.update(position: 6) + expect(TourStop.find(tour_stop.id).position).to eq(6) + request_data = data(tour, stop, 1) + request_data[:id] = tour_stop.id + post :update, params: { id: tour_stop.id, data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(200) + expect(attributes[:position]).not_to eq('1') + expect(TourStop.find(tour_stop.id).position).to eq(1) + end + + it 'return 422 authenciated by super but tour stop and position are nil' do + tour = create(:tour, stops: create_list(:stop, rand(4..6))) + user = create(:user, super: true) + tour_stop = TourStop.find_by(tour: tour, stop: tour.stops.first) + request_data = data(tour, tour.stops.first, 4) + request_data[:relationships][:tour][:data] = nil + request_data[:relationships][:stop][:data] = nil + request_data[:attributes][:position] = nil + signed_cookie(user) + post :update, params: { id: tour_stop.id, data: request_data, tenant: TourSet.first.subdir } + expect(response.status).to eq(422) + expect(errors).to include('Tour must exist') + expect(errors).to include('Stop must exist') + expect(errors).to include('Position can\'t be blank') + end + end + end + + describe 'DELETE #destroy' do + it 'return 405 when unauthenciated' do + tour = create(:tour) + stop = create(:stop) + tour.stops << stop + tour_stop = TourStop.find_by(tour: tour, stop: stop) + post :destroy, params: { id: tour_stop.id, tenant: TourSet.first.subdir } + expect(response.status).to eq(405) + end + + it 'return 405 when authenciated but not an admin for current tenant' do + tour = create(:tour) + stop = create(:stop) + tour.stops << stop + tour_stop = TourStop.find_by(tour: tour, stop: stop) + user = create(:user, super: false) + user.tour_sets = [] + signed_cookie(user) + post :destroy, params: { id: tour_stop.id, tenant: TourSet.first.subdir } + expect(response.status).to eq(405) + end + + it 'return 405 and one less tour when authenciated but an admin for current tenant' do + tour = create(:tour) + stop = create(:stop) + tour.stops << stop + tour_stop = TourStop.find_by(tour: tour, stop: stop) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + tour_count = Tour.count + post :destroy, params: { id: tour_stop.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(Tour.count).to eq(tour_count) + end + + it 'return 405 and one less tour when authenciated by super' do + tour = create(:tour) + stop = create(:stop) + tour.stops << stop + tour_stop = TourStop.find_by(tour: tour, stop: stop) + user = create(:user, super: true) + signed_cookie(user) + tour_count = Tour.count + post :destroy, params: { id: tour_stop.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(Tour.count).to eq(tour_count) + end + + it 'return 405 and one less tour when authenciated by tour author' do + tour = create(:tour) + stop = create(:stop) + tour.stops << stop + tour_stop = TourStop.find_by(tour: tour, stop: stop) + user = create(:user, super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + new_title = Faker::Name.unique.name + tour_count = Tour.count + post :destroy, params: { id: tour_stop.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(405) + expect(Tour.count).to eq(tour_count) + end + end end diff --git a/spec/controllers/v3/tours_controller_spec.rb b/spec/controllers/v3/tours_controller_spec.rb index f03f4ecc..635f7cca 100644 --- a/spec/controllers/v3/tours_controller_spec.rb +++ b/spec/controllers/v3/tours_controller_spec.rb @@ -2,126 +2,397 @@ require 'rails_helper' -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - RSpec.describe V3::ToursController, type: :controller do - # This should return the minimal set of attributes required to create a valid - # Tour. As you add validations to Tour, be sure to - # adjust the attributes here as well. - let(:valid_attributes) do - skip('Add a hash of attributes valid for your model') - end + describe 'GET #index' do + it 'returns a 200 response and empty tour when none found' do + StopSlug.all.each { |t| t.delete } + Stop.all.each { |t| t.delete } + Tour.all.each { |t| t.delete } + get :index, params: { tenant: 'public' } + expect(json).to be_empty + expect(response.status).to eq(200) + end - let(:invalid_attributes) do - skip('Add a hash of attributes invalid for your model') - end + it 'returns a 200 response' do + tour = create(:tour) + tour.update(published: true) + get :index, params: { tenant: tour.tenant } + expect(response.status).to eq(200) + expect(json.count).to eq(Tour.published.count) + end - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # ToursController. Be sure to keep this updated too. - let(:valid_session) { {} } + it 'returns a 200 response when requeted by slug' do + tour = create(:tour) + tour.update(published: true) + get :index, params: { tenant: tour.tenant, slug: tour.slug } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.title) + end - describe 'GET #index' do - it 'returns a success response' do - Tour.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success + # This is for when an authenticated person is viewing an unpublished tour. + # This situation occurs when Ember FastBoot tries to pre-render an unpublished + # tour. FastBoot does not have credentials to send. A 404 response causes + # FastBoot to throw an error and prevents the client from rendering. + it 'returns a 200 response and empty tour when tour is not published' do + tour = create(:tour) + tour.update(published: false) + get :index, params: { tenant: tour.tenant, slug: tour.slug } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(tour.title) + expect(attributes[:title]).to eq('....') + end + + it 'returns a 200 response when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: tour.tenant, slug: tour.slug } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.title) + end + + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + tour = create(:tour, published: false) + tour.update(published: false) + user = create(:user) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + get :index, params: { tenant: tour.tenant, slug: tour.slug } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.title) + end + + it 'returns all Tour objects when requested by tenant admin' do + create_list(:tour, rand(4..5)) + user = create(:user, super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(json.count).to eq(Tour.count) + end + + it 'returns only tours where requester is an author' do + Tour.first.update(published: true) + new_tours = create_list(:tour, rand(4..6), published: false) + user = create(:user, super: false) + user.tour_sets = [] + user.tours << [Tour.published.first, new_tours.first, new_tours.last] + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(json.count).to eq((user.tours + Tour.published).uniq.count) + expect(json.count).to be < Tour.count end end describe 'GET #show' do - it 'returns a success response' do - tour = Tour.create! valid_attributes - get :show, params: { id: tour.to_param }, session: valid_session - expect(response).to be_success + it 'returns a 200 response' do + tour = create(:tour, published: false, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, rand(5..7))) + tour.update(published: true) + tour.save + get :show, params: { tenant: tour.tenant, id: tour.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.title) + expect(attributes[:est_time]).to eq('About 2 hours bicycling') + end + + # This is for when an authenticated person is viewing an unpublished tour. + # This situation occurs when Ember FastBoot tries to pre-render an unpublished + # tour. FastBoot does not have credentials to send. A 404 response causes + # FastBoot to throw an error and prevents the client from rendering. + it 'returns a 200 response and empty tour when tour is not published' do + tour = create(:tour) + tour.update(published: false) + cookies[:auth] = nil + get :show, params: { tenant: tour.tenant, id: tour.id } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(tour.title) + expect(attributes[:title]).to eq('....') + end + + it 'returns a 200 response when request is authenticated by tour author and tour is unpublished' do + tour = create(:tour, published: false) + tour.update(published: false, media: create_list(:medium, 3)) + user = create(:user) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + get :show, params: { tenant: tour.tenant, id: tour.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.title) + end + + it 'returns a 200 response when request is authenticated by tenant admin and tour is unpublished' do + tour = create(:tour, published: false, medium: create(:medium)) + tour.update(published: false) + user = create(:user) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + get :show, params: { tenant: tour.tenant, id: tour.id } + expect(response.status).to eq(200) + expect(attributes[:title]).to eq(tour.title) + end + + it 'retuns a tour with center lat/lng based on request' do + request.env['ipinfo'] = MockIpinfo.new + tour = create(:tour, stops: []) + user = create(:user, super: true) + signed_cookie(user) + get :show, params: { tenant: Apartment::Tenant.current, id: tour.id } + expect(attributes[:bounds][:centerLat]).not_to eq(33.75432) + expect(attributes[:bounds][:centerLng]).not_to eq(-84.38979) end end describe 'POST #create' do context 'with valid params' do - it 'creates a new Tour' do - expect do - post :create, params: { tour: valid_attributes }, session: valid_session - end.to change(Tour, :count).by(1) + it 'return 401 when unauthenciated' do + post :create, params: { data: { type: 'tours', attributes: { title: 'Burrito Tour' } }, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) end - it 'renders a JSON response with the new tour' do - post :create, params: { tour: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(tour_url(Tour.last)) + it 'return 401 when authenciated but not an admin for current tenant' do + user = create(:user) + user.update(super: false) + user.tour_sets = [] + signed_cookie(user) + post :create, params: { data: { type: 'tours', attributes: { title: 'Burrito Tour' } }, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'return 201 when authenciated but an admin for current tenant' do + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + original_tour_count = Tour.count + post :create, params: { data: { type: 'tours', attributes: { title: 'Burrito Tour' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(Tour.count).to eq(original_tour_count + 1) + end + + it 'return 201 when authenciated by super' do + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + original_tour_count = Tour.count + post :create, params: { data: { type: 'tours', attributes: { title: 'Burrito Tour' } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(201) + expect(Tour.count).to eq(original_tour_count + 1) end end context 'with invalid params' do - it 'renders a JSON response with errors for the new tour' do - post :create, params: { tour: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'returns 422 when invalid attributes' do + user = create(:user, super: true) + signed_cookie(user) + original_tour_count = Tour.count + post :create, params: { data: { type: 'tours', attributes: { title: nil } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(422) + expect(Tour.count).to eq(original_tour_count) + expect(errors).to include('Title can\'t be blank') + end + + it 'returns 422 when title already used' do + user = create(:user, super: true) + signed_cookie(user) + title = Faker::Movies::HitchhikersGuideToTheGalaxy.location + create(:tour, title: title) + post :create, params: { data: { type: 'tours', attributes: { title: title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(422) + expect(Tour.where(title: title).count). to eq(1) end end end describe 'PUT #update' do context 'with valid params' do - let(:new_attributes) do - skip('Add a hash of attributes valid for your model') - end + it 'return 401 when unauthenciated' do + tour = create(:tour, published: false) + post :update, params: { id: tour.id, data: { type: 'tours', attributes: { title: 'Burrito Tour' } }, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end - it 'updates the requested tour' do - tour = Tour.create! valid_attributes - put :update, params: { id: tour.to_param, tour: new_attributes }, session: valid_session - tour.reload - skip('Add assertions for updated state') + it 'return 401 when authenciated but not an admin for current tenant' do + tour = create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + signed_cookie(user) + post :update, params: { id: tour.id, data: { type: 'tours', attributes: { title: 'Burrito Tour' } }, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) end - it 'renders a JSON response with the tour' do - tour = Tour.create! valid_attributes + it 'return 200 and updated tour when authenciated but an admin for current tenant' do + tour = create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + new_title = Faker::Name.unique.name + post :update, params: { id: tour.id, data: { type: 'tours', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(tour.title) + expect(attributes[:title]).to eq(new_title) + expect(Tour.find(tour.id).title).to eq(new_title) + end - put :update, params: { id: tour.to_param, tour: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') + it 'return 200 and updated tour when authenciated by super' do + tour = create(:tour, published: false) + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + new_title = Faker::Name.unique.name + post :update, params: { id: tour.id, data: { type: 'tours', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(tour.title) + expect(attributes[:title]).to eq(new_title) + expect(Tour.find(tour.id).title).to eq(new_title) end - end - context 'with invalid params' do - it 'renders a JSON response with errors for the tour' do - tour = Tour.create! valid_attributes + it 'return 200 and updated tour when authenciated by tour author' do + tour = create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + new_title = Faker::Name.unique.name + post :update, params: { id: tour.id, data: { type: 'tours', attributes: { title: new_title } }, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(attributes[:title]).not_to eq(tour.title) + expect(attributes[:title]).to eq(new_title) + expect(Tour.find(tour.id).title).to eq(new_title) + end - put :update, params: { id: tour.to_param, tour: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') + it 'returns 200 and adds stop to a tour' do + tour = create(:tour) + create_list(:stop, 5) + Stop.all.each { |stop| tour.stops << stop } + serialized_tour = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::TourSerializer.new(tour)).to_json).with_indifferent_access + original_stop_count = serialized_tour[:data][:relationships][:stops][:data].count + original_tour_stop_count = serialized_tour[:data][:relationships][:tour_stops][:data].count + expect(original_stop_count).to be >= 5 + expect(original_tour_stop_count).to be >= 5 + expect(original_stop_count).to eq(original_tour_stop_count) + expect(original_stop_count).to eq(tour.stops.count) + expect(original_tour_stop_count).to eq(tour.tour_stops.count) + stop = create(:stop) + expect(serialized_tour[:data][:relationships][:stops][:data].map { |s| s['id'] }).not_to include(stop.id.to_s) + serialized_tour[:data][:relationships][:stops][:data].push(JSON.parse("{\"id\":\"#{stop.id}\",\"type\":\"stops\"}")) + expect(serialized_tour[:data][:relationships][:stops][:data].count).to eq(original_stop_count + 1) + expect(serialized_tour[:data][:relationships][:stops][:data].map { |s| s['id'] }).to include(stop.id.to_s) + expect(tour.stops).not_to include(stop) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + post :update, params: { id: tour.id, data: serialized_tour[:data], tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(Tour.find(tour.id).stops).to include(stop) + expect(relationships[:stops][:data].count).to eq(original_stop_count + 1) + expect(relationships[:tour_stops][:data].count).to eq(original_tour_stop_count + 1) + end + + it 'returns 200 and removes a stop from a tour' do + tour = create(:tour) + create_list(:stop, 5) + Stop.all.each { |stop| tour.stops << stop } + serialized_tour = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::TourSerializer.new(tour)).to_json).with_indifferent_access + original_stop_count = serialized_tour[:data][:relationships][:stops][:data].count + original_tour_stop_count = serialized_tour[:data][:relationships][:tour_stops][:data].count + expect(original_stop_count).to be >= 5 + expect(original_tour_stop_count).to be >= 5 + expect(original_stop_count).to eq(original_tour_stop_count) + expect(original_stop_count).to eq(tour.stops.count) + expect(original_tour_stop_count).to eq(tour.tour_stops.count) + expect(serialized_tour[:data][:relationships][:stops][:data].count).to eq(original_stop_count) + stop = serialized_tour[:data][:relationships][:stops][:data].pop + expect(serialized_tour[:data][:relationships][:stops][:data].map { |s| s['id'] }).not_to include(stop[:id]) + expect(serialized_tour[:data][:relationships][:stops][:data].count).to eq(original_stop_count - 1) + expect(tour.stops).to include(Stop.find(stop[:id])) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + post :update, params: { id: tour.id, data: serialized_tour[:data], tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(Tour.find(tour.id).stops).not_to include(Stop.find(stop[:id])) + expect(relationships[:stops][:data].count).to eq(original_stop_count - 1) + expect(relationships[:tour_stops][:data].count).to eq(original_tour_stop_count - 1) + end + + it 'returns 422 when title in nil' do + tour = create(:tour) + serialized_tour = JSON.parse(ActiveModelSerializers::Adapter::JsonApi.new(V3::TourSerializer.new(tour)).to_json).with_indifferent_access + serialized_tour[:data][:attributes][:title] = nil + user = create(:user, super: true) + signed_cookie(user) + post :update, params: { id: tour.id, data: serialized_tour[:data], tenant: Apartment::Tenant.current } + expect(response.status).to eq(422) + expect(errors).to include('Title can\'t be blank') end end end describe 'DELETE #destroy' do - it 'destroys the requested tour' do - tour = Tour.create! valid_attributes - expect do - delete :destroy, params: { id: tour.to_param }, session: valid_session - end.to change(Tour, :count).by(-1) + it 'return 401 when unauthenciated' do + tour = create(:tour, published: false) + post :destroy, params: { id: tour.id, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'return 401 when authenciated but not an admin for current tenant' do + tour = create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + signed_cookie(user) + post :destroy, params: { id: tour.id, tenant: TourSet.first.subdir } + expect(response.status).to eq(401) + end + + it 'return 204 and one less tour when authenciated but an admin for current tenant' do + tour = create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + tour_count = Tour.count + post :destroy, params: { id: tour.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(Tour.count).to eq(tour_count - 1) + end + + it 'return 204 and one less tour when authenciated by super' do + tour = create(:tour, published: false) + user = create(:user) + user.tour_sets = [] + user.update(super: true) + signed_cookie(user) + tour_count = Tour.count + post :destroy, params: { id: tour.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(Tour.count).to eq(tour_count - 1) + end + + it 'return 204 and one less tour when authenciated by tour author' do + tour = create(:tour, published: false) + user = create(:user) + user.update(super: false) + user.tour_sets = [] + user.tours << tour + signed_cookie(user) + tour_count = Tour.count + post :destroy, params: { id: tour.id, tenant: Apartment::Tenant.current } + expect(response.status).to eq(204) + expect(Tour.count).to eq(tour_count - 1) end end end diff --git a/spec/controllers/v3/users_controller_spec-fix.rb b/spec/controllers/v3/users_controller_spec-fix.rb deleted file mode 100644 index eef9da37..00000000 --- a/spec/controllers/v3/users_controller_spec-fix.rb +++ /dev/null @@ -1,127 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -# This spec was generated by rspec-rails when you ran the scaffold generator. -# It demonstrates how one might use RSpec to specify the controller code that -# was generated by Rails when you ran the scaffold generator. -# -# It assumes that the implementation code is generated by the rails scaffold -# generator. If you are using any extension libraries to generate different -# controller code, this generated spec may or may not pass. -# -# It only uses APIs available in rails and/or rspec-rails. There are a number -# of tools you can use to make these specs even more expressive, but we're -# sticking to rails and rspec-rails APIs to keep things simple and stable. -# -# Compared to earlier versions of this generator, there is very limited use of -# stubs and message expectations in this spec. Stubs are only used when there -# is no simpler way to get a handle on the object needed for the example. -# Message expectations are only used when there is no simpler way to specify -# that an instance is receiving a specific message. -# -# Also compared to earlier versions of this generator, there are no longer any -# expectations of assigns and templates rendered. These features have been -# removed from Rails core in Rails 5, but can be added back in via the -# `rails-controller-testing` gem. - -RSpec.describe V3::UsersController, type: :controller do - # This should return the minimal set of attributes required to create a valid - # User. As you add validations to User, be sure to - # adjust the attributes here as well. - let(:valid_attributes) do - skip('Add a hash of attributes valid for your model') - end - - let(:invalid_attributes) do - skip('Add a hash of attributes invalid for your model') - end - - # This should return the minimal set of values that should be in the session - # in order to pass any filters (e.g. authentication) defined in - # UsersController. Be sure to keep this updated too. - let(:valid_session) { {} } - - describe 'GET #index' do - it 'returns a success response' do - # user = User.create! valid_attributes - get :index, params: {}, session: valid_session - expect(response).to be_success - end - end - - describe 'GET #show' do - it 'returns a success response' do - user = User.create! valid_attributes - get :show, params: { id: user.to_param }, session: valid_session - expect(response).to be_success - end - end - - describe 'POST #create' do - context 'with valid params' do - it 'creates a new User' do - expect do - post :create, params: { user: valid_attributes }, session: valid_session - end.to change(User, :count).by(1) - end - - it 'renders a JSON response with the new user' do - post :create, params: { user: valid_attributes }, session: valid_session - expect(response).to have_http_status(:created) - expect(response.content_type).to eq('application/json') - expect(response.location).to eq(user_url(User.last)) - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the new user' do - post :create, params: { user: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe 'PUT #update' do - context 'with valid params' do - let(:new_attributes) do - skip('Add a hash of attributes valid for your model') - end - - it 'updates the requested user' do - user = User.create! valid_attributes - put :update, params: { id: user.to_param, user: new_attributes }, session: valid_session - user.reload - skip('Add assertions for updated state') - end - - it 'renders a JSON response with the user' do - user = User.create! valid_attributes - - put :update, params: { id: user.to_param, user: valid_attributes }, session: valid_session - expect(response).to have_http_status(:ok) - expect(response.content_type).to eq('application/json') - end - end - - context 'with invalid params' do - it 'renders a JSON response with errors for the user' do - user = User.create! valid_attributes - - put :update, params: { id: user.to_param, user: invalid_attributes }, session: valid_session - expect(response).to have_http_status(:unprocessable_entity) - expect(response.content_type).to eq('application/json') - end - end - end - - describe 'DELETE #destroy' do - it 'destroys the requested user' do - user = User.create! valid_attributes - expect do - delete :destroy, params: { id: user.to_param }, session: valid_session - end.to change(User, :count).by(-1) - end - end -end diff --git a/spec/controllers/v3/users_controller_spec.rb b/spec/controllers/v3/users_controller_spec.rb new file mode 100644 index 00000000..872f7425 --- /dev/null +++ b/spec/controllers/v3/users_controller_spec.rb @@ -0,0 +1,279 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe V3::UsersController, type: :controller do + describe 'GET #index' do + context 'unauthorized' do + it 'returns a success response but empty json when request is unauthenticated' do + create_list(:user, rand(4..5)) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json).to be_empty + end + + it 'returns a success response but empty json when request is unauthenticated' do + create_list(:user, rand(4..5)) + user = User.last + user.update(super: false) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json).to be_empty + end + end + + context 'authorized' do + it 'returns current user when requested by current user' do + user = create(:user) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current, me: true } + expect(response.status).to eq(200) + expect(json[:id]).to eq(user.id.to_s) + end + + it 'returns list of users when requested by super' do + create_list(:user, rand(4..7)) + User.all.each {|user| user.tours << create_list(:tour, rand(0..3))} + user = User.last + user.update(super: true) + signed_cookie(user) + get :index, params: { tenant: Apartment::Tenant.current } + expect(json.count).to eq(User.count) + expect(User.all.map(&:all_tours).all? { |tours| tours.empty? }).not_to be true + expect(json.map { |user| user[:attributes][:all_tours].empty? }).to all(be true) + end + end + end + + describe 'GET #show' do + let(:user) { create(:user, super: false) } + let(:other_user) { create(:user, super: false) } + + context 'unauthorized' do + it 'returns 401 when unauthenticated' do + signed_cookie(user) + get :show, params: { id: other_user.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + + it 'returns 401 when unauthorized tenant admin' do + user.tour_sets << create(:tour_set) + signed_cookie(user) + get :show, params: { id: other_user.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + + it 'returns 401 when unauthorized tour author' do + user.tours << create(:tour) + signed_cookie(user) + get :show, params: { id: other_user.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(401) + end + end + + context 'authorized' do + it 'returns user when requested by self' do + signed_cookie(user) + get :show, params: { id: user.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json[:id]).to eq(user.id.to_s) + end + + it 'returns user when requested by super' do + user.update(super: true) + other_user.tours << create_list(:tour, 3) + signed_cookie(user) + get :show, params: { id: other_user.to_param, tenant: Apartment::Tenant.current } + expect(response.status).to eq(200) + expect(json[:id]).to eq(other_user.id.to_s) + expect(json[:attributes][:all_tours].count).to eq(3) + end + + end + end + + describe 'POST #create' do + let(:user) { create(:user, super: false) } + let(:valid_params) { { data: { type: 'users', attributes: { display_name: Faker::Music::Hiphop.artist, email: Faker::Internet.email } }, tenant: 'public' } } + + context 'unauthorized' do + it 'does not create a new User when unauthenticated' do + expect do + post :create, params: valid_params + end.to change(User, :count).by(0) + end + + it 'does not create a new User for tenant admin' do + user.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) + signed_cookie(user) + expect do + post :create, params: valid_params + end.to change(User, :count).by(0) + end + + it 'does not create a new User for tour author' do + user.tours << create(:tour) + signed_cookie(user) + expect do + post :create, params: valid_params + end.to change(User, :count).by(0) + end + end + + context 'authorized' do + it 'creates a new User for super' do + user.update(super: true) + signed_cookie(user) + expect do + post :create, params: valid_params + end.to change(User, :count).by(1) + end + + it 'does not create a new User for super with invalid_params' do + user.update(super: true) + valid_params[:data][:attributes].delete(:email) + signed_cookie(user) + expect do + post :create, params: valid_params + end.to change(User, :count).by(0) + end + + it 'responds with errors when creating a new User for super with invalid_params' do + user.update(super: true) + valid_params[:data][:attributes].delete(:email) + signed_cookie(user) + post :create, params: valid_params + expect(response.status).to eq(422) + expect(errors).to include('Email can\'t be blank') + end + end + end + + describe 'PUT #update' do + context 'unauthorized' do + it 'does not update when unauthenticated' do + user = create(:user, super: false) + initial_display_name = user.display_name + new_display_name = Faker::Music::Hiphop.artist + update_params = { id: user.to_param, tenant: 'public', data: { type: 'users', attributes: { display_name: new_display_name } } } + put :update, params: update_params + expect(response.status).to eq(401) + user.reload + expect(user.display_name).not_to eq(new_display_name) + end + + it 'does not update when authenticated as user not the one being updated' do + user = create(:user, super: false) + user_to_update = create(:user) + user.tour_sets << create(:tour_set) + initial_display_name = user_to_update.display_name + new_display_name = Faker::Music::Hiphop.artist + update_params = { id: user_to_update.to_param, tenant: 'public', data: { type: 'users', attributes: { display_name: new_display_name } } } + signed_cookie(user) + put :update, params: update_params + expect(response.status).to eq(401) + user_to_update.reload + expect(user_to_update.display_name).not_to eq(new_display_name) + end + + it 'does not update when authenticated by tenant admin' do + user = create(:user) + user_to_update = create(:user) + user.tour_sets << create(:tour_set) + initial_display_name = user_to_update.display_name + new_display_name = "#{Faker::Music::Hiphop.artist}!" + update_params = { id: user_to_update.to_param, tenant: 'public', data: { type: 'users', attributes: { display_name: new_display_name } } } + signed_cookie(user) + put :update, params: update_params + expect(response.status).to eq(401) + user_to_update.reload + expect(user_to_update.display_name).not_to eq(new_display_name) + end + + it 'does not update when authenticated by tour author' do + user = create(:user) + user_to_update = create(:user) + user.tours << create(:tour) + initial_display_name = user_to_update.display_name + new_display_name = Faker::Music::Hiphop.artist + update_params = { id: user_to_update.to_param, tenant: 'public', data: { type: 'users', attributes: { display_name: new_display_name } } } + signed_cookie(user) + put :update, params: update_params + expect(response.status).to eq(401) + user_to_update.reload + expect(user_to_update.display_name).not_to eq(new_display_name) + end + end + + context 'authorized' do + it 'updates user when requested by self' do + user = create(:user) + initial_display_name = Faker::Music::Hiphop.artist + new_display_name = Faker::Music::Hiphop.artist + update_params = { id: user.to_param, tenant: 'public', data: { type: 'users', attributes: { display_name: new_display_name } } } + signed_cookie(user) + put :update, params: update_params + expect(response.status).to eq(200) + user.reload + expect(user.display_name).to eq(new_display_name) + end + + it 'updates user when requested by super' do + user = create(:user, super: true) + user_to_update = create(:user) + initial_display_name = user_to_update.display_name + new_display_name = Faker::Music::Hiphop.artist + update_params = { id: user_to_update.to_param, tenant: 'public', data: { type: 'users', attributes: { display_name: new_display_name } } } + signed_cookie(user) + put :update, params: update_params + expect(response.status).to eq(200) + user_to_update.reload + expect(user_to_update.display_name).to eq(new_display_name) + end + + it 'returns 422 when email in nil' do + user = create(:user, super: true) + user_to_update = create(:user) + initial_email = user_to_update.email + update_params = { id: user_to_update.to_param, tenant: 'public', data: { type: 'users', attributes: { email: nil } } } + signed_cookie(user) + put :update, params: update_params + expect(response.status).to eq(422) + expect(errors).to include('Email can\'t be blank') + user_to_update.reload + expect(user_to_update.email).to eq(initial_email) + end + end + end + + describe 'DELETE #destroy' do + context 'unauthorized' do + it 'does not destroy the requested user when unauthenticated' do + user = create(:user) + expect do + delete :destroy, params: { id: user.to_param, tenant: 'public' } + end.to change(User, :count).by(0) + end + + it 'does not destroy the requested user when authenticated' do + user = create(:user) + signed_cookie(user) + expect do + delete :destroy, params: { id: user.to_param, tenant: 'public' } + end.to change(User, :count).by(0) + end + end + + context 'authorized' do + it 'destroys user when requested by super' do + super_user = create(:user, super: true) + user = create(:user) + signed_cookie(super_user) + expect do + delete :destroy, params: { id: user.to_param, tenant: 'public' } + end.to change(User, :count).by(-1) + end + end + end +end \ No newline at end of file diff --git a/spec/factories/distance_matrix.json b/spec/factories/distance_matrix.json new file mode 100644 index 00000000..11332b2b --- /dev/null +++ b/spec/factories/distance_matrix.json @@ -0,0 +1,42 @@ +{ + "destination_addresses": ["Lexington, MA, USA", "Concord, MA, USA"], + "origin_addresses": ["Boston, MA, USA", "Charlestown, Boston, MA, USA"], + "rows": + [ + { + "elements": + [ + { + "distance": { "text": "23.6 km", "value": 23644 }, + "duration": { "text": "28 mins", "value": 1673 }, + "duration_in_traffic": { "text": "34 mins", "value": 2026 }, + "status": "OK" + }, + { + "distance": { "text": "41.3 km", "value": 41294 }, + "duration": { "text": "34 mins", "value": 2063 }, + "duration_in_traffic": { "text": "37 mins", "value": 2231 }, + "status": "OK" + } + ] + }, + { + "elements": + [ + { + "distance": { "text": "31.2 km", "value": 31219 }, + "duration": { "text": "26 mins", "value": 1545 }, + "duration_in_traffic": { "text": "32 mins", "value": 1942 }, + "status": "OK" + }, + { + "distance": { "text": "29.6 km", "value": 29594 }, + "duration": { "text": "32 mins", "value": 1895 }, + "duration_in_traffic": { "text": "37 mins", "value": 2218 }, + "status": "OK" + } + ] + } + ], + "status": "OK" +} \ No newline at end of file diff --git a/spec/factories/distance_matrix2.json b/spec/factories/distance_matrix2.json new file mode 100755 index 00000000..1cf1a558 --- /dev/null +++ b/spec/factories/distance_matrix2.json @@ -0,0 +1,42 @@ +{ + "destination_addresses": ["Lexington, MA, USA", "Concord, MA, USA"], + "origin_addresses": ["Boston, MA, USA", "Charlestown, Boston, MA, USA"], + "rows": + [ + { + "elements": + [ + { + "distance": { "text": "23.6 km", "value": 23644 }, + "duration": { "text": "28 mins", "value": 1673 }, + "duration_in_traffic": { "text": "34 mins", "value": 2026 }, + "status": "OK" + }, + { + "distance": { "text": "41.3 km", "value": 41294 }, + "duration": { "text": "34 mins", "value": 1063 }, + "duration_in_traffic": { "text": "37 mins", "value": 2231 }, + "status": "OK" + } + ] + }, + { + "elements": + [ + { + "distance": { "text": "31.2 km", "value": 31219 }, + "duration": { "text": "26 mins", "value": 1545 }, + "duration_in_traffic": { "text": "32 mins", "value": 1942 }, + "status": "OK" + }, + { + "distance": { "text": "29.6 km", "value": 29594 }, + "duration": { "text": "32 mins", "value": 895 }, + "duration_in_traffic": { "text": "37 mins", "value": 2218 }, + "status": "OK" + } + ] + } + ], + "status": "OK" +} \ No newline at end of file diff --git a/spec/factories/distance_matrix_zero.json b/spec/factories/distance_matrix_zero.json new file mode 100644 index 00000000..b8da0655 --- /dev/null +++ b/spec/factories/distance_matrix_zero.json @@ -0,0 +1,16 @@ +{ + "destination_addresses": ["Lexington, MA, USA", "Concord, MA, USA"], + "origin_addresses": ["Boston, MA, USA", "Charlestown, Boston, MA, USA"], + "rows": + [ + { + "elements": + [ + { + "status": "ZERO_RESULTS" + } + ] + } + ], + "status": "OK" +} \ No newline at end of file diff --git a/spec/factories/flat_pages.rb b/spec/factories/flat_pages.rb index 6f5dcb36..220f4fa0 100644 --- a/spec/factories/flat_pages.rb +++ b/spec/factories/flat_pages.rb @@ -4,6 +4,6 @@ FactoryBot.define do factory :flat_page do title { Faker::Movies::HitchhikersGuideToTheGalaxy.planet } - content { Faker::Hipster.paragraph(sentence_count: 2, supplemental: true, random_sentences_to_add: 4) } + body { Faker::Hipster.paragraph(sentence_count: 2, supplemental: true, random_sentences_to_add: 4) } end end diff --git a/spec/factories/images/0.jpg b/spec/factories/images/0.jpg index a386aac0..2a5abaa2 100644 Binary files a/spec/factories/images/0.jpg and b/spec/factories/images/0.jpg differ diff --git a/spec/factories/images/atl.png b/spec/factories/images/atl.png new file mode 100644 index 00000000..981239af Binary files /dev/null and b/spec/factories/images/atl.png differ diff --git a/spec/factories/images/atl_base64.txt b/spec/factories/images/atl_base64.txt new file mode 100644 index 00000000..696717e1 --- /dev/null +++ b/spec/factories/images/atl_base64.txt @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/spec/factories/images/gif_base64.txt b/spec/factories/images/gif_base64.txt new file mode 100644 index 00000000..6e32523c --- /dev/null +++ b/spec/factories/images/gif_base64.txt @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/spec/factories/images/giphy.gif b/spec/factories/images/giphy.gif new file mode 100644 index 00000000..dbb996e8 Binary files /dev/null and b/spec/factories/images/giphy.gif differ diff --git a/spec/factories/images/icon_base64.txt b/spec/factories/images/icon_base64.txt new file mode 100644 index 00000000..e4d9313f --- /dev/null +++ b/spec/factories/images/icon_base64.txt @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/spec/factories/images/jp2_base64.txt b/spec/factories/images/jp2_base64.txt new file mode 100644 index 00000000..01750512 --- /dev/null +++ b/spec/factories/images/jp2_base64.txt @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/spec/factories/images/mapicon.png b/spec/factories/images/mapicon.png new file mode 100644 index 00000000..2d989b4d Binary files /dev/null and b/spec/factories/images/mapicon.png differ diff --git a/spec/factories/images/png_base64.txt b/spec/factories/images/png_base64.txt new file mode 100644 index 00000000..709c788c --- /dev/null +++ b/spec/factories/images/png_base64.txt @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/spec/factories/test.jpg b/spec/factories/images/test.jpg similarity index 100% rename from spec/factories/test.jpg rename to spec/factories/images/test.jpg diff --git a/spec/factories/login.rb b/spec/factories/login.rb index fb0d8b9f..4d7ba13c 100644 --- a/spec/factories/login.rb +++ b/spec/factories/login.rb @@ -1,16 +1,12 @@ # frozen_string_literal: true # spec/factories/login.rb -FactoryBot.define do - factory :login do - identification { Faker::Internet.email } - oauth2_token { Faker::Omniauth.google[:credentials][:token] } - uid { Faker::Number.number } - provider { 'google' } +require 'faker' +require 'jwt' - after(:create) do |login| - login.user.display_name = Faker::Name.name - login.user.save - end +FactoryBot.define do + factory :login, class: EcdsRailsAuthEngine::Login do + provider { Faker::Internet.domain_name } + user_id { nil } end end diff --git a/spec/factories/map_icons.rb b/spec/factories/map_icons.rb new file mode 100755 index 00000000..f65c076c --- /dev/null +++ b/spec/factories/map_icons.rb @@ -0,0 +1,10 @@ +# frozen_string_literal: true + +# spec/factories/map_icons.rb +FactoryBot.define do + factory :map_icon do + filename { Faker::File.file_name(dir: '', ext: 'png', directory_separator: '') } + base_sixty_four { File.read(Rails.root.join('spec/factories/images/icon_base64.txt')) } + created_at { Faker::Number.number(digits: 10) } + end +end diff --git a/spec/factories/map_overlays.rb b/spec/factories/map_overlays.rb new file mode 100755 index 00000000..345b9038 --- /dev/null +++ b/spec/factories/map_overlays.rb @@ -0,0 +1,14 @@ +# frozen_string_literal: true + +# spec/factories/map_icons.rb +FactoryBot.define do + factory :map_overlay do + south { nil } + east { nil } + north { nil } + west { nil } + filename { Faker::File.file_name(dir: '', ext: 'png', directory_separator: '') } + base_sixty_four { File.read(Rails.root.join('spec/factories/base64_image.txt')) } + association :tour + end +end diff --git a/spec/factories/media.rb b/spec/factories/media.rb index 4b81b8e7..f85290f5 100644 --- a/spec/factories/media.rb +++ b/spec/factories/media.rb @@ -5,8 +5,10 @@ factory :medium do title { Faker::TvShows::RickAndMorty.character } caption { Faker::TvShows::RickAndMorty.quote } - # original_image { Rack::Test::UploadedFile.new(Rails.root.join('spec', 'support', 'images', 'otblogo.png'), 'image/png') } - remote_original_image_url { Faker::Placeholdit.image } + filename { Faker::File.file_name(dir: '', ext: 'png', directory_separator: '') } + base_sixty_four { File.read(Rails.root.join('spec/factories/base64_image.txt')) } created_at { Faker::Number.number(digits: 10) } + video_provider { 'keiner' } + video { nil } end end diff --git a/spec/factories/stop_media.rb b/spec/factories/stop_media.rb index 96f74cf2..1d899d27 100644 --- a/spec/factories/stop_media.rb +++ b/spec/factories/stop_media.rb @@ -3,7 +3,8 @@ # spec/factories/stop_media.rb FactoryBot.define do factory :stop_medium do - stop_id { nil } - medium_id { nil } + association :stop + association :medium + position { 1 } end end diff --git a/spec/factories/stops.rb b/spec/factories/stops.rb index 8707f2bb..677a4713 100644 --- a/spec/factories/stops.rb +++ b/spec/factories/stops.rb @@ -4,6 +4,7 @@ FactoryBot.define do factory :stop do sequence :title do |s| + # This ensures unique titles "#{Faker::Movies::HitchhikersGuideToTheGalaxy.planet}#{s}" end description { Faker::Hipster.paragraph(sentence_count: 2, supplemental: true, random_sentences_to_add: 4) } @@ -18,7 +19,7 @@ # https://github.com/thoughtbot/factory_bot/blob/master/GETTING_STARTED.md#transient-attributes after(:create) do |stop, evaluator| - create_list(:medium, evaluator.media_count, stops: [stop]) + # create_list(:medium, evaluator.media_count, stops: [stop]) end end end diff --git a/spec/factories/token.rb b/spec/factories/token.rb new file mode 100755 index 00000000..a3093281 --- /dev/null +++ b/spec/factories/token.rb @@ -0,0 +1,11 @@ +# frozen_string_literal: true + +# spec/factories/login.rb +require 'faker' +require 'jwt' + +FactoryBot.define do + factory :token, class: EcdsRailsAuthEngine::Token do + token { JWT.encode(Faker::Beer.style, Faker::Address.zip, 'HS256') } + end +end diff --git a/spec/factories/tour_authors.rb b/spec/factories/tour_authors.rb index 5d4198aa..723c5f08 100644 --- a/spec/factories/tour_authors.rb +++ b/spec/factories/tour_authors.rb @@ -1,7 +1,9 @@ # frozen_string_literal: true -# spec/factories/tour_sets.rb +# spec/factories/tour_authors.rb FactoryBot.define do - factory :tour_authors do + factory :tour_author do + association :tour + association :user end end diff --git a/spec/factories/tour_flat_pages.rb b/spec/factories/tour_flat_pages.rb new file mode 100755 index 00000000..333dc389 --- /dev/null +++ b/spec/factories/tour_flat_pages.rb @@ -0,0 +1,9 @@ +# frozen_string_literal: true + +# spec/factories/tour_flat_pages.rb +FactoryBot.define do + factory :tour_flat_page do + association :tour + association :flat_page + end +end diff --git a/spec/factories/tour_media.rb b/spec/factories/tour_media.rb new file mode 100644 index 00000000..937b7656 --- /dev/null +++ b/spec/factories/tour_media.rb @@ -0,0 +1,10 @@ +# frozen_string_literal: true + +# spec/factories/tour_media.rb +FactoryBot.define do + factory :tour_medium do + association :tour + association :medium + position { 1 } + end +end diff --git a/spec/factories/tour_sets.rb b/spec/factories/tour_sets.rb index 2119d116..39bd751e 100644 --- a/spec/factories/tour_sets.rb +++ b/spec/factories/tour_sets.rb @@ -5,5 +5,7 @@ factory :tour_set do # This is to really make sure the name of the tenant is unique. name { Faker::Name.unique.name } + base_sixty_four { nil } + logo_title { nil } end end diff --git a/spec/factories/tour_stops.rb b/spec/factories/tour_stops.rb index 4a8bab19..5f9a5bf3 100644 --- a/spec/factories/tour_stops.rb +++ b/spec/factories/tour_stops.rb @@ -5,5 +5,6 @@ factory :tour_stop do tour_id { nil } stop_id { nil } + position { nil } end end diff --git a/spec/factories/tours.rb b/spec/factories/tours.rb index 4b9fe1bb..600f404c 100644 --- a/spec/factories/tours.rb +++ b/spec/factories/tours.rb @@ -6,8 +6,10 @@ title { Faker::Name.unique.name } description { "

#{Faker::TvShows::RickAndMorty.quote}

#{Faker::TvShows::RickAndMorty.quote}

#{Faker::TvShows::RickAndMorty.quote}

" } published { Faker::Boolean.boolean(true_ratio: 0.5) } - theme { Theme.create! } - mode { Mode.create! } + theme { create(:theme) } + mode { Mode.first } + link_address { Faker::Internet.url } + is_geo { true } factory :tour_with_stops do transient do @@ -16,7 +18,7 @@ # https://github.com/thoughtbot/factory_bot/blob/master/GETTING_STARTED.md#transient-attributes after(:create) do |tour, evaluator| - create_list(:stop, evaluator.stops_count, tours: [tour]) + create_list(:tour_stop, evaluator.stops_count, tour: tour, stop: create(:stop)) end end diff --git a/spec/factories/users.rb b/spec/factories/users.rb index 88b19c1e..97e4cb7a 100644 --- a/spec/factories/users.rb +++ b/spec/factories/users.rb @@ -3,7 +3,7 @@ # spec/factories/users.rb FactoryBot.define do factory :user do - login { create(:login) } - # tour_sets { TourSet.where(subdir: 'atlanta') } + email { Faker::Internet.email } + display_name { Faker::Music::Hiphop.artist } end end diff --git a/spec/models/map_icon_spec.rb b/spec/models/map_icon_spec.rb new file mode 100644 index 00000000..5ab515b1 --- /dev/null +++ b/spec/models/map_icon_spec.rb @@ -0,0 +1,13 @@ +require 'rails_helper' + +RSpec.describe MapIcon, type: :model do + context 'size error' do + it 'fails validation when image it too big' do + icon = MapIcon.create( + base_sixty_four: File.read(Rails.root.join('spec/factories/images/png_base64.txt')), + filename: Faker::File.file_name(dir: '', ext: 'png', directory_separator: '') + ) + expect(icon.errors.full_messages).to include 'Icons should be no bigger that 80 by 80 pixels' + end + end +end diff --git a/spec/models/map_overlay_spec.rb b/spec/models/map_overlay_spec.rb new file mode 100644 index 00000000..bc869cd2 --- /dev/null +++ b/spec/models/map_overlay_spec.rb @@ -0,0 +1,14 @@ +require 'rails_helper' + +RSpec.describe MapOverlay, type: :model do + it 'has nil values for south, east, north, and west' do + mo = create(:map_overlay, tour: create(:tour, stops: [])) + expect([mo.south, mo.east, mo.north, mo.west]).to all(be nil) + end + + it 'has values for south, east, north, and west based on tour stops' do + tour = create(:tour, stops: create_list(:stop, 3)) + mo = create(:map_overlay, tour: tour) + expect([mo.south, mo.east, mo.north, mo.west]).to all(be_a String) + end +end diff --git a/spec/models/medium_spec.rb b/spec/models/medium_spec.rb index 24651ce0..e6df5afb 100644 --- a/spec/models/medium_spec.rb +++ b/spec/models/medium_spec.rb @@ -5,4 +5,105 @@ RSpec.describe Medium, type: :model do it { should have_many(:stop_media) } it { should have_many(:stops) } + + context 'video' do + it 'gets image from youtube and sets embed' do + medium = create(:medium, video: 'F9ULbmCvmxY', base_sixty_four: nil, video_provider: 'youtube') + expect(medium.embed).to eq("//www.youtube.com/embed/#{medium.video}") + expect(medium.file.attached?).to be true + end + + it 'gets image from youtube when downloaded image is a StrinIO object and sets embed' do + file = File.open(Rails.root + 'spec/factories/images/atl.png') + string_io = StringIO.new(file.read) + base64 = VideoProps.encode_image(string_io) + medium = create(:medium, base_sixty_four: base64) + expect(medium.file.attached?).to be true + end + + it 'gets nothing when YouTube video is not found' do + medium = create(:medium, video: 'CvmxYF9ULbm', base_sixty_four: nil, video_provider: 'youtube') + expect(medium.embed).to be nil + expect(medium.provider).to be nil + expect(medium.file.attached?).to be false + end + + it 'gets image from vimeo and sets embed' do + medium = create(:medium, video: '310645255', base_sixty_four: nil, video_provider: 'vimeo') + expect(medium.embed).to eq("//player.vimeo.com/video/#{medium.video}") + expect(medium.file.attached?).to be true + end + + it 'gets image from soundcloud and sets embed' do + iframe = '
FiendBassy · Boca Raton (with A$AP Ferg)
' + medium = create(:medium, video: iframe, base_sixty_four: nil, video_provider: 'soundcloud') + expect(medium.embed).to eq("//w.soundcloud.com/player/?url=https%3A//api.soundcloud.com/tracks/#{medium.video}&color=%23ff5500&auto_play=false&hide_related=true&show_comments=false&show_user=false&show_reposts=false&show_teaser=false&visual=true&sharing=false") + expect(medium.file.attached?).to be true + expect(medium.title).to eq('FiendBassy: Boca Raton (with A$AP Ferg)') + end + + it 'gets default image from when no image found for soundcloud and sets embed' do + iframe = '
Emory Center for Digital Scholarship · Subsatellite Launch
' + medium = create(:medium, video: iframe, base_sixty_four: nil, video_provider: 'soundcloud') + expect(medium.embed).to eq("//w.soundcloud.com/player/?url=https%3A//api.soundcloud.com/tracks/#{medium.video}&color=%23ff5500&auto_play=false&hide_related=true&show_comments=false&show_user=false&show_reposts=false&show_teaser=false&visual=true&sharing=false") + expect(medium.file.attached?).to be true + expect(medium.title).to eq('Emory Center for Digital Scholarship') + end + + it 'replaces file for video' do + medium = create(:medium, video: 'F9ULbmCvmxY', base_sixty_four: nil, video_provider: 'youtube') + original_checksum = medium.file.blob.checksum + expect(original_checksum).to eq(Digest::MD5.file(Rails.root.join('spec/factories/images/0.jpg')).base64digest) + medium.update(base_sixty_four: File.read(Rails.root.join('spec/factories/images/png_base64.txt'))) + expect(medium.file.blob.checksum).not_to eq(original_checksum) + expect(medium.file.blob.checksum).to eq(Digest::MD5.file(Rails.root.join('spec/factories/images/atl.png')).base64digest) + end + + it 'updates title and caption of video' do + medium = create(:medium, video: 'F9ULbmCvmxY', base_sixty_four: nil, video_provider: 'youtube') + original_checksum = medium.file.blob.checksum + expect(medium.title).to include('Goodie') + expect(medium.caption).to include('Goodie') + medium.update(title: 'Outkast') + medium.update(caption: 'GOATs') + expect(medium.title).not_to include('Goodie') + expect(medium.caption).not_to include('Goodie') + expect(medium.title).to include('Outkast') + expect(medium.caption).to include('GOATs') + # medium.update(base_sixty_four: File.read(Rails.root.join('spec/factories/images/png_base64.txt'))) + # expect(medium.file.blob.checksum).not_to eq(original_checksum) + # expect(medium.file.blob.checksum).to eq(Digest::MD5.file(Rails.root.join('spec/factories/images/atl.png')).base64digest) + end + + it 'skips video_props when provider in nil' do + medium = create(:medium, video: 'ACod3', base_sixty_four: nil) + expect(medium.file.attached?).to be false + end + end + + + context 'createing images' do + it 'sets widths for variants' do + medium = create( + :medium, + filename: Faker::File.file_name(dir: '', ext: 'jpg', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/images/atl_base64.txt')), + video: nil + ) + + medium.save + expect(medium.lqip_width).not_to be nil + end + + it 'saves a gif' do + medium = create( + :medium, + filename: Faker::File.file_name(dir: '', ext: 'gif', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/images/gif_base64.txt')), + video: nil + ) + + expect(medium.file.blob.checksum).to eq('4fqkSXu+qjQuQWCms8xBBQ==') + end + end end diff --git a/spec/models/role_spec.rb b/spec/models/role_spec.rb deleted file mode 100644 index 41d40605..00000000 --- a/spec/models/role_spec.rb +++ /dev/null @@ -1,5 +0,0 @@ -require 'rails_helper' - -RSpec.describe Role, type: :model do - pending "add some examples to (or delete) #{__FILE__}" -end diff --git a/spec/models/slug_spec.rb b/spec/models/slug_spec.rb index f7511438..95b75790 100644 --- a/spec/models/slug_spec.rb +++ b/spec/models/slug_spec.rb @@ -1,5 +1,5 @@ require 'rails_helper' RSpec.describe Slug, type: :model do - pending "add some examples to (or delete) #{__FILE__}" + it { should belong_to(:tour) } end diff --git a/spec/models/stop_slug_spec.rb b/spec/models/stop_slug_spec.rb index da4362c1..475ecc26 100644 --- a/spec/models/stop_slug_spec.rb +++ b/spec/models/stop_slug_spec.rb @@ -1,5 +1,5 @@ require 'rails_helper' RSpec.describe StopSlug, type: :model do - pending "add some examples to (or delete) #{__FILE__}" + it { should belong_to(:stop) } end diff --git a/spec/models/stop_spec.rb b/spec/models/stop_spec.rb index 62a0a016..fc93751f 100644 --- a/spec/models/stop_spec.rb +++ b/spec/models/stop_spec.rb @@ -8,4 +8,17 @@ # it { should validate_presence_of(:title) } it { should have_many(:stop_media) } it { should have_many(:media) } + + it 'has specified splash' do + stop = create(:stop, medium: create(:medium)) + expect(stop.splash).not_to be nil + end + + it 'has uses the first medium for splash' do + stop = create(:stop) + create_list(:medium, 3) + Medium.all.each { |medium| stop.media << medium } + expect(stop.splash).not_to be nil + expect(stop.splash[:title]).to eq(StopMedium.find_by(position: 1).medium.title) + end end diff --git a/spec/models/stop_tag_spec.rb b/spec/models/stop_tag_spec.rb deleted file mode 100644 index 19751ec0..00000000 --- a/spec/models/stop_tag_spec.rb +++ /dev/null @@ -1,7 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -RSpec.describe StopTag, type: :model do - it { should validate_presence_of(:title) } -end diff --git a/spec/models/tour_author_spec.rb b/spec/models/tour_author_spec.rb index 6ce24081..379098d3 100644 --- a/spec/models/tour_author_spec.rb +++ b/spec/models/tour_author_spec.rb @@ -1,5 +1,6 @@ require 'rails_helper' RSpec.describe TourAuthor, type: :model do - pending "add some examples to (or delete) #{__FILE__}" + it { should belong_to(:tour) } + it { should belong_to(:user) } end diff --git a/spec/models/tour_collection_spec.rb b/spec/models/tour_collection_spec.rb deleted file mode 100644 index 7edfa7d6..00000000 --- a/spec/models/tour_collection_spec.rb +++ /dev/null @@ -1,7 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -RSpec.describe TourCollection, type: :model do - pending "add some examples to (or delete) #{__FILE__}" -end diff --git a/spec/models/tour_set_spec.rb b/spec/models/tour_set_spec.rb index 873dbf19..e299b5a8 100644 --- a/spec/models/tour_set_spec.rb +++ b/spec/models/tour_set_spec.rb @@ -4,4 +4,32 @@ RSpec.describe TourSet, type: :model do it { should validate_presence_of(:name) } + + it 'creates four travel modes' do + tour_set = create(:tour_set) + Apartment::Tenant.switch! tour_set.subdir + expect(Mode.count).to eq(4) + end + + it 'attaches logo' do + tour_set = create(:tour_set) + expect(tour_set.logo.attached?).to be false + tour_set.update( + logo_title: Faker::File.file_name(dir: '', ext: 'png', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/images/png_base64.txt')) + ) + expect(tour_set.logo.attached?).to be true + end + + it 'removes logo' do + tour_set = create( + :tour_set, + logo_title: Faker::File.file_name(dir: '', ext: 'png', directory_separator: ''), + base_sixty_four: File.read(Rails.root.join('spec/factories/images/png_base64.txt')) + ) + tour_set.save + expect(tour_set.logo.attached?).to be true + tour_set.update(base_sixty_four: nil) + expect(tour_set.logo.attached?).to be false + end end diff --git a/spec/models/tour_set_user_spec.rb b/spec/models/tour_set_user_spec.rb deleted file mode 100644 index 8ea7360b..00000000 --- a/spec/models/tour_set_user_spec.rb +++ /dev/null @@ -1,5 +0,0 @@ -require 'rails_helper' - -RSpec.describe TourSetAdmin, type: :model do - pending "add some examples to (or delete) #{__FILE__}" -end diff --git a/spec/models/tour_spec.rb b/spec/models/tour_spec.rb index df3eea38..590687d0 100644 --- a/spec/models/tour_spec.rb +++ b/spec/models/tour_spec.rb @@ -9,4 +9,154 @@ it { expect(subject).to have_many(:tour_stops) } it { expect(Tour.reflect_on_association(:theme).macro).to eq(:belongs_to) } it { expect(Tour.reflect_on_association(:mode).macro).to eq(:belongs_to) } + + it 'gets a duration' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), published: false) + tour.update(published: true) + tour.save + expect(tour.duration).to eq(7336) + end + + it 'gets no duration when unpublished' do + tour = create(:tour, mode: Mode.find_by(title: 'TRANSIT'), stops: create_list(:stop, 5), published: false) + tour.update(published: false) + tour.save + expect(tour.duration).to be nil + end + + it 'gets duration when tour is updated to published' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), published: false) + tour.update(published: false) + expect(tour.duration).to be nil + tour.update(published: true) + expect(tour.saved_change_to_attribute?(:published)).to be true + expect(tour.duration).to eq(7336) + end + + it 'updates duration when mode changes' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), published: false) + tour.update(published: true) + tour.save + expect(tour.duration).to eq(7336) + tour.mode = Mode.find_by(title: 'TRANSIT') + expect(tour.will_save_change_to_mode_id?).to be true + tour.save + expect(tour.duration).to eq(6336) + expect(tour.saved_change_to_attribute?(:duration)).to be true + expect(tour.saved_change_to_attribute?(:saved_stop_order)).to be false + end + + it 'updates duration when stop order chages' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), published: false) + 5.times { |i| create(:tour_stop, tour: tour, stop: create(:stop), position: i + 1) } + tour.update(published: true) + tour.save + expect(tour.duration).to eq(7336) + # Trick the network stub to fetch different distance matrix but doesn't presist a + # change to the tour's mode. + tour.mode.title = 'TRANSIT' + tour.tour_stops.order(:position).last.update(position: 0) + tour.validate + # Make sure duration isn't being updated because we changed the mode or published status. + expect(tour.will_save_change_to_mode_id?).to be false + expect(tour.will_save_change_to_published?).to be false + expect(tour.will_save_change_to_saved_stop_order?).to be true + tour.save + expect(tour.duration).to eq(6336) + expect(tour.saved_change_to_attribute?(:duration)).to be true + end + + it 'gets no duration whin invalid request is made to Google' do + tour = create(:tour, mode: Mode.find_by(title: 'DRIVING'), stops: create_list(:stop, 5), published: false) + tour.update(published: true) + tour.save + expect(tour.duration).to be nil + end + + it 'gets no duration whin response has ZERO_RESULTS' do + tour = create(:tour, mode: Mode.find_by(title: 'WALKING'), stops: create_list(:stop, 4), published: false) + tour.update(published: true) + tour.save + expect(tour.duration).to be nil + end + + it 'does not update the duration when other attributes are updaeted' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), published: false) + tour.update(published: true) + tour.save + expect(tour.duration).to eq(7336) + # Trick the network stub to fetch different distance matrix but doesn't presist a + # change to the tour's mode. In this case, it should NOT fetch. This is just to + # test that it does not actually make teh request when we don't want it to. + tour.mode.title = 'TRANSIT' + tour.update( + title: Faker::Music::Prince.band, + description: Faker::Music::Prince.lyric + ) + expect(tour.saved_change_to_attribute?(:title)).to be true + expect(tour.saved_change_to_attribute?(:description)).to be true + expect(tour.saved_change_to_attribute?(:duration)).to be false + expect(tour.saved_change_to_attribute?(:published)).to be false + expect(tour.saved_change_to_attribute?(:saved_stop_order)).to be false + expect(tour.duration).to eq(7336) + end + + it 'when restricted to overlay bounds, tour bounds mirror overlay' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), published: false) + mo = create(:map_overlay, tour: tour) + mo.update( + south: '33.73324867399921', + north: '33.81498938289962', + east: '-84.25453244903566', + west: '-84.37135369046021' + ) + tour.update(restrict_bounds_to_overlay: true) + expect(tour.bounds[:south]).to eq(33.723031085386665) + end + + it 'has no bounds when no stops' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), published: false) + expect(tour.bounds).to be nil + end + + it 'does not restrict bounds to overlay when no overlay' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), restrict_bounds_to_overlay: true) + expect(tour.restrict_bounds_to_overlay).to be false + end + + it 'sets restrict_bounds to false when restricted to overlay bounds' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), restrict_bounds_to_overlay: true) + mo = create(:map_overlay, tour: tour) + expect(tour.restrict_bounds).to be true + tour.update(restrict_bounds_to_overlay: true) + expect(tour.restrict_bounds).to be false + expect(tour.restrict_bounds_to_overlay).to be true + end + + it 'sets restrict_to_overlay_bounds when updated to restrict_bounds' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5), restrict_bounds_to_overlay: true) + mo = create(:map_overlay, tour: tour) + expect(tour.restrict_bounds).to be true + tour.update(restrict_bounds_to_overlay: true) + expect(tour.restrict_bounds).to be false + expect(tour.restrict_bounds_to_overlay).to be true + tour.update(restrict_bounds: true) + expect(tour.restrict_bounds).to be true + expect(tour.restrict_bounds_to_overlay).to be false + end + + it 'allows both restrictions to be false' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5)) + mo = create(:map_overlay, tour: tour) + expect(tour.restrict_bounds).to be true + tour.update(restrict_bounds: false) + expect(tour.restrict_bounds).to be false + expect(tour.restrict_bounds_to_overlay).to be false + end + + it 'will not allow restriction to overlay if no overlay' do + tour = create(:tour, mode: Mode.find_by(title: 'BICYCLING'), stops: create_list(:stop, 5)) + tour.update(restrict_bounds_to_overlay: true) + expect(tour.restrict_bounds_to_overlay).to be false + end end diff --git a/spec/models/tour_stop_spec.rb b/spec/models/tour_stop_spec.rb index 7d129373..3f7bab93 100644 --- a/spec/models/tour_stop_spec.rb +++ b/spec/models/tour_stop_spec.rb @@ -5,5 +5,4 @@ RSpec.describe TourStop, type: :model do it { should belong_to(:tour) } it { should belong_to(:stop) } - it "is not valid without a title" end diff --git a/spec/models/tour_tag_spec.rb b/spec/models/tour_tag_spec.rb deleted file mode 100644 index 9e26929f..00000000 --- a/spec/models/tour_tag_spec.rb +++ /dev/null @@ -1,7 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -RSpec.describe TourTag, type: :model do - it { should validate_presence_of(:title) } -end diff --git a/spec/models/user_spec-fix.rb b/spec/models/user_spec-fix.rb deleted file mode 100644 index ebcc26cf..00000000 --- a/spec/models/user_spec-fix.rb +++ /dev/null @@ -1,19 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' - -RSpec.describe User, type: :model do - it { should have_one(:login) } - it { expect(User.reflect_on_association(:login).macro).to eq(:has_one) } - - context 'creates login' do - it 'creates login' do - pw = Faker::Internet.password(min_length: 8) - u = User.create!(displayname: Faker::Movies::HitchhikersGuideToTheGalaxy.character) - Login.create!(identification: 'foo@bar.com', password: pw, password_confirmation: pw, user: u) - # RailsApiAuth uses `has_secure_password` The `authenticate` method returns - # the `Login` object. This just checks that the password authenticates - expect(u.login.authenticate(pw).user).to eq(u) - end - end -end diff --git a/spec/models/user_spec.rb b/spec/models/user_spec.rb new file mode 100644 index 00000000..0f1a4be1 --- /dev/null +++ b/spec/models/user_spec.rb @@ -0,0 +1,47 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe User, type: :model do + # it { should have_one(:login) } + # it { expect(User.reflect_on_association(:login).macro).to eq(:has_one) } + + context 'tour author across tenants' do + it 'lists all tours across tenants' do + # pw = Faker::Internet.password(min_length: 8) + # u = User.create!(displayname: Faker::Movies::HitchhikersGuideToTheGalaxy.character) + # Login.create!(identification: 'foo@bar.com', password: pw, password_confirmation: pw, user: u) + # # RailsApiAuth uses `has_secure_password` The `authenticate` method returns + # # the `Login` object. This just checks that the password authenticates + # expect(u.login.authenticate(pw).user).to eq(u) + TourSet.all.each { |tour_set| tour_set.delete } + user = create(:user) + create_list(:tour_set, 4) + TourSet.all.each do |tour_set| + Apartment::Tenant.switch! tour_set.subdir + user.tours << create_list(:tour, 2) + end + expect(user.all_tours.count).to eq(8) + end + end + + context 'has login' do + it 'has no provider' do + user = create(:user) + expect(user.provider).to be nil + end + + it 'has no provider' do + user = create(:user) + login = create(:login, user_id: user.id) + expect(user.provider).to eq(login.provider) + end + end + + context 'has default' do + it 'terms accepted defaults to false' do + user = create(:user) + expect(user.terms_accepted).to be(false) + end + end +end diff --git a/spec/models/v3/flat_page_spec.rb b/spec/models/v3/flat_page_spec.rb deleted file mode 100644 index 87dcab8c..00000000 --- a/spec/models/v3/flat_page_spec.rb +++ /dev/null @@ -1,5 +0,0 @@ -require 'rails_helper' - -RSpec.describe V3::FlatPage, type: :model do - pending "add some examples to (or delete) #{__FILE__}" -end diff --git a/spec/models/v3/tour_medium_spec.rb b/spec/models/v3/tour_medium_spec.rb deleted file mode 100644 index c55075d3..00000000 --- a/spec/models/v3/tour_medium_spec.rb +++ /dev/null @@ -1,5 +0,0 @@ -require 'rails_helper' - -RSpec.describe V3::TourMedium, type: :model do - pending "add some examples to (or delete) #{__FILE__}" -end diff --git a/spec/rails_helper.rb b/spec/rails_helper.rb index 47734592..8484bcbb 100644 --- a/spec/rails_helper.rb +++ b/spec/rails_helper.rb @@ -3,7 +3,7 @@ # This file is copied to spec/ when you run 'rails generate rspec:install' require 'spec_helper' ENV['RAILS_ENV'] ||= 'test' -require File.expand_path('../../config/environment', __FILE__) +require(File.expand_path('../config/environment', __dir__)) # Prevent database truncation if the environment is production abort('The Rails environment is running in production mode!') if Rails.env.production? require 'rspec/rails' @@ -43,9 +43,14 @@ # If you're not using ActiveRecord, or you'd prefer not to run each of your # examples within a transaction, remove the following line or assign false # instead of true. - config.use_transactional_fixtures = true + if ENV['DB_ADAPTER'] == 'postgresql' + # config.use_transactional_fixtures = true + end config.include RequestSpecHelper, type: :request + config.include(RequestSpecHelper, type: :controller) + config.include(SignedCookieHelper, type: :request) + config.include(SignedCookieHelper, type: :controller) config.include FactoryBot::Syntax::Methods # start by truncating all the tables but then use the faster transaction strategy the rest of the time. config.before(:suite) do @@ -66,20 +71,31 @@ # start the transaction strategy as examples are run config.around(:each) do |example| - # DatabaseCleaner.cleaning do - example.run - # end + example.run end config.before(:each) do + MiniMagick.configure do |config| + config.validate_on_create = false + end # Start transaction for this test # DatabaseCleaner.start # Switch into the default tenant Apartment::Tenant.switch! TourSet.find(TourSet.pluck(:id).sample).subdir + + # Set the host for ActiveStorage urls + ActiveStorage::Current.host = 'http://test.host' + # Switch to the below version for Rails 7 + # ActiveStorage::Current.url_options = { host: 'http://test.host' } # host! 'atlanta.lvh.me' # load Rails.root + 'db/seeds.rb' - stub_request(:any, 'https://placehold.it/300x300.png') - .to_return(body: File.open(Rails.root + 'spec/factories/images/300x300.png'), status: 200) + + # Stub a network requests + stub_request(:get, 'https://placehold.it/300x300.png_1000x1000') + .to_return( + body: File.open(Rails.root + 'spec/factories/images/0.jpg'), + status: 200 + ) stub_request(:get, 'https://vimeo.com/api/oembed.json?url=https://vimeo.com/310645255') .to_return( @@ -97,7 +113,7 @@ ) .to_return( status: 200, - body: '{ "title": "CycloramaBattleSites.org Stop 2", "thumbnail_url": "https://placehold.it/300x300.png" }', + body: '{ "title": "CycloramaBattleSites.org Stop 2", "thumbnail_url": "https://placehold.it/300x300.png", "thumbnail_width": 100, "thumbnail_height": 100 }', headers: { 'content-type': 'application/json' } ) @@ -117,6 +133,12 @@ status: 200 ) + stub_request(:get, /http:\/\/test\.host\/rails\/active_storage\/.*/) + .to_return( + body: File.open(Rails.root + 'spec/factories/images/atl.png'), + status: 200 + ) + stub_request( :get, 'https://www.googleapis.com/youtube/v3/videos?id=F9ULbmCvmxY&key=AIzaSyAafrj3VvNLJNXeW5-NNCVwY5cdB06p1_s&part=snippet' @@ -136,9 +158,47 @@ stub_request(:get, 'https://vimeo.com/https://youtu.be/F9ULbmCvmxY') .to_return(status: 404, body: '', headers: {}) - stub_request(:get, 'https://vimeo.com/https://www.youtube.com/watch?v=F9ULbmCvmxY') - .to_return(status: 404, body: '', headers: {}) + stub_request(:get, 'https://www.googleapis.com/youtube/v3/videos?id=CvmxYF9ULbm&key=AIzaSyAafrj3VvNLJNXeW5-NNCVwY5cdB06p1_s&part=snippet') + .to_return( + status: 200, + body: '{"kind": "youtube#videoListResponse", "etag": "YIUPVpqNjppyCWOZfL-19bLb7uk", "items": [ ], "pageInfo": { "totalResults": 0, "resultsPerPage": 0 } }', + headers: { 'content-type': 'application/json' } + ) + stub_request(:get, 'https://w.soundcloud.com/player/?url=https%3A//api.soundcloud.com/tracks/431162745&color=%23ff5500&auto_play=false&hide_related=true&show_comments=false&show_user=false&show_reposts=false&show_teaser=false&visual=true&sharing=false') + .to_return( + status: 200, + body: '', + headers: {} + ) + + stub_request(:get, 'https://w.soundcloud.com/player/?auto_play=false&color=%23ff5500&hide_related=true&sharing=false&show_comments=false&show_reposts=false&show_teaser=false&show_user=false&url=https://api.soundcloud.com/tracks/457871163&visual=true') + .to_return( + status: 200, + body: '
', + headers: {} + ) + + stub_request(:get, /https:\/\/i1\.sndcdn.com\/artworks-.*\.jpg/) + .to_return( + body: File.open(Rails.root + 'spec/factories/images/0.jpg'), + status: 200, + headers: {} + ) + + stub_request(:get, /http:\/\/127\.0\.0\.1:.*\/json\/version/).to_return(body: '{}', status: 200) + + stub_request(:get, /http.*:\/\/maps\.googleapis\.com\/maps\/api\/.*BICYCLING.*/) + .to_return(body: File.read(Rails.root + 'spec/factories/distance_matrix.json'), status: 200, headers: { 'Content-Type': 'application/json' }) + + stub_request(:get, /http.*:\/\/maps\.googleapis\.com\/maps\/api\/.*TRANSIT.*/) + .to_return(body: File.read(Rails.root + 'spec/factories/distance_matrix2.json'), status: 200, headers: { 'Content-Type': 'application/json' }) + + stub_request(:get, /http.*:\/\/maps\.googleapis\.com\/maps\/api\/.*WALKING.*/) + .to_return(body: File.read(Rails.root + 'spec/factories/distance_matrix_zero.json'), status: 200, headers: { 'Content-Type': 'application/json' }) + + stub_request(:get, /http.*:\/\/maps\.googleapis\.com\/maps\/api\/.*DRIVING.*/) + .to_return(body: '{"status": "INVALID_REQUEST"}', status: 200, headers: { 'Content-Type': 'application/json' }) end config.after(:each) do @@ -178,6 +238,17 @@ end config.after(:suite) do - TourSet.all.each { |ts| ts.destroy } + # TourSet.all.each { |ts| ts.destroy } + end + + # Class to mock IPinfo + class MockIpinfo + def longitude + Faker::Address.longitude + end + + def latitude + Faker::Address.latitude + end end end diff --git a/spec/requests/tours_request_spec.rb b/spec/requests/tours_request_spec.rb new file mode 100755 index 00000000..a0b877c0 --- /dev/null +++ b/spec/requests/tours_request_spec.rb @@ -0,0 +1,19 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe 'V3::Tours', type: :request do + describe 'GET /:tenant/tours' do + before { + get "/#{Apartment::Tenant.current}/tours", headers: { 'HTTP_USER_AGENT': 'bot' } + } + + it 'returns only published tours' do + expect(json.size).to eq(Tour.published.count) + end + + it 'returns status code 200' do + expect(response).to have_http_status(200) + end + end +end diff --git a/spec/requests/v3/flat_pages_spec.rb b/spec/requests/v3/flat_pages_spec.rb deleted file mode 100644 index 6a32be05..00000000 --- a/spec/requests/v3/flat_pages_spec.rb +++ /dev/null @@ -1,117 +0,0 @@ -# frozen_string_literal: true - -require 'rails_helper' -require 'ecds_rails_auth_engine' - -RSpec.describe 'V3::FlatPages', type: :request do - let!(:theme) { create(:theme) } - let!(:tours) { create_list(:tour_with_flat_pages, 1, theme: theme) } - let!(:flat_page) { tours.first.flat_pages.first } - let(:tour_id) { tours.first.id } - - context 'create tour with flat pages' do - before { get "/#{Apartment::Tenant.current}/flat-pages" } - - it 'associates flat_page with tour' do - expect(response).to have_http_status(200) - expect(json.size).to eq(3) - end - end - - context 'flat page included in tours payload' do - before { - tour = create(:tour_with_flat_pages, theme: theme) - tour.published = true - tour.save - get "/#{Apartment::Tenant.current}/tours?slug=#{tour.slug}" - } - - it 'creates tour with existing flat page' do - expect(included.select { |d| d['type'] == 'tour_flat_pages' }.length).to eq(3) - end - end - - context 'get specific flat page by id' do - before { get "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}" } - - it 'returns requested flat page' do - expect(response).to have_http_status(200) - expect(attributes['title']).to eq(FlatPage.first.title) - end - end - - # valid payload - let(:valid_attributes) do - factory_to_json_api(FactoryBot.build(:flat_page)) - end - - describe 'POST /tenant/flat-pages' do - - context 'create page not authenticated' do - before { post "/#{Apartment::Tenant.current}/flat-pages", params: valid_attributes } - it 'returns status code 401' do - expect(response).to have_http_status(401) - end - end - - context 'when created by tour set admin' do - before { User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) } - before { post "/#{Apartment::Tenant.current}/flat-pages", params: valid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } - it 'creates a tour' do - expect(response).to have_http_status(201) - end - end - - context 'create without valid params' do - before { post "/#{Apartment::Tenant.current}/flat-pages", params: {foo: 'bar'}, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } - it 'returns unprocessable entity' do - expect(response).to have_http_status(422) - end - end - end - - describe 'PUT /tenant/flat-pages/' do - context 'update page not authenticated' do - before { put "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes} - it 'returns status code 401' do - expect(response).to have_http_status(401) - end - end - - context 'when updated by tour set admin' do - before { User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) } - before { put "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } - it 'creates a tour' do - expect(response).to have_http_status(200) - end - end - - context 'update without valid params' do - before { - invalid_attributes = {data: {type: 'flat_pages', attributes: {title: nil}}} - put "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.last.id}", params: invalid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } - } - it 'returns unprocessable entity' do - expect(response).to have_http_status(422) - end - end - - end - - describe 'DELETE /tenant/flat-pages/' do - context 'delete page not authenticated' do - before { delete "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes} - it 'returns status code 401' do - expect(response).to have_http_status(401) - end - end - - context 'when deletes by tour set admin' do - before { User.first.tour_sets << TourSet.find_by(subdir: Apartment::Tenant.current) } - before { delete "/#{Apartment::Tenant.current}/flat-pages/#{FlatPage.first.id}", params: valid_attributes, headers: { Authorization: "Bearer #{User.first.login.oauth2_token}" } } - it 'creates a tour' do - expect(response).to have_http_status(204) - end - end - end -end diff --git a/spec/routing/map_icons_routing_spec.rb.fix b/spec/routing/map_icons_routing_spec.rb.fix new file mode 100644 index 00000000..7ed19400 --- /dev/null +++ b/spec/routing/map_icons_routing_spec.rb.fix @@ -0,0 +1,30 @@ +require "rails_helper" + +RSpec.describe MapIconsController, type: :routing do + describe "routing" do + it "routes to #index" do + expect(get: "/map_icons").to route_to("map_icons#index") + end + + it "routes to #show" do + expect(get: "/map_icons/1").to route_to("map_icons#show", id: "1") + end + + + it "routes to #create" do + expect(post: "/map_icons").to route_to("map_icons#create") + end + + it "routes to #update via PUT" do + expect(put: "/map_icons/1").to route_to("map_icons#update", id: "1") + end + + it "routes to #update via PATCH" do + expect(patch: "/map_icons/1").to route_to("map_icons#update", id: "1") + end + + it "routes to #destroy" do + expect(delete: "/map_icons/1").to route_to("map_icons#destroy", id: "1") + end + end +end diff --git a/spec/routing/tour_authors_routing_spec.rb.fix b/spec/routing/tour_authors_routing_spec.rb.fix new file mode 100644 index 00000000..2945dc40 --- /dev/null +++ b/spec/routing/tour_authors_routing_spec.rb.fix @@ -0,0 +1,30 @@ +require "rails_helper" + +RSpec.describe TourAuthorsController, type: :routing do + describe "routing" do + it "routes to #index" do + expect(get: "/tour_authors").to route_to("tour_authors#index") + end + + it "routes to #show" do + expect(get: "/tour_authors/1").to route_to("tour_authors#show", id: "1") + end + + + it "routes to #create" do + expect(post: "/tour_authors").to route_to("tour_authors#create") + end + + it "routes to #update via PUT" do + expect(put: "/tour_authors/1").to route_to("tour_authors#update", id: "1") + end + + it "routes to #update via PATCH" do + expect(patch: "/tour_authors/1").to route_to("tour_authors#update", id: "1") + end + + it "routes to #destroy" do + expect(delete: "/tour_authors/1").to route_to("tour_authors#destroy", id: "1") + end + end +end diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index 1fe4732a..77c71181 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -1,10 +1,44 @@ # frozen_string_literal: true +# if ENV['COV'] == 'simple' +# require 'simplecov' +# else +# require 'coveralls' +# Coveralls.wear! +# end + +# require 'simplecov' +# SimpleCov.start + require 'coveralls' -Coveralls.wear! +require 'simplecov' +require 'simplecov-lcov' + +SimpleCov::Formatter::LcovFormatter.config.report_with_single_file = true + +SimpleCov.formatter = SimpleCov::Formatter::MultiFormatter.new( + [ + SimpleCov::Formatter::HTMLFormatter, + Coveralls::SimpleCov::Formatter, + SimpleCov::Formatter::LcovFormatter + ] +) + +# SimpleCov.start 'rails' +SimpleCov.start 'rails' do + add_filter [ + '/lib/snippets.rb', + '/app/channels/', + '/app/mailers/', + '/app/jobs/', + '/app/uploaders/' +] +end require 'webmock/rspec' WebMock.disable_net_connect!(allow_localhost: true) +WebMock.disable_net_connect!(allow: '45.33.24.119') +WebMock.disable_net_connect!(allow: 'http://test.host') # This file was generated by the `rails generate rspec:install` command. Conventionally, all # specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`. diff --git a/spec/string_spec.rb b/spec/string_spec.rb new file mode 100644 index 00000000..c87d4411 --- /dev/null +++ b/spec/string_spec.rb @@ -0,0 +1,16 @@ +# frozen_string_literal: true + +require 'rails_helper' + +RSpec.describe 'Using custom parameterize_intl method' do + it 'uses custom parameterize_intl method' do + expect('My Awesome Tour'.parameterize_intl).to eq('my-awesome-tour') + expect('Csodálatos túrám'.parameterize_intl).to eq('csodalatos-turam') + expect('הסיור המדהים שלי'.parameterize_intl).to eq('הסיור-המדהים-שלי') + expect('我的精 彩之旅!!'.parameterize_intl).to eq('我的精-彩之旅') + expect('Мій чудовий тур'.parameterize_intl).to eq('мій-чудовий-тур') + expect('جولتي الرائعة'.parameterize_intl).to eq('جولتي-الرائعة') + expect('我的精 彩之旅!! (mix)'.parameterize_intl).to eq('我的精-彩之旅-mix') + expect('💩 🤯 🤷🏼'.parameterize_intl).to eq('💩-🤯-🤷🏼') + end +end diff --git a/spec/support/cookies.rb b/spec/support/cookies.rb new file mode 100644 index 00000000..b611978a --- /dev/null +++ b/spec/support/cookies.rb @@ -0,0 +1,6 @@ +# spec/support/cookies.rb +class Rack::Test::CookieJar + def encrypted; self; end + def signed; self; end + def permanent; self; end # I needed this, too +end \ No newline at end of file diff --git a/spec/support/database_cleaner.rb b/spec/support/database_cleaner.rb index 4df8a28c..69f033be 100644 --- a/spec/support/database_cleaner.rb +++ b/spec/support/database_cleaner.rb @@ -7,7 +7,7 @@ DatabaseCleaner.strategy = :transaction end - config.before(:each, :js => true) do + config.before(:each, js: true) do DatabaseCleaner.strategy = :truncation end @@ -16,6 +16,10 @@ end config.after(:each) do - DatabaseCleaner.clean + begin + DatabaseCleaner.clean + rescue NoMethodError + # IDK + end end -end \ No newline at end of file +end diff --git a/spec/support/request_spec_helper.rb b/spec/support/request_spec_helper.rb index 71d8377d..5f587f24 100644 --- a/spec/support/request_spec_helper.rb +++ b/spec/support/request_spec_helper.rb @@ -4,7 +4,11 @@ module RequestSpecHelper # Parse JSON response to ruby hash def json - JSON.parse(response.body)['data'] + JSON.parse(response.body).with_indifferent_access[:data] + end + + def errors + JSON.parse(response.body).with_indifferent_access[:errors].map { |e| e[:detail] } end def response_id @@ -12,6 +16,9 @@ def response_id end def attributes + if json.is_a?(Array) + return json.map { |record| record[:attributes] } + end json['attributes'] end @@ -20,7 +27,7 @@ def relationships end def included - JSON.parse(response.body)['included'] + JSON.parse(response.body).with_indifferent_access[:included] end def hash_to_json_api(model, attributes) diff --git a/spec/support/signed_cookie.rb b/spec/support/signed_cookie.rb new file mode 100755 index 00000000..9a86c888 --- /dev/null +++ b/spec/support/signed_cookie.rb @@ -0,0 +1,19 @@ +# frozen_string_literal: true + +# +# Helper for making authenticated requests for controller specs. +# +module SignedCookieHelper + def signed_cookie(user) + login = EcdsRailsAuthEngine::Login.create!(who: user.email) + login.user_id = user.id + login.save! + create(:token, login: login, token: TokenService.create(login)) + cookies.signed[:auth] = { + value: login.tokens.first.token, + httponly: true, + same_site: :none, + secure: 'Secure' + } + end +end diff --git a/temp_gemfile b/temp_gemfile new file mode 100644 index 00000000..e69de29b