# frozen_string_literal: true require 'fog/core/model' module Fog module Storage class Aliyun class File < Fog::Model identity :key, aliases: 'name' attribute :date, aliases: 'Date' attribute :content_length, aliases: 'Content-Length', type: :integer attribute :content_type, aliases: 'Content-Type' attribute :connection, aliases: 'Connection' attribute :content_disposition, aliases: 'Content-Disposition' attribute :etag, aliases: 'Etag' attribute :last_modified, aliases: 'Last-Modified', type: :time attribute :accept_ranges, aliases: 'Accept-Ranges' attribute :server, aliases: 'Server' attribute :object_type, aliases: 'x-oss-object-type' def body attributes[:body] ||= if last_modified collection.get(identity).body else '' end end def body=(new_body) attributes[:body] = new_body end attr_reader :directory def copy(target_directory_key, target_file_key, options = {}) requires :directory, :key source_object = if directory.key == '' key else directory.key + '/' + key end target_object = if target_directory_key == '' target_file_key else target_directory_key + '/' + target_file_key end service.copy_object(nil, source_object, nil, target_object, options) target_directory = service.directories.new(key: target_directory_key) target_directory.files.get(target_file_key) end def destroy requires :directory, :key object = if directory.key == '' key else directory.key + '/' + key end service.delete_object(object) true end def metadata attributes[:metadata] ||= headers_to_metadata end def owner=(new_owner) if new_owner attributes[:owner] = { display_name: new_owner['DisplayName'], id: new_owner['ID'] } end end def public=(new_public) new_public end # Get a url for file. # # required attributes: directory, key # # @param expires [String] number of seconds (since 1970-01-01 00:00) before url expires # @param options [Hash] # @return [String] url # def url(expires, options = {}) requires :directory, :key object = if directory.key == '' key else directory.key + '/' + key end service.get_object_http_url_public(object, expires, options) end def public_url requires :key collection.get_url(key) end def save(options = {}) requires :body, :directory, :key options['Content-Type'] = content_type if content_type options['Content-Disposition'] = content_disposition if content_disposition options.merge!(metadata_to_headers) object = if directory.key == '' key else directory.key + '/' + key end if body.is_a?(::File) data = service.put_object(object, body, options).data elsif body.is_a?(String) data = service.put_object_with_body(object, body, options).data else raise Fog::Storage::Aliyun::Error, " Forbidden: Invalid body type: #{body.class}!" end update_attributes_from(data) refresh_metadata self.content_length = Fog::Storage.get_body_size(body) self.content_type ||= Fog::Storage.get_content_type(body) true end private attr_writer :directory def refresh_metadata metadata.reject! { |_k, v| v.nil? } end def headers_to_metadata key_map = key_mapping Hash[metadata_attributes.map { |k, v| [key_map[k], v] }] end def key_mapping key_map = metadata_attributes key_map.each_pair { |k, _v| key_map[k] = header_to_key(k) } end def header_to_key(opt) opt.gsub(metadata_prefix, '').split('-').map { |k| k[0, 1].downcase + k[1..-1] }.join('_').to_sym end def metadata_to_headers header_map = header_mapping Hash[metadata.map { |k, v| [header_map[k], v] }] end def header_mapping header_map = metadata.dup header_map.each_pair { |k, _v| header_map[k] = key_to_header(k) } end def key_to_header(key) metadata_prefix + key.to_s.split(/[-_]/).map(&:capitalize).join('-') end def metadata_attributes if last_modified object = if directory.key == '' key else directory.key + '/' + key end headers = service.head_object(object).data[:headers] headers.select! { |k, _v| metadata_attribute?(k) } else {} end end def metadata_attribute?(key) key.to_s =~ /^#{metadata_prefix}/ end def metadata_prefix 'X-Object-Meta-' end def update_attributes_from(data) merge_attributes(data[:headers].reject { |key, _value| ['Content-Length', 'Content-Type'].include?(key) }) end end end end end