From aa7e179a983f9ea7b7e5ed3641cc3fa288b4ad49 Mon Sep 17 00:00:00 2001 From: pabois <pierreandre.boissinot@noesya.coop> Date: Fri, 8 Oct 2021 16:59:29 +0200 Subject: [PATCH] active storazge lock --- .../blobs/redirect_controller.rb | 19 +++++++ .../representations/redirect_controller.rb | 18 +++++++ app/controllers/concerns/.keep | 0 .../active_storage/check_abilities.rb | 10 ++++ app/models/research/journal/article.rb | 2 +- config/initializers/active_storage.rb | 34 +++++++++++++ ...dd_university_id_to_active_storage_blob.rb | 6 +++ db/schema.rb | 6 ++- public/403.html | 50 +++++++++++++++++++ test/fixtures/research/journal/articles.yml | 2 +- test/models/research/journal/article_test.rb | 2 +- 11 files changed, 144 insertions(+), 5 deletions(-) create mode 100644 app/controllers/active_storage/blobs/redirect_controller.rb create mode 100644 app/controllers/active_storage/representations/redirect_controller.rb delete mode 100644 app/controllers/concerns/.keep create mode 100644 app/controllers/concerns/active_storage/check_abilities.rb create mode 100644 config/initializers/active_storage.rb create mode 100644 db/migrate/20211008124136_add_university_id_to_active_storage_blob.rb create mode 100644 public/403.html diff --git a/app/controllers/active_storage/blobs/redirect_controller.rb b/app/controllers/active_storage/blobs/redirect_controller.rb new file mode 100644 index 000000000..906eade46 --- /dev/null +++ b/app/controllers/active_storage/blobs/redirect_controller.rb @@ -0,0 +1,19 @@ +# frozen_string_literal: true + +# Take a signed permanent reference for a blob and turn it into an expiring service URL for download. +# Note: These URLs are publicly accessible. If you need to enforce access protection beyond the +# security-through-obscurity factor of the signed blob references, you'll need to implement your own +# authenticated redirection controller. +class ActiveStorage::Blobs::RedirectController < ActiveStorage::BaseController + include ActiveStorage::CheckAbilities + include ActiveStorage::SetBlob + + before_action :check_abilities, only: :show + skip_before_action :handle_two_factor_authentication + + def show + expires_in ActiveStorage.service_urls_expire_in + redirect_to @blob.url(disposition: params[:disposition]) + end + +end diff --git a/app/controllers/active_storage/representations/redirect_controller.rb b/app/controllers/active_storage/representations/redirect_controller.rb new file mode 100644 index 000000000..a4c71baaf --- /dev/null +++ b/app/controllers/active_storage/representations/redirect_controller.rb @@ -0,0 +1,18 @@ +# frozen_string_literal: true + +# Take a signed permanent reference for a blob representation and turn it into an expiring service URL for download. +# Note: These URLs are publicly accessible. If you need to enforce access protection beyond the +# security-through-obscurity factor of the signed blob and variation reference, you'll need to implement your own +# authenticated redirection controller. +class ActiveStorage::Representations::RedirectController < ActiveStorage::Representations::BaseController + include ActiveStorage::CheckAbilities + + before_action :check_abilities, only: :show + skip_before_action :handle_two_factor_authentication + + def show + expires_in ActiveStorage.service_urls_expire_in + redirect_to @representation.url(disposition: params[:disposition]) + end + +end diff --git a/app/controllers/concerns/.keep b/app/controllers/concerns/.keep deleted file mode 100644 index e69de29bb..000000000 diff --git a/app/controllers/concerns/active_storage/check_abilities.rb b/app/controllers/concerns/active_storage/check_abilities.rb new file mode 100644 index 000000000..ea59d518d --- /dev/null +++ b/app/controllers/concerns/active_storage/check_abilities.rb @@ -0,0 +1,10 @@ +module ActiveStorage::CheckAbilities + extend ActiveSupport::Concern + + private + + def check_abilities + university = University.with_host(request.host) + render(file: Rails.root.join('public/403.html'), formats: [:html], status: 403, layout: false) and return if university.present? && @blob.university_id != university.id + end +end diff --git a/app/models/research/journal/article.rb b/app/models/research/journal/article.rb index 20a415c65..0410c664d 100644 --- a/app/models/research/journal/article.rb +++ b/app/models/research/journal/article.rb @@ -10,7 +10,7 @@ # text :text # title :string # created_at :datetime not null -# updated_at :date not null +# updated_at :datetime not null # research_journal_id :uuid not null # research_journal_volume_id :uuid # university_id :uuid not null diff --git a/config/initializers/active_storage.rb b/config/initializers/active_storage.rb new file mode 100644 index 000000000..efe52fbbf --- /dev/null +++ b/config/initializers/active_storage.rb @@ -0,0 +1,34 @@ +require 'active_storage/attachment' +require 'active_storage/filename' + +ActiveStorage::Engine.config.active_storage.content_types_to_serve_as_binary.delete('image/svg+xml') + +# Hook ActiveStorage::Attachment to add brand_id to attachments records +ActiveStorage::Attachment.class_eval do + after_save :denormalize_university_id_for_blob + + def denormalize_university_id_for_blob + university_id = case self.record.class.name + when 'University' + self.record.id + when 'ActiveStorage::VariantRecord' + self.record.blob.university_id + else + self.record.university_id + end + + self.blob.update_column(:university_id, university_id) + end +end + +# Override ActiveStorage::Filename#sanitized to remove accents and all special chars +# Base method: https://github.com/rails/rails/blob/v6.1.3/activestorage/app/models/active_storage/filename.rb#L57 +ActiveStorage::Filename.class_eval do + def sanitized + base_filename = base.encode(Encoding::UTF_8, invalid: :replace, undef: :replace, replace: "�") + .strip + .tr("\u{202E}%$|:;/\t\r\n\\", "-") + .parameterize(preserve_case: true) + [base_filename, extension_with_delimiter].join('') + end +end diff --git a/db/migrate/20211008124136_add_university_id_to_active_storage_blob.rb b/db/migrate/20211008124136_add_university_id_to_active_storage_blob.rb new file mode 100644 index 000000000..1dcdf0425 --- /dev/null +++ b/db/migrate/20211008124136_add_university_id_to_active_storage_blob.rb @@ -0,0 +1,6 @@ +class AddUniversityIdToActiveStorageBlob < ActiveRecord::Migration[6.1] + def change + add_reference :active_storage_blobs, :university, type: :uuid, index: true + + end +end diff --git a/db/schema.rb b/db/schema.rb index ef44af74f..a31dc6a06 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -10,7 +10,7 @@ # # It's strongly recommended that you check this file into your version control system. -ActiveRecord::Schema.define(version: 2021_10_08_090117) do +ActiveRecord::Schema.define(version: 2021_10_08_124136) do # These are extensions that must be enabled in order to support this database enable_extension "pgcrypto" @@ -35,7 +35,9 @@ ActiveRecord::Schema.define(version: 2021_10_08_090117) do t.bigint "byte_size", null: false t.string "checksum", null: false t.datetime "created_at", null: false + t.uuid "university_id" t.index ["key"], name: "index_active_storage_blobs_on_key", unique: true + t.index ["university_id"], name: "index_active_storage_blobs_on_university_id" end create_table "active_storage_variant_records", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t| @@ -178,7 +180,7 @@ ActiveRecord::Schema.define(version: 2021_10_08_090117) do t.uuid "research_journal_id", null: false t.uuid "research_journal_volume_id" t.datetime "created_at", precision: 6, null: false - t.date "updated_at", null: false + t.datetime "updated_at", precision: 6, null: false t.uuid "updated_by_id" t.text "abstract" t.text "references" diff --git a/public/403.html b/public/403.html new file mode 100644 index 000000000..edacef6a0 --- /dev/null +++ b/public/403.html @@ -0,0 +1,50 @@ +<!DOCTYPE html> +<html lang="fr"> + <head> + <title>You do not have access to this page (403)</title> + <meta charset="utf-8"> + <meta name="viewport" content="width=device-width,initial-scale=1"> + <style> + + .error-page { + font-family: arial, sans-serif; + margin: 0; + } + + .error-page div.dialog { + width: 95%; + max-width: 22em; + margin: calc(50vh - 140px) auto 0; + } + + .error-page div.dialog span{ + font-size: 200px; + font-weight: 900; + letter-spacing: -0.04em; + display: block; + width: 100%; + line-height: 0.9; + } + + .error-page h1 { + font-size: 16px; + line-height: 1.5em; + } + + .error-page h1 em { + font-weight: 400; + } + + </style> + </head> +<body class="error-page"> + <!-- This file lives in public/500.html --> + <div class="dialog"> + <span> + 403. + </span> + <div> + <h1>You do not have access to this page.</h1> + </div> + </body> +</html> diff --git a/test/fixtures/research/journal/articles.yml b/test/fixtures/research/journal/articles.yml index 432f9dfdf..9db4f619d 100644 --- a/test/fixtures/research/journal/articles.yml +++ b/test/fixtures/research/journal/articles.yml @@ -10,7 +10,7 @@ # text :text # title :string # created_at :datetime not null -# updated_at :date not null +# updated_at :datetime not null # research_journal_id :uuid not null # research_journal_volume_id :uuid # university_id :uuid not null diff --git a/test/models/research/journal/article_test.rb b/test/models/research/journal/article_test.rb index 12249debb..d1013ba1a 100644 --- a/test/models/research/journal/article_test.rb +++ b/test/models/research/journal/article_test.rb @@ -10,7 +10,7 @@ # text :text # title :string # created_at :datetime not null -# updated_at :date not null +# updated_at :datetime not null # research_journal_id :uuid not null # research_journal_volume_id :uuid # university_id :uuid not null -- GitLab