Skip to content

Commit 89c8d11

Browse files
author
Jon Yurek
committed
Adds IO adapters to abstract the things that can be assigned.
Needs work for S3 Attachments.
1 parent 6388652 commit 89c8d11

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

42 files changed

+723
-511
lines changed

.DS_Store

12 KB
Binary file not shown.

Gemfile.lock

+19-19
Original file line numberDiff line numberDiff line change
@@ -19,13 +19,10 @@ GEM
1919
activesupport (= 3.2.2)
2020
arel (~> 3.0.2)
2121
tzinfo (~> 0.3.29)
22-
activerecord-jdbc-adapter (1.2.2)
23-
activerecord-jdbcsqlite3-adapter (1.2.2)
24-
activerecord-jdbc-adapter (~> 1.2.2)
25-
jdbc-sqlite3 (~> 3.7.2)
2622
activesupport (3.2.2)
2723
i18n (~> 0.6)
2824
multi_json (~> 1.0)
25+
addressable (2.2.7)
2926
appraisal (0.4.1)
3027
bundler
3128
rake
@@ -40,7 +37,8 @@ GEM
4037
json (~> 1.4)
4138
nokogiri (<= 1.5.0)
4239
uuidtools (~> 2.1)
43-
bouncy-castle-java (1.5.0146.1)
40+
bourne (1.1.2)
41+
mocha (= 0.10.5)
4442
builder (3.0.0)
4543
capybara (1.1.2)
4644
mime-types (>= 1.16)
@@ -52,6 +50,7 @@ GEM
5250
childprocess (0.3.1)
5351
ffi (~> 1.0.6)
5452
cocaine (0.2.1)
53+
coderay (1.0.5)
5554
cucumber (1.1.9)
5655
builder (>= 2.1.2)
5756
diff-lcs (>= 1.1.2)
@@ -62,7 +61,6 @@ GEM
6261
excon (0.6.6)
6362
fakeweb (1.3.0)
6463
ffi (1.0.11)
65-
ffi (1.0.11-java)
6664
fog (0.9.0)
6765
builder
6866
excon (~> 0.6.1)
@@ -74,31 +72,30 @@ GEM
7472
nokogiri (>= 1.4.4)
7573
ruby-hmac
7674
formatador (0.2.1)
77-
gherkin (2.9.1)
78-
json (>= 1.4.6)
79-
gherkin (2.9.1-java)
75+
gherkin (2.9.3)
8076
json (>= 1.4.6)
8177
httparty (0.8.1)
8278
multi_json
8379
multi_xml
8480
i18n (0.6.0)
85-
jdbc-sqlite3 (3.7.2)
86-
jruby-openssl (0.7.6.1)
87-
bouncy-castle-java (>= 1.5.0146.1)
88-
json (1.6.5)
89-
json (1.6.5-java)
81+
json (1.6.6)
82+
launchy (2.1.0)
83+
addressable (~> 2.2.6)
9084
metaclass (0.0.1)
85+
method_source (0.7.1)
9186
mime-types (1.18)
9287
mocha (0.10.5)
9388
metaclass (~> 0.0.1)
94-
multi_json (1.1.0)
89+
multi_json (1.2.0)
9590
multi_xml (0.4.2)
9691
net-scp (1.0.4)
9792
net-ssh (>= 1.99.1)
9893
net-ssh (2.3.0)
9994
nokogiri (1.4.7)
100-
nokogiri (1.4.7-java)
101-
weakling (>= 0.0.3)
95+
pry (0.9.8.4)
96+
coderay (~> 1.0.5)
97+
method_source (~> 0.7.1)
98+
slop (>= 2.4.4, < 3)
10299
rack (1.4.1)
103100
rack-test (0.6.1)
104101
rack (>= 1.0)
@@ -123,32 +120,35 @@ GEM
123120
shoulda-matchers (~> 1.0.0)
124121
shoulda-context (1.0.0)
125122
shoulda-matchers (1.0.0)
123+
slop (2.4.4)
126124
sqlite3 (1.3.5)
127125
term-ansicolor (1.0.7)
128126
tzinfo (0.3.32)
129127
uuidtools (2.1.2)
130-
weakling (0.0.4-java)
131128
xpath (0.1.4)
132129
nokogiri (~> 1.3)
133130

134131
PLATFORMS
135-
java
136132
ruby
137133

138134
DEPENDENCIES
139135
activerecord-jdbcsqlite3-adapter
140136
appraisal (~> 0.4.0)
141137
aruba
142138
aws-sdk (~> 1.3.8)
139+
bourne
143140
bundler
144141
capybara
145142
cocaine (~> 0.2)
146143
cucumber (~> 1.1.0)
147144
fakeweb
148145
fog
149146
jruby-openssl
147+
launchy
150148
mocha
151149
nokogiri (~> 1.4.7)
152150
paperclip!
151+
pry
152+
rake
153153
shoulda
154154
sqlite3

features/step_definitions/rails_steps.rb

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
When /^I print "([^\"]*)"$/ do |whatever|
2+
puts whatever
3+
end
4+
15
Given /^I generate a new rails application$/ do
26
steps %{
37
When I run `bundle exec #{new_application_command} #{APP_NAME}`

features/support/env.rb

+4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,10 @@
11
require 'aruba/cucumber'
22
require 'capybara/cucumber'
33
require 'test/unit/assertions'
4+
require 'pry'
5+
6+
$CUCUMBER=1
7+
48
World(Test::Unit::Assertions)
59

610
Before do

images.rake

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
namespace :images do
2+
desc "Regenerate images"
3+
task :regenerate => :environment do
4+
require 'open-uri'
5+
OpportunityPhoto.all.each do |photo|
6+
begin
7+
old_name = photo.image_file_name
8+
new_image = open(photo.image.url(:original, escape: false))
9+
class << new_image
10+
def original_filename; @original_filename; end
11+
def original_filename=(name); @original_filename = name; end
12+
end
13+
new_image.original_filename = old_name
14+
photo.image = new_image
15+
photo.save
16+
rescue => e
17+
puts "ERROR: #{e.message} while processing #{photo.id}"
18+
end
19+
end
20+
end
21+
end

lib/.DS_Store

6 KB
Binary file not shown.

lib/paperclip.rb

+10-2
Original file line numberDiff line numberDiff line change
@@ -29,8 +29,6 @@
2929
require 'digest'
3030
require 'tempfile'
3131
require 'paperclip/version'
32-
require 'paperclip/upfile'
33-
require 'paperclip/iostream'
3432
require 'paperclip/geometry'
3533
require 'paperclip/processor'
3634
require 'paperclip/tempfile'
@@ -49,6 +47,7 @@
4947
require 'paperclip/logger'
5048
require 'paperclip/helpers'
5149
require 'paperclip/railtie'
50+
require 'mime/types'
5251
require 'logger'
5352
require 'cocaine'
5453

@@ -203,3 +202,12 @@ def attachment_definitions
203202
end
204203
end
205204
end
205+
206+
# This stuff needs to be run after Paperclip is defined.
207+
require 'paperclip/io_adapters/registry'
208+
require 'paperclip/io_adapters/identity_adapter'
209+
require 'paperclip/io_adapters/file_adapter'
210+
require 'paperclip/io_adapters/stringio_adapter'
211+
require 'paperclip/io_adapters/nil_adapter'
212+
require 'paperclip/io_adapters/attachment_adapter'
213+
require 'paperclip/io_adapters/uploaded_file_adapter'

lib/paperclip/.DS_Store

6 KB
Binary file not shown.

lib/paperclip/attachment.rb

+29-55
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,6 @@ module Paperclip
77
# when the model saves, deletes when the model is destroyed, and processes
88
# the file upon assignment.
99
class Attachment
10-
include IOStream
11-
1210
def self.default_options
1311
@default_options ||= {
1412
:convert_options => {},
@@ -90,40 +88,25 @@ def initialize(name, instance, options = {})
9088
# new_user.avatar = old_user.avatar
9189
def assign uploaded_file
9290
ensure_required_accessors!
91+
file = Paperclip.io_adapters.for(uploaded_file)
9392

94-
if uploaded_file.is_a?(Paperclip::Attachment)
95-
uploaded_filename = uploaded_file.original_filename
96-
uploaded_file = uploaded_file.to_file(:original)
97-
close_uploaded_file = uploaded_file.respond_to?(:close)
98-
else
99-
instance_write(:uploaded_file, uploaded_file) if uploaded_file
100-
end
101-
102-
return nil unless valid_assignment?(uploaded_file)
103-
104-
uploaded_file.binmode if uploaded_file.respond_to? :binmode
10593
self.clear
94+
return nil if file.nil?
10695

107-
return nil if uploaded_file.nil?
108-
109-
uploaded_filename ||= uploaded_file.original_filename
110-
stores_fingerprint = @instance.respond_to?("#{name}_fingerprint".to_sym)
111-
@queued_for_write[:original] = to_tempfile(uploaded_file)
112-
instance_write(:file_name, cleanup_filename(uploaded_filename.strip))
113-
instance_write(:content_type, uploaded_file.content_type.to_s.strip)
114-
instance_write(:file_size, uploaded_file.size.to_i)
115-
instance_write(:fingerprint, generate_fingerprint(uploaded_file)) if stores_fingerprint
96+
@queued_for_write[:original] = file
97+
instance_write(:file_name, cleanup_filename(file.original_filename))
98+
instance_write(:content_type, file.content_type)
99+
instance_write(:file_size, file.size)
100+
instance_write(:fingerprint, file.fingerprint) if instance_respond_to?(:fingerprint)
116101
instance_write(:updated_at, Time.now)
117102

118103
@dirty = true
119104

120105
post_process(*@options[:only_process]) if post_processing
121106

122107
# Reset the file size if the original file was reprocessed.
123-
instance_write(:file_size, @queued_for_write[:original].size.to_i)
124-
instance_write(:fingerprint, generate_fingerprint(@queued_for_write[:original])) if stores_fingerprint
125-
ensure
126-
uploaded_file.close if close_uploaded_file
108+
instance_write(:file_size, @queued_for_write[:original].size)
109+
instance_write(:fingerprint, @queued_for_write[:original].fingerprint) if instance_respond_to?(:fingerprint)
127110
end
128111

129112
# Returns the public URL of the attachment with a given style. This does
@@ -252,16 +235,10 @@ def size
252235
instance_read(:file_size) || (@queued_for_write[:original] && @queued_for_write[:original].size)
253236
end
254237

255-
# Returns the hash of the file as originally assigned, and lives in the
256-
# <attachment>_fingerprint attribute of the model.
238+
# Returns the fingerprint of the file, if one's defined. The fingerprint is
239+
# stored in the <attachment>_fingerpring attribute of the model.
257240
def fingerprint
258-
if instance_read(:fingerprint)
259-
instance_read(:fingerprint)
260-
elsif @instance.respond_to?("#{name}_fingerprint".to_sym)
261-
@queued_for_write[:original] && generate_fingerprint(@queued_for_write[:original])
262-
else
263-
nil
264-
end
241+
instance_read(:fingerprint)
265242
end
266243

267244
# Returns the content_type of the file as originally assigned, and lives
@@ -307,26 +284,16 @@ def generate_fingerprint(source)
307284
# thumbnails forcefully, by reobtaining the original file and going through
308285
# the post-process again.
309286
def reprocess!(*style_args)
310-
new_original = Tempfile.new("paperclip-reprocess")
311-
new_original.binmode
312-
if old_original = to_file(:original)
313-
new_original.write( old_original.respond_to?(:get) ? old_original.get : old_original.read )
314-
new_original.rewind
315-
316-
@queued_for_write = { :original => new_original }
317-
instance_write(:updated_at, Time.now)
318-
post_process(*style_args)
319-
320-
old_original.close if old_original.respond_to?(:close)
321-
old_original.unlink if old_original.respond_to?(:unlink)
322-
287+
saved_only_process, @options[:only_process] = @options[:only_process], style_args
288+
begin
289+
assign(self)
323290
save
324-
else
325-
true
291+
rescue Errno::EACCES => e
292+
warn "#{e} - skipping file."
293+
false
294+
ensure
295+
@options[:only_process] = saved_only_process
326296
end
327-
rescue Errno::EACCES => e
328-
warn "#{e} - skipping file"
329-
false
330297
end
331298

332299
# Returns true if a file has been assigned.
@@ -336,6 +303,12 @@ def file?
336303

337304
alias :present? :file?
338305

306+
# Determines whether the instance responds to this attribute. Used to prevent
307+
# calculations on fields we won't even store.
308+
def instance_respond_to?(attr)
309+
instance.respond_to?(:"#{name}_#{attr}")
310+
end
311+
339312
# Writes the attachment-specific attribute on the instance. For example,
340313
# instance_write(:file_name, "me.jpg") will write "me.jpg" to the instance's
341314
# "avatar_file_name" field (assuming the attachment is called avatar).
@@ -428,6 +401,7 @@ def post_process_style(name, style) #:nodoc:
428401
@queued_for_write[name] = style.processors.inject(@queued_for_write[:original]) do |file, processor|
429402
Paperclip.processor(processor).make(file, style.processor_options, self)
430403
end
404+
@queued_for_write[name] = Paperclip.io_adapters.for(@queued_for_write[name])
431405
rescue Paperclip::Error => e
432406
log("An error was received while processing: #{e.inspect}")
433407
(@errors[:processing] ||= []) << e.message if @options[:whiny]
@@ -463,8 +437,8 @@ def flush_errors #:nodoc:
463437
# called by storage after the writes are flushed and before @queued_for_writes is cleared
464438
def after_flush_writes
465439
@queued_for_write.each do |style, file|
466-
file.close unless file.closed?
467-
file.unlink if file.respond_to?(:unlink) && file.path.present? && File.exist?(file.path)
440+
# file.close unless file.closed?
441+
# file.unlink if file.respond_to?(:unlink) && file.path.present? && File.exist?(file.path)
468442
end
469443
end
470444

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
module Paperclip
2+
class AttachmentAdapter
3+
4+
def initialize(target)
5+
@target = target
6+
cache_current_values
7+
end
8+
9+
def original_filename
10+
@original_filename
11+
end
12+
13+
def content_type
14+
@content_type
15+
end
16+
17+
def size
18+
@size
19+
end
20+
21+
def nil?
22+
false
23+
end
24+
25+
def fingerprint
26+
@fingerprint ||= Digest::MD5.file(path).to_s
27+
end
28+
29+
def read(length = nil, buffer = nil)
30+
@tempfile.read(length, buffer)
31+
end
32+
33+
def eof?
34+
@tempfile.eof?
35+
end
36+
37+
def path
38+
@tempfile.path
39+
end
40+
41+
private
42+
43+
def cache_current_values
44+
@tempfile = copy_to_tempfile(@target)
45+
@original_filename = @target.original_filename
46+
@content_type = @target.content_type
47+
@size = @tempfile.size || @target.size
48+
end
49+
50+
def copy_to_tempfile(src)
51+
dest = Tempfile.new(src.original_filename)
52+
FileUtils.cp(src.path(:original), dest.path)
53+
dest
54+
end
55+
56+
end
57+
end
58+
59+
Paperclip.io_adapters.register Paperclip::AttachmentAdapter do |target|
60+
Paperclip::Attachment === target
61+
end

0 commit comments

Comments
 (0)