-
Notifications
You must be signed in to change notification settings - Fork 71
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
画像のバリデーションをMimeTypeを用いるものに変更 #8092
base: main
Are you sure you want to change the base?
Conversation
0291ea2
to
770c936
Compare
@MikotoMakizuru |
app/models/user.rb
Outdated
in: %w[image/png image/jpg image/jpeg image/gif image/heic image/heif], | ||
message: 'はPNG, JPG, GIF, HEIC, HEIF形式にしてください' | ||
} | ||
attr_accessor :uploaded_avatar |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
avatarではなく、uploaded_avatarにvalidationをかけているのは理由があります。
理由
uploaded_avatarはアップロードされたファイル自体を指す。(#ActionDispatch::Http::UploadedFile:hogehoge)これはIO互換の操作が可能。
ただavatarに一度代入されてしまうと、ActiveStorageにアタッチされたデータになる(#ActiveStorage::Attached::One:hogehoge)
後者だと、ファイルの実体に対するcheckができなくなってしまう。
そのため、uploaded_avatarとして切り出しアップロードされた画像自体にバリデーションをかけています。
@reckyy 今から見ます〜 |
app/models/user.rb
Outdated
mime_type = Marcel::Magic.by_magic(uploaded_avatar)&.type | ||
return if mime_type&.start_with?('image/png', 'image/jpg', 'image/jpeg', 'image/gif', 'image/heic', 'image/heif') | ||
|
||
errors.add(:avatar, 'は指定された拡張子(PNG, JPG, GIF, HEIC, HEIF形式)になっていないか、あるいは画像が破損している可能性があります') |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
分けられるのであれば、拡張子が指定以外のものか、画像が破損しているか、別々のバリデーションメッセージのほうがユーザにとってわかりやすいかなと思いました。
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@MikotoMakizuru
ご提案ありがとうございます。
回答としては、分けるのは難しいため今の仕様になっています。
なぜかというと、
mime_type
がnilの場合でも必ず破損しているわけではない(テキストファイルなど)- 何かアップロードに問題が発生した(画像が部分的にしかアップロードされなかったとか?)
のように、画像が破損しているか拡張子が規定から外れているかの判別は難しいためです。
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
何かアップロードに問題が発生した(画像が部分的にしかアップロードされなかったとか?)
どのようなときのこのこと言っているのか分からないですが、画像処理の gem を使うといいかもしれません。
- gem 追加
Gemefile
gem 'rmagick'
- インストール
$ bundle
- バリデーション処理実装
user.rb
# frozen_string_literal: true
require 'rmagick'
class User < ApplicationRecord
(省略)
def validate_uploaded_avatar_content_type
return unless uploaded_avatar
allowed_extensions = %w[png jpg jpeg gif heic heif]
file_extension = File.extname(uploaded_avatar.original_filename).delete('.').downcase
unless allowed_extensions.include?(file_extension)
errors.add(:avatar, 'はPNG, JPG, JPEG, GIF, HEIC, HEIF形式にしてください')
return
end
begin
Magick::Image.read(uploaded_avatar.tempfile.path)
rescue Magick::ImageMagickError
errors.add(:avatar, 'が破損している可能性があります')
end
end
end
動作確認
2024-12-05.2.11.25.mov
使った gem:https://github.com/rmagick/rmagick
あと、
- jpeg は指定 OK なのにバリデーションメッセージに「JPEG」が含まれていないのが気になりました。
- rubocop というコミットコメントは指摘されるような気がします。
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@MikotoMakizuru
今外出中なのですが、取り急ぎ返信だけさせていただきます〜。
どのようなときのこのこと言っているのか分からないですが
思いつかなかったので曖昧な例を挙げてしまいました、すみません。
こちらについてはご放念ください。
画像処理の gem を使うといいかもしれません。
ご提案、そして動画まで添付頂きありがとうございます!
ご提案頂いたgemは、数ヶ月前にImageMagickからlibvipsに移行した都合上、非推奨だと思われます。
ruby-vipsを使えば出来るかもですが、後述のように現在は不要だと考えています。
1点ご相談なのですが、自分としてはエラーメッセージを細分化する処理を実装するのは少し過剰かなと考えています。
画像が壊れているかどうかの判別は完璧にはできないと思っているからです。
理由はkomagataさんのコメントに同じです。
そのため、このエラーメッセージの細分化については現時点では一旦保留とし、最終的な判断はkomagataさんのレビュー時に仰ぐ形で進めたいと思っていますが、いかがでしょうか。
jpeg は指定 OK なのにバリデーションメッセージに「JPEG」が含まれていないのが気になりました。
含めておきます。
rubocop というコミットコメントは指摘されるような気がします。
「Rubocopの指摘を修正」に変更しておきます。
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
確かにバリデーション1つで gem を入れるのはやりすぎな気もしますね、仕様を komagata さんに相談する形でレビュー進めていただいて問題ありません。
また、指定された拡張子以外のファイル選択を出来ないようにすることで、model で考慮すべきことも減らせると思うので、その辺の仕様も相談してみると良いと思います!
@reckyy 1点コメントさせていただきました〜 |
770c936
to
d77d438
Compare
66605d2
to
51906f4
Compare
@@ -8,6 +8,7 @@ def edit | |||
end | |||
|
|||
def update | |||
@user.uploaded_avatar = user_params[:avatar] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
これって必須でしょうか?
通常のアップロードのフローでValidation処理を入れられないでしょうか。
Issue
概要
riraさんが作成したPRの概要から引用させていただきます。
変更確認方法
fix/broken-images-can-be-registered
をローカルに取り込むScreenshot
変更前
変更後