service.rb 5.9 KB
Newer Older
1 2
# frozen_string_literal: true

3
require "active_storage/log_subscriber"
4
require "active_storage/downloader"
5 6
require "action_dispatch"
require "action_dispatch/http/content_disposition"
7

8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
module ActiveStorage
  # Abstract class serving as an interface for concrete services.
  #
  # The available services are:
  #
  # * +Disk+, to manage attachments saved directly on the hard drive.
  # * +GCS+, to manage attachments through Google Cloud Storage.
  # * +S3+, to manage attachments through Amazon S3.
  # * +AzureStorage+, to manage attachments through Microsoft Azure Storage.
  # * +Mirror+, to be able to use several services to manage attachments.
  #
  # Inside a Rails application, you can set-up your services through the
  # generated <tt>config/storage.yml</tt> file and reference one
  # of the aforementioned constant under the +service+ key. For example:
  #
  #   local:
  #     service: Disk
  #     root: <%= Rails.root.join("storage") %>
  #
  # You can checkout the service's constructor to know which keys are required.
  #
  # Then, in your application's configuration, you can specify the service to
  # use like this:
  #
  #   config.active_storage.service = :local
  #
  # If you are using Active Storage outside of a Ruby on Rails application, you
  # can configure the service to use like this:
  #
  #   ActiveStorage::Blob.service = ActiveStorage::Service.configure(
  #     :Disk,
  #     root: Pathname("/foo/bar/storage")
  #   )
  class Service
    extend ActiveSupport::Autoload
    autoload :Configurator
44
    attr_accessor :name
45

46 47 48 49
    class << self
      # Configure an Active Storage service by name from a set of configurations,
      # typically loaded from a YAML file. The Active Storage engine uses this
      # to set the global Active Storage service when the app boots.
50
      def configure(service_name, configurations)
51 52
        Configurator.build(service_name, configurations)
      end
53

54 55 56 57 58 59
      # Override in subclasses that stitch together multiple services and hence
      # need to build additional services using the configurator.
      #
      # Passes the configurator and all of the service's config as keyword args.
      #
      # See MirrorService for an example.
60 61 62 63
      def build(configurator:, name:, service: nil, **service_config) #:nodoc:
        new(**service_config).tap do |service_instance|
          service_instance.name = name
        end
64
      end
65
    end
66

67 68
    # Upload the +io+ to the +key+ specified. If a +checksum+ is provided, the service will
    # ensure a match when the upload has completed or raise an ActiveStorage::IntegrityError.
69
    def upload(key, io, checksum: nil, **options)
70
      raise NotImplementedError
71
    end
D
David Heinemeier Hansson 已提交
72

73 74 75 76 77 78
    # Update metadata for the file identified by +key+ in the service.
    # Override in subclasses only if the service needs to store specific
    # metadata that has to be updated upon identification.
    def update_metadata(key, **metadata)
    end

79
    # Return the content of the file at the +key+.
80 81 82
    def download(key)
      raise NotImplementedError
    end
D
David Heinemeier Hansson 已提交
83

84
    # Return the partial content in the byte +range+ of the file at the +key+.
85 86 87 88
    def download_chunk(key, range)
      raise NotImplementedError
    end

89 90
    def open(*args, **options, &block)
      ActiveStorage::Downloader.new(self).open(*args, **options, &block)
91 92
    end

93
    # Delete the file at the +key+.
94 95 96
    def delete(key)
      raise NotImplementedError
    end
97

G
George Claghorn 已提交
98 99 100 101 102
    # Delete files at keys starting with the +prefix+.
    def delete_prefixed(prefix)
      raise NotImplementedError
    end

103
    # Return +true+ if a file exists at the +key+.
104 105 106
    def exist?(key)
      raise NotImplementedError
    end
D
David Heinemeier Hansson 已提交
107

108
    # Returns the URL for the file at the +key+. This returns a permanent URL for public files, and returns a
109 110 111
    # short-lived URL for private files. For private files you can provide the +disposition+ (+:inline+ or +:attachment+),
    # +filename+, and +content_type+ that you wish the file to be served with on request. Additionally, you can also provide
    # the amount of seconds the URL will be valid for, specified in +expires_in+.
112 113 114 115 116 117 118 119 120 121 122 123 124
    def url(key, **options)
      instrument :url, key: key do |payload|
        generated_url =
          if public?
            public_url(key, **options)
          else
            private_url(key, **options)
          end

        payload[:url] = generated_url

        generated_url
      end
125
    end
126

127 128
    # Returns a signed, temporary URL that a direct upload file can be PUT to on the +key+.
    # The URL will be valid for the amount of seconds specified in +expires_in+.
J
Jeffrey Guenther 已提交
129
    # You must also provide the +content_type+, +content_length+, and +checksum+ of the file
130 131 132
    # that will be uploaded. All these attributes will be validated by the service upon upload.
    def url_for_direct_upload(key, expires_in:, content_type:, content_length:, checksum:)
      raise NotImplementedError
133 134
    end

135
    # Returns a Hash of headers for +url_for_direct_upload+ requests.
136 137
    def headers_for_direct_upload(key, filename:, content_type:, content_length:, checksum:)
      {}
138
    end
139

140
    def public?
141
      @public
142 143
    end

144
    private
145 146 147 148 149 150 151 152 153
      def private_url(key, expires_in:, filename:, disposition:, content_type:, **)
        raise NotImplementedError
      end

      def public_url(key, **)
        raise NotImplementedError
      end


G
George Claghorn 已提交
154
      def instrument(operation, payload = {}, &block)
155 156
        ActiveSupport::Notifications.instrument(
          "service_#{operation}.active_storage",
G
George Claghorn 已提交
157
          payload.merge(service: service_name), &block)
158 159 160 161 162 163
      end

      def service_name
        # ActiveStorage::Service::DiskService => Disk
        self.class.name.split("::").third.remove("Service")
      end
G
George Claghorn 已提交
164 165

      def content_disposition_with(type: "inline", filename:)
166 167
        disposition = (type.to_s.presence_in(%w( attachment inline )) || "inline")
        ActionDispatch::Http::ContentDisposition.format(disposition: disposition, filename: filename.sanitized)
G
George Claghorn 已提交
168
      end
169
  end
D
David Heinemeier Hansson 已提交
170
end