Skip to content
This repository was archived by the owner on Nov 9, 2017. It is now read-only.

Commit a19d943

Browse files
author
David Heinemeier Hansson
authored
Direct uploads for S3
1 parent b1cf901 commit a19d943

16 files changed

+138
-21
lines changed

Gemfile

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ gem 'rake'
66
gem 'byebug'
77

88
gem 'sqlite3'
9+
gem 'httparty'
910

1011
gem 'aws-sdk', require: false
1112
gem 'google-cloud-storage', require: false

Gemfile.lock

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,8 @@ GEM
8383
multi_json (~> 1.11)
8484
os (~> 0.9)
8585
signet (~> 0.7)
86+
httparty (0.15.5)
87+
multi_xml (>= 0.5.2)
8688
httpclient (2.8.3)
8789
i18n (0.8.4)
8890
jmespath (1.3.1)
@@ -100,6 +102,7 @@ GEM
100102
mini_portile2 (2.1.0)
101103
minitest (5.10.2)
102104
multi_json (1.12.1)
105+
multi_xml (0.6.0)
103106
multipart-post (2.0.0)
104107
nokogiri (1.7.2)
105108
mini_portile2 (~> 2.1.0)
@@ -139,6 +142,7 @@ DEPENDENCIES
139142
bundler (~> 1.15)
140143
byebug
141144
google-cloud-storage
145+
httparty
142146
rake
143147
sqlite3
144148

lib/active_storage/blob.rb

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,10 @@ def build_after_upload(io:, filename:, content_type: nil, metadata: nil)
2525
def create_after_upload!(io:, filename:, content_type: nil, metadata: nil)
2626
build_after_upload(io: io, filename: filename, content_type: content_type, metadata: metadata).tap(&:save!)
2727
end
28+
29+
def create_before_direct_upload!(filename:, byte_size:, checksum:, content_type: nil, metadata: nil)
30+
create! filename: filename, byte_size: byte_size, checksum: checksum, content_type: content_type, metadata: metadata
31+
end
2832
end
2933

3034
# We can't wait until the record is first saved to have a key for it
@@ -40,6 +44,10 @@ def url(expires_in: 5.minutes, disposition: :inline)
4044
service.url key, expires_in: expires_in, disposition: disposition, filename: filename
4145
end
4246

47+
def url_for_direct_upload(expires_in: 5.minutes)
48+
service.url_for_direct_upload key, expires_in: expires_in, content_type: content_type, content_length: byte_size
49+
end
50+
4351

4452
def upload(io)
4553
self.checksum = compute_checksum_in_chunks(io)
Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
require "action_controller"
2+
require "active_storage/blob"
3+
4+
class ActiveStorage::DirectUploadsController < ActionController::Base
5+
def create
6+
blob = ActiveStorage::Blob.create_before_direct_upload!(blob_args)
7+
render json: { url: blob.url_for_direct_upload, sgid: blob.to_sgid.to_param }
8+
end
9+
10+
private
11+
def blob_args
12+
params.require(:blob).permit(:filename, :byte_size, :checksum, :content_type, :metadata).to_h.symbolize_keys
13+
end
14+
end

lib/active_storage/engine.rb

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -16,10 +16,11 @@ class Engine < Rails::Engine # :nodoc:
1616

1717
initializer "active_storage.routes" do
1818
require "active_storage/disk_controller"
19+
require "active_storage/direct_uploads_controller"
1920

2021
config.after_initialize do |app|
2122
app.routes.prepend do
22-
get "/rails/blobs/:encoded_key/*filename" => "active_storage/disk#show", as: :rails_disk_blob
23+
eval(File.read(File.expand_path("../routes.rb", __FILE__)))
2324
end
2425
end
2526
end

lib/active_storage/routes.rb

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
get "/rails/active_storage/disk/:encoded_key/*filename" => "active_storage/disk#show", as: :rails_disk_blob
2+
post "/rails/active_storage/direct_uploads" => "active_storage/direct_uploads#create", as: :rails_direct_uploads

lib/active_storage/service.rb

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -78,6 +78,10 @@ def url(key, expires_in:, disposition:, filename:)
7878
raise NotImplementedError
7979
end
8080

81+
def url_for_direct_upload(key, expires_in:, content_type:, content_length:)
82+
raise NotImplementedError
83+
end
84+
8185
private
8286
def instrument(operation, key, payload = {}, &block)
8387
ActiveSupport::Notifications.instrument(

lib/active_storage/service/disk_service.rb

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,7 @@ def url(key, expires_in:, disposition:, filename:)
5959
if defined?(Rails) && defined?(Rails.application)
6060
Rails.application.routes.url_helpers.rails_disk_blob_path(verified_key_with_expiration, disposition: disposition, filename: filename)
6161
else
62-
"/rails/blobs/#{verified_key_with_expiration}/#{filename}?disposition=#{disposition}"
62+
"/rails/active_storage/disk/#{verified_key_with_expiration}/#{filename}?disposition=#{disposition}"
6363
end
6464

6565
payload[:url] = generated_url

lib/active_storage/service/s3_service.rb

Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,17 @@ def url(key, expires_in:, disposition:, filename:)
5656
end
5757
end
5858

59+
def url_for_direct_upload(key, expires_in:, content_type:, content_length:)
60+
instrument :url, key do |payload|
61+
generated_url = object_for(key).presigned_url :put, expires_in: expires_in,
62+
content_type: content_type, content_length: content_length
63+
64+
payload[:url] = generated_url
65+
66+
generated_url
67+
end
68+
end
69+
5970
private
6071
def object_for(key)
6172
bucket.object(key)

test/blob_test.rb

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,6 @@ class ActiveStorage::BlobTest < ActiveSupport::TestCase
2323

2424
private
2525
def expected_url_for(blob, disposition: :inline)
26-
"/rails/blobs/#{ActiveStorage::VerifiedKeyWithExpiration.encode(blob.key, expires_in: 5.minutes)}/#{blob.filename}?disposition=#{disposition}"
26+
"/rails/active_storage/disk/#{ActiveStorage::VerifiedKeyWithExpiration.encode(blob.key, expires_in: 5.minutes)}/#{blob.filename}?disposition=#{disposition}"
2727
end
2828
end

0 commit comments

Comments
 (0)