class UploadService module Utils module_function def corrupt?(filename) image = Vips::Image.new_from_file(filename, fail: true) image.stats false rescue Vips::Error true end def distribute_files(file, record, type, original_post_id: nil) # need to do this for hybrid storage manager post = Post.new post.id = original_post_id if original_post_id.present? post.md5 = record.md5 post.file_ext = record.file_ext [Danbooru.config.storage_manager, Danbooru.config.backup_storage_manager].each do |sm| sm.store_file(file, post, type) end end def is_downloadable?(source) source =~ /^https?:\/\// end def generate_resizes(media_file) preview_file = media_file.preview(Danbooru.config.small_image_width, Danbooru.config.small_image_width) crop_file = media_file.crop(Danbooru.config.small_image_width, Danbooru.config.small_image_width) if media_file.is_ugoira? sample_file = media_file.convert elsif media_file.is_image? && media_file.width > Danbooru.config.large_image_width sample_file = media_file.preview(Danbooru.config.large_image_width, nil) else sample_file = nil end [preview_file, crop_file, sample_file] end def process_file(upload, file, original_post_id: nil) upload.file = file media_file = upload.media_file upload.file_ext = media_file.file_ext.to_s upload.file_size = media_file.file_size upload.md5 = media_file.md5 upload.image_width = media_file.width upload.image_height = media_file.height upload.validate!(:file) upload.tag_string = "#{upload.tag_string} #{Utils.automatic_tags(upload, file)}" preview_file, crop_file, sample_file = Utils.generate_resizes(media_file) begin Utils.distribute_files(file, upload, :original, original_post_id: original_post_id) Utils.distribute_files(sample_file, upload, :large, original_post_id: original_post_id) if sample_file.present? Utils.distribute_files(preview_file, upload, :preview, original_post_id: original_post_id) if preview_file.present? Utils.distribute_files(crop_file, upload, :crop, original_post_id: original_post_id) if crop_file.present? ensure preview_file.try(:close!) crop_file.try(:close!) sample_file.try(:close!) end end # these methods are only really used during upload processing even # though logically they belong on upload. post can rely on the # automatic tag that's added. def is_animated_gif?(upload, file) return false if upload.file_ext != "gif" # Check whether the gif has multiple frames by trying to load the second frame. result = Vips::Image.gifload(file.path, page: 1) rescue $ERROR_INFO if result.is_a?(Vips::Image) true elsif result.is_a?(Vips::Error) && result.message =~ /too few frames in GIF file/ false else raise result end end def is_animated_png?(upload, file) upload.file_ext == "png" && APNGInspector.new(file.path).inspect!.animated? end def is_video_with_audio?(upload, file) return false if !upload.is_video? # avoid ffprobe'ing the file if it's not a video (issue #3826) video = FFMPEG::Movie.new(file.path) video.audio_channels.present? end def automatic_tags(upload, file) tags = [] tags << "video_with_sound" if is_video_with_audio?(upload, file) tags << "animated_gif" if is_animated_gif?(upload, file) tags << "animated_png" if is_animated_png?(upload, file) tags.join(" ") end def get_file_for_upload(upload, file: nil) return file if file.present? raise "No file or source URL provided" if upload.source_url.blank? download = Downloads::File.new(upload.source_url, upload.referer_url) file, strategy = download.download! if download.data[:ugoira_frame_data].present? upload.context = { "ugoira" => { "frame_data" => download.data[:ugoira_frame_data], "content_type" => "image/jpeg" } } end return file end end end