paperclip_with_versions 2.3.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. data/LICENSE +26 -0
  2. data/README.rdoc +174 -0
  3. data/Rakefile +103 -0
  4. data/generators/paperclip/USAGE +5 -0
  5. data/generators/paperclip/paperclip_generator.rb +27 -0
  6. data/generators/paperclip/templates/paperclip_migration.rb.erb +19 -0
  7. data/init.rb +1 -0
  8. data/lib/paperclip.rb +356 -0
  9. data/lib/paperclip/attachment.rb +414 -0
  10. data/lib/paperclip/callback_compatability.rb +33 -0
  11. data/lib/paperclip/geometry.rb +115 -0
  12. data/lib/paperclip/interpolations.rb +108 -0
  13. data/lib/paperclip/iostream.rb +58 -0
  14. data/lib/paperclip/matchers.rb +4 -0
  15. data/lib/paperclip/matchers/have_attached_file_matcher.rb +49 -0
  16. data/lib/paperclip/matchers/validate_attachment_content_type_matcher.rb +66 -0
  17. data/lib/paperclip/matchers/validate_attachment_presence_matcher.rb +48 -0
  18. data/lib/paperclip/matchers/validate_attachment_size_matcher.rb +83 -0
  19. data/lib/paperclip/processor.rb +49 -0
  20. data/lib/paperclip/storage.rb +243 -0
  21. data/lib/paperclip/thumbnail.rb +73 -0
  22. data/lib/paperclip/upfile.rb +49 -0
  23. data/shoulda_macros/paperclip.rb +117 -0
  24. data/tasks/paperclip_tasks.rake +79 -0
  25. data/test/attachment_test.rb +815 -0
  26. data/test/database.yml +4 -0
  27. data/test/fixtures/12k.png +0 -0
  28. data/test/fixtures/50x50.png +0 -0
  29. data/test/fixtures/5k.png +0 -0
  30. data/test/fixtures/bad.png +1 -0
  31. data/test/fixtures/s3.yml +8 -0
  32. data/test/fixtures/text.txt +0 -0
  33. data/test/fixtures/twopage.pdf +0 -0
  34. data/test/geometry_test.rb +177 -0
  35. data/test/helper.rb +108 -0
  36. data/test/integration_test.rb +483 -0
  37. data/test/interpolations_test.rb +124 -0
  38. data/test/iostream_test.rb +71 -0
  39. data/test/matchers/have_attached_file_matcher_test.rb +21 -0
  40. data/test/matchers/validate_attachment_content_type_matcher_test.rb +30 -0
  41. data/test/matchers/validate_attachment_presence_matcher_test.rb +21 -0
  42. data/test/matchers/validate_attachment_size_matcher_test.rb +50 -0
  43. data/test/paperclip_test.rb +327 -0
  44. data/test/processor_test.rb +10 -0
  45. data/test/storage_test.rb +303 -0
  46. data/test/thumbnail_test.rb +227 -0
  47. data/test/upfile_test.rb +28 -0
  48. metadata +161 -0
@@ -0,0 +1,4 @@
1
+ test:
2
+ adapter: sqlite3
3
+ database: ":memory:"
4
+
Binary file
Binary file
Binary file
@@ -0,0 +1 @@
1
+ This is not an image.
@@ -0,0 +1,8 @@
1
+ development:
2
+ key: 54321
3
+ production:
4
+ key: 12345
5
+ test:
6
+ bucket: <%= ENV['S3_BUCKET'] %>
7
+ access_key_id: <%= ENV['S3_KEY'] %>
8
+ secret_access_key: <%= ENV['S3_SECRET'] %>
File without changes
Binary file
@@ -0,0 +1,177 @@
1
+ require 'test/helper'
2
+
3
+ class GeometryTest < Test::Unit::TestCase
4
+ context "Paperclip::Geometry" do
5
+ should "correctly report its given dimensions" do
6
+ assert @geo = Paperclip::Geometry.new(1024, 768)
7
+ assert_equal 1024, @geo.width
8
+ assert_equal 768, @geo.height
9
+ end
10
+
11
+ should "set height to 0 if height dimension is missing" do
12
+ assert @geo = Paperclip::Geometry.new(1024)
13
+ assert_equal 1024, @geo.width
14
+ assert_equal 0, @geo.height
15
+ end
16
+
17
+ should "set width to 0 if width dimension is missing" do
18
+ assert @geo = Paperclip::Geometry.new(nil, 768)
19
+ assert_equal 0, @geo.width
20
+ assert_equal 768, @geo.height
21
+ end
22
+
23
+ should "be generated from a WxH-formatted string" do
24
+ assert @geo = Paperclip::Geometry.parse("800x600")
25
+ assert_equal 800, @geo.width
26
+ assert_equal 600, @geo.height
27
+ end
28
+
29
+ should "be generated from a xH-formatted string" do
30
+ assert @geo = Paperclip::Geometry.parse("x600")
31
+ assert_equal 0, @geo.width
32
+ assert_equal 600, @geo.height
33
+ end
34
+
35
+ should "be generated from a Wx-formatted string" do
36
+ assert @geo = Paperclip::Geometry.parse("800x")
37
+ assert_equal 800, @geo.width
38
+ assert_equal 0, @geo.height
39
+ end
40
+
41
+ should "be generated from a W-formatted string" do
42
+ assert @geo = Paperclip::Geometry.parse("800")
43
+ assert_equal 800, @geo.width
44
+ assert_equal 0, @geo.height
45
+ end
46
+
47
+ should "ensure the modifier is nil if not present" do
48
+ assert @geo = Paperclip::Geometry.parse("123x456")
49
+ assert_nil @geo.modifier
50
+ end
51
+
52
+ should "treat x and X the same in geometries" do
53
+ @lower = Paperclip::Geometry.parse("123x456")
54
+ @upper = Paperclip::Geometry.parse("123X456")
55
+ assert_equal 123, @lower.width
56
+ assert_equal 123, @upper.width
57
+ assert_equal 456, @lower.height
58
+ assert_equal 456, @upper.height
59
+ end
60
+
61
+ ['>', '<', '#', '@', '%', '^', '!', nil].each do |mod|
62
+ should "ensure the modifier #{mod.inspect} is preserved" do
63
+ assert @geo = Paperclip::Geometry.parse("123x456#{mod}")
64
+ assert_equal mod, @geo.modifier
65
+ assert_equal "123x456#{mod}", @geo.to_s
66
+ end
67
+ end
68
+
69
+ ['>', '<', '#', '@', '%', '^', '!', nil].each do |mod|
70
+ should "ensure the modifier #{mod.inspect} is preserved with no height" do
71
+ assert @geo = Paperclip::Geometry.parse("123x#{mod}")
72
+ assert_equal mod, @geo.modifier
73
+ assert_equal "123#{mod}", @geo.to_s
74
+ end
75
+ end
76
+
77
+ should "make sure the modifier gets passed during transformation_to" do
78
+ assert @src = Paperclip::Geometry.parse("123x456")
79
+ assert @dst = Paperclip::Geometry.parse("123x456>")
80
+ assert_equal ["123x456>", nil], @src.transformation_to(@dst)
81
+ end
82
+
83
+ should "generate correct ImageMagick formatting string for W-formatted string" do
84
+ assert @geo = Paperclip::Geometry.parse("800")
85
+ assert_equal "800", @geo.to_s
86
+ end
87
+
88
+ should "generate correct ImageMagick formatting string for Wx-formatted string" do
89
+ assert @geo = Paperclip::Geometry.parse("800x")
90
+ assert_equal "800", @geo.to_s
91
+ end
92
+
93
+ should "generate correct ImageMagick formatting string for xH-formatted string" do
94
+ assert @geo = Paperclip::Geometry.parse("x600")
95
+ assert_equal "x600", @geo.to_s
96
+ end
97
+
98
+ should "generate correct ImageMagick formatting string for WxH-formatted string" do
99
+ assert @geo = Paperclip::Geometry.parse("800x600")
100
+ assert_equal "800x600", @geo.to_s
101
+ end
102
+
103
+ should "be generated from a file" do
104
+ file = File.join(File.dirname(__FILE__), "fixtures", "5k.png")
105
+ file = File.new(file, 'rb')
106
+ assert_nothing_raised{ @geo = Paperclip::Geometry.from_file(file) }
107
+ assert @geo.height > 0
108
+ assert @geo.width > 0
109
+ end
110
+
111
+ should "be generated from a file path" do
112
+ file = File.join(File.dirname(__FILE__), "fixtures", "5k.png")
113
+ assert_nothing_raised{ @geo = Paperclip::Geometry.from_file(file) }
114
+ assert @geo.height > 0
115
+ assert @geo.width > 0
116
+ end
117
+
118
+ should "not generate from a bad file" do
119
+ file = "/home/This File Does Not Exist.omg"
120
+ assert_raise(Paperclip::NotIdentifiedByImageMagickError){ @geo = Paperclip::Geometry.from_file(file) }
121
+ end
122
+
123
+ [['vertical', 900, 1440, true, false, false, 1440, 900, 0.625],
124
+ ['horizontal', 1024, 768, false, true, false, 1024, 768, 1.3333],
125
+ ['square', 100, 100, false, false, true, 100, 100, 1]].each do |args|
126
+ context "performing calculations on a #{args[0]} viewport" do
127
+ setup do
128
+ @geo = Paperclip::Geometry.new(args[1], args[2])
129
+ end
130
+
131
+ should "#{args[3] ? "" : "not"} be vertical" do
132
+ assert_equal args[3], @geo.vertical?
133
+ end
134
+
135
+ should "#{args[4] ? "" : "not"} be horizontal" do
136
+ assert_equal args[4], @geo.horizontal?
137
+ end
138
+
139
+ should "#{args[5] ? "" : "not"} be square" do
140
+ assert_equal args[5], @geo.square?
141
+ end
142
+
143
+ should "report that #{args[6]} is the larger dimension" do
144
+ assert_equal args[6], @geo.larger
145
+ end
146
+
147
+ should "report that #{args[7]} is the smaller dimension" do
148
+ assert_equal args[7], @geo.smaller
149
+ end
150
+
151
+ should "have an aspect ratio of #{args[8]}" do
152
+ assert_in_delta args[8], @geo.aspect, 0.0001
153
+ end
154
+ end
155
+ end
156
+
157
+ [[ [1000, 100], [64, 64], "x64", "64x64+288+0" ],
158
+ [ [100, 1000], [50, 950], "x950", "50x950+22+0" ],
159
+ [ [100, 1000], [50, 25], "50x", "50x25+0+237" ]]. each do |args|
160
+ context "of #{args[0].inspect} and given a Geometry #{args[1].inspect} and sent transform_to" do
161
+ setup do
162
+ @geo = Paperclip::Geometry.new(*args[0])
163
+ @dst = Paperclip::Geometry.new(*args[1])
164
+ @scale, @crop = @geo.transformation_to @dst, true
165
+ end
166
+
167
+ should "be able to return the correct scaling transformation geometry #{args[2]}" do
168
+ assert_equal args[2], @scale
169
+ end
170
+
171
+ should "be able to return the correct crop transformation geometry #{args[3]}" do
172
+ assert_equal args[3], @crop
173
+ end
174
+ end
175
+ end
176
+ end
177
+ end
@@ -0,0 +1,108 @@
1
+ require 'rubygems'
2
+ require 'test/unit'
3
+ require 'shoulda'
4
+ require 'tempfile'
5
+
6
+ gem 'jferris-mocha'#, '0.9.5.0.1241126838'
7
+ require 'mocha'
8
+
9
+ gem 'sqlite3-ruby'
10
+
11
+ require 'active_record'
12
+ require 'active_support'
13
+ begin
14
+ require 'ruby-debug'
15
+ rescue LoadError
16
+ puts "ruby-debug not loaded"
17
+ end
18
+
19
+ ROOT = File.join(File.dirname(__FILE__), '..')
20
+ RAILS_ROOT = ROOT
21
+ RAILS_ENV = "test"
22
+
23
+ $LOAD_PATH << File.join(ROOT, 'lib')
24
+ $LOAD_PATH << File.join(ROOT, 'lib', 'paperclip')
25
+
26
+ require File.join(ROOT, 'lib', 'paperclip.rb')
27
+
28
+ require 'shoulda_macros/paperclip'
29
+
30
+ FIXTURES_DIR = File.join(File.dirname(__FILE__), "fixtures")
31
+ config = YAML::load(IO.read(File.dirname(__FILE__) + '/database.yml'))
32
+ ActiveRecord::Base.logger = Logger.new(File.dirname(__FILE__) + "/debug.log")
33
+ ActiveRecord::Base.establish_connection(config['test'])
34
+
35
+ def reset_class class_name
36
+ ActiveRecord::Base.send(:include, Paperclip)
37
+ Object.send(:remove_const, class_name) rescue nil
38
+ klass = Object.const_set(class_name, Class.new(ActiveRecord::Base))
39
+ klass.class_eval{ include Paperclip }
40
+ klass
41
+ end
42
+
43
+ def reset_table table_name, &block
44
+ block ||= lambda { |table| true }
45
+ ActiveRecord::Base.connection.create_table :dummies, {:force => true}, &block
46
+ end
47
+
48
+ def modify_table table_name, &block
49
+ ActiveRecord::Base.connection.change_table :dummies, &block
50
+ end
51
+
52
+ def rebuild_model options = {}
53
+ ActiveRecord::Base.connection.create_table :dummies, :force => true do |table|
54
+ table.column :other, :string
55
+ table.column :avatar_file_name, :string
56
+ table.column :avatar_content_type, :string
57
+ table.column :avatar_file_size, :integer
58
+ table.column :avatar_updated_at, :datetime
59
+ end
60
+ rebuild_class options
61
+ end
62
+
63
+ def rebuild_class options = {}
64
+ ActiveRecord::Base.send(:include, Paperclip)
65
+ Object.send(:remove_const, "Dummy") rescue nil
66
+ Object.const_set("Dummy", Class.new(ActiveRecord::Base))
67
+ Dummy.class_eval do
68
+ include Paperclip
69
+ has_attached_file :avatar, options
70
+ end
71
+ end
72
+
73
+ def temporary_rails_env(new_env)
74
+ old_env = Object.const_defined?("RAILS_ENV") ? RAILS_ENV : nil
75
+ silence_warnings do
76
+ Object.const_set("RAILS_ENV", new_env)
77
+ end
78
+ yield
79
+ silence_warnings do
80
+ Object.const_set("RAILS_ENV", old_env)
81
+ end
82
+ end
83
+
84
+ class FakeModel
85
+ attr_accessor :avatar_file_name,
86
+ :avatar_file_size,
87
+ :avatar_last_updated,
88
+ :avatar_content_type,
89
+ :id
90
+
91
+ def errors
92
+ @errors ||= []
93
+ end
94
+
95
+ def run_callbacks name, *args
96
+ end
97
+ end
98
+
99
+ def attachment options
100
+ Paperclip::Attachment.new(:avatar, FakeModel.new, options)
101
+ end
102
+
103
+ def silence_warnings
104
+ old_verbose, $VERBOSE = $VERBOSE, nil
105
+ yield
106
+ ensure
107
+ $VERBOSE = old_verbose
108
+ end
@@ -0,0 +1,483 @@
1
+ require 'test/helper'
2
+
3
+ class IntegrationTest < Test::Unit::TestCase
4
+ context "Many models at once" do
5
+ setup do
6
+ rebuild_model
7
+ @file = File.new(File.join(FIXTURES_DIR, "5k.png"), 'rb')
8
+ 300.times do |i|
9
+ Dummy.create! :avatar => @file
10
+ end
11
+ end
12
+
13
+ should "not exceed the open file limit" do
14
+ assert_nothing_raised do
15
+ dummies = Dummy.find(:all)
16
+ dummies.each { |dummy| dummy.avatar }
17
+ end
18
+ end
19
+ end
20
+
21
+ context "An attachment" do
22
+ setup do
23
+ rebuild_model :styles => { :thumb => "50x50#" }
24
+ @dummy = Dummy.new
25
+ @file = File.new(File.join(File.dirname(__FILE__),
26
+ "fixtures",
27
+ "5k.png"), 'rb')
28
+ @dummy.avatar = @file
29
+ assert @dummy.save
30
+ end
31
+
32
+ teardown { @file.close }
33
+
34
+ should "create its thumbnails properly" do
35
+ assert_match /\b50x50\b/, `identify "#{@dummy.avatar.path(:thumb)}"`
36
+ end
37
+
38
+ context "redefining its attachment styles" do
39
+ setup do
40
+ Dummy.class_eval do
41
+ has_attached_file :avatar, :styles => { :thumb => "150x25#" }
42
+ has_attached_file :avatar, :styles => { :thumb => "150x25#", :dynamic => lambda { |a| '50x50#' } }
43
+ end
44
+ @d2 = Dummy.find(@dummy.id)
45
+ @d2.avatar.reprocess!
46
+ @d2.save
47
+ end
48
+
49
+ should "create its thumbnails properly" do
50
+ assert_match /\b150x25\b/, `identify "#{@dummy.avatar.path(:thumb)}"`
51
+ assert_match /\b50x50\b/, `identify "#{@dummy.avatar.path(:dynamic)}"`
52
+ end
53
+ end
54
+ end
55
+
56
+ context "A model that modifies its original" do
57
+ setup do
58
+ rebuild_model :styles => { :original => "2x2#" }
59
+ @dummy = Dummy.new
60
+ @file = File.new(File.join(File.dirname(__FILE__),
61
+ "fixtures",
62
+ "5k.png"), 'rb')
63
+ @dummy.avatar = @file
64
+ end
65
+
66
+ should "report the file size of the processed file and not the original" do
67
+ assert_not_equal @file.size, @dummy.avatar.size
68
+ end
69
+
70
+ teardown { @file.close }
71
+ end
72
+
73
+ context "A model with attachments scoped under an id" do
74
+ setup do
75
+ rebuild_model :styles => { :large => "100x100",
76
+ :medium => "50x50" },
77
+ :path => ":rails_root/tmp/:id/:attachments/:style.:extension"
78
+ @dummy = Dummy.new
79
+ @file = File.new(File.join(File.dirname(__FILE__),
80
+ "fixtures",
81
+ "5k.png"), 'rb')
82
+ @dummy.avatar = @file
83
+ end
84
+
85
+ teardown { @file.close }
86
+
87
+ context "when saved" do
88
+ setup do
89
+ @dummy.save
90
+ @saved_path = @dummy.avatar.path(:large)
91
+ end
92
+
93
+ should "have a large file in the right place" do
94
+ assert File.exists?(@dummy.avatar.path(:large))
95
+ end
96
+
97
+ context "and deleted" do
98
+ setup do
99
+ @dummy.avatar.clear
100
+ @dummy.save
101
+ end
102
+
103
+ should "not have a large file in the right place anymore" do
104
+ assert ! File.exists?(@saved_path)
105
+ end
106
+
107
+ should "not have its next two parent directories" do
108
+ assert ! File.exists?(File.dirname(@saved_path))
109
+ assert ! File.exists?(File.dirname(File.dirname(@saved_path)))
110
+ end
111
+
112
+ before_should "not die if an unexpected SystemCallError happens" do
113
+ FileUtils.stubs(:rmdir).raises(Errno::EPIPE)
114
+ end
115
+ end
116
+ end
117
+ end
118
+
119
+ context "A model with no attachment validation" do
120
+ setup do
121
+ rebuild_model :styles => { :large => "300x300>",
122
+ :medium => "100x100",
123
+ :thumb => ["32x32#", :gif] },
124
+ :default_style => :medium,
125
+ :url => "/:attachment/:class/:style/:id/:basename.:extension",
126
+ :path => ":rails_root/tmp/:attachment/:class/:style/:id/:basename.:extension"
127
+ @dummy = Dummy.new
128
+ end
129
+
130
+ should "have its definition return false when asked about whiny_thumbnails" do
131
+ assert ! Dummy.attachment_definitions[:avatar][:whiny_thumbnails]
132
+ end
133
+
134
+ context "when validates_attachment_thumbnails is called" do
135
+ setup do
136
+ Dummy.validates_attachment_thumbnails :avatar
137
+ end
138
+
139
+ should "have its definition return true when asked about whiny_thumbnails" do
140
+ assert_equal true, Dummy.attachment_definitions[:avatar][:whiny_thumbnails]
141
+ end
142
+ end
143
+
144
+ context "redefined to have attachment validations" do
145
+ setup do
146
+ rebuild_model :styles => { :large => "300x300>",
147
+ :medium => "100x100",
148
+ :thumb => ["32x32#", :gif] },
149
+ :whiny_thumbnails => true,
150
+ :default_style => :medium,
151
+ :url => "/:attachment/:class/:style/:id/:basename.:extension",
152
+ :path => ":rails_root/tmp/:attachment/:class/:style/:id/:basename.:extension"
153
+ end
154
+
155
+ should "have its definition return true when asked about whiny_thumbnails" do
156
+ assert_equal true, Dummy.attachment_definitions[:avatar][:whiny_thumbnails]
157
+ end
158
+ end
159
+ end
160
+
161
+ context "A model with no convert_options setting" do
162
+ setup do
163
+ rebuild_model :styles => { :large => "300x300>",
164
+ :medium => "100x100",
165
+ :thumb => ["32x32#", :gif] },
166
+ :default_style => :medium,
167
+ :url => "/:attachment/:class/:style/:id/:basename.:extension",
168
+ :path => ":rails_root/tmp/:attachment/:class/:style/:id/:basename.:extension"
169
+ @dummy = Dummy.new
170
+ end
171
+
172
+ should "have its definition return nil when asked about convert_options" do
173
+ assert ! Dummy.attachment_definitions[:avatar][:convert_options]
174
+ end
175
+
176
+ context "redefined to have convert_options setting" do
177
+ setup do
178
+ rebuild_model :styles => { :large => "300x300>",
179
+ :medium => "100x100",
180
+ :thumb => ["32x32#", :gif] },
181
+ :convert_options => "-strip -depth 8",
182
+ :default_style => :medium,
183
+ :url => "/:attachment/:class/:style/:id/:basename.:extension",
184
+ :path => ":rails_root/tmp/:attachment/:class/:style/:id/:basename.:extension"
185
+ end
186
+
187
+ should "have its definition return convert_options value when asked about convert_options" do
188
+ assert_equal "-strip -depth 8", Dummy.attachment_definitions[:avatar][:convert_options]
189
+ end
190
+ end
191
+ end
192
+
193
+ context "A model with a filesystem attachment" do
194
+ setup do
195
+ rebuild_model :styles => { :large => "300x300>",
196
+ :medium => "100x100",
197
+ :thumb => ["32x32#", :gif] },
198
+ :whiny_thumbnails => true,
199
+ :default_style => :medium,
200
+ :url => "/:attachment/:class/:style/:id/:basename.:extension",
201
+ :path => ":rails_root/tmp/:attachment/:class/:style/:id/:basename.:extension"
202
+ @dummy = Dummy.new
203
+ @file = File.new(File.join(FIXTURES_DIR, "5k.png"), 'rb')
204
+ @bad_file = File.new(File.join(FIXTURES_DIR, "bad.png"), 'rb')
205
+
206
+ assert @dummy.avatar = @file
207
+ assert @dummy.valid?
208
+ assert @dummy.save
209
+ end
210
+
211
+ should "write and delete its files" do
212
+ [["434x66", :original],
213
+ ["300x46", :large],
214
+ ["100x15", :medium],
215
+ ["32x32", :thumb]].each do |geo, style|
216
+ cmd = %Q[identify -format "%wx%h" "#{@dummy.avatar.path(style)}"]
217
+ assert_equal geo, `#{cmd}`.chomp, cmd
218
+ end
219
+
220
+ saved_paths = [:thumb, :medium, :large, :original].collect{|s| @dummy.avatar.path(s) }
221
+
222
+ @d2 = Dummy.find(@dummy.id)
223
+ assert_equal "100x15", `identify -format "%wx%h" "#{@d2.avatar.path}"`.chomp
224
+ assert_equal "434x66", `identify -format "%wx%h" "#{@d2.avatar.path(:original)}"`.chomp
225
+ assert_equal "300x46", `identify -format "%wx%h" "#{@d2.avatar.path(:large)}"`.chomp
226
+ assert_equal "100x15", `identify -format "%wx%h" "#{@d2.avatar.path(:medium)}"`.chomp
227
+ assert_equal "32x32", `identify -format "%wx%h" "#{@d2.avatar.path(:thumb)}"`.chomp
228
+
229
+ @dummy.avatar = "not a valid file but not nil"
230
+ assert_equal File.basename(@file.path), @dummy.avatar_file_name
231
+ assert @dummy.valid?
232
+ assert @dummy.save
233
+
234
+ saved_paths.each do |p|
235
+ assert File.exists?(p)
236
+ end
237
+
238
+ @dummy.avatar.clear
239
+ assert_nil @dummy.avatar_file_name
240
+ assert @dummy.valid?
241
+ assert @dummy.save
242
+
243
+ saved_paths.each do |p|
244
+ assert ! File.exists?(p)
245
+ end
246
+
247
+ @d2 = Dummy.find(@dummy.id)
248
+ assert_nil @d2.avatar_file_name
249
+ end
250
+
251
+ should "work exactly the same when new as when reloaded" do
252
+ @d2 = Dummy.find(@dummy.id)
253
+
254
+ assert_equal @dummy.avatar_file_name, @d2.avatar_file_name
255
+ [:thumb, :medium, :large, :original].each do |style|
256
+ assert_equal @dummy.avatar.path(style), @d2.avatar.path(style)
257
+ end
258
+
259
+ saved_paths = [:thumb, :medium, :large, :original].collect{|s| @dummy.avatar.path(s) }
260
+
261
+ @d2.avatar.clear
262
+ assert @d2.save
263
+
264
+ saved_paths.each do |p|
265
+ assert ! File.exists?(p)
266
+ end
267
+ end
268
+
269
+ should "know the difference between good files, bad files, and not files" do
270
+ expected = @dummy.avatar.to_file
271
+ @dummy.avatar = "not a file"
272
+ assert @dummy.valid?
273
+ assert_equal expected.path, @dummy.avatar.path
274
+ expected.close
275
+
276
+ @dummy.avatar = @bad_file
277
+ assert ! @dummy.valid?
278
+ end
279
+
280
+ should "know the difference between good files, bad files, and not files when validating" do
281
+ Dummy.validates_attachment_presence :avatar
282
+ @d2 = Dummy.find(@dummy.id)
283
+ @d2.avatar = @file
284
+ assert @d2.valid?, @d2.errors.full_messages.inspect
285
+ @d2.avatar = @bad_file
286
+ assert ! @d2.valid?
287
+ end
288
+
289
+ should "be able to reload without saving and not have the file disappear" do
290
+ @dummy.avatar = @file
291
+ assert @dummy.save
292
+ @dummy.avatar.clear
293
+ assert_nil @dummy.avatar_file_name
294
+ @dummy.reload
295
+ assert_equal "5k.png", @dummy.avatar_file_name
296
+ end
297
+
298
+ context "that is assigned its file from another Paperclip attachment" do
299
+ setup do
300
+ @dummy2 = Dummy.new
301
+ @file2 = File.new(File.join(FIXTURES_DIR, "12k.png"), 'rb')
302
+ assert @dummy2.avatar = @file2
303
+ @dummy2.save
304
+ end
305
+
306
+ should "work when assigned a file" do
307
+ assert_not_equal `identify -format "%wx%h" "#{@dummy.avatar.path(:original)}"`,
308
+ `identify -format "%wx%h" "#{@dummy2.avatar.path(:original)}"`
309
+
310
+ assert @dummy.avatar = @dummy2.avatar
311
+ @dummy.save
312
+ assert_equal `identify -format "%wx%h" "#{@dummy.avatar.path(:original)}"`,
313
+ `identify -format "%wx%h" "#{@dummy2.avatar.path(:original)}"`
314
+ end
315
+ end
316
+
317
+ end
318
+
319
+ context "A model with an attachments association and a Paperclip attachment" do
320
+ setup do
321
+ Dummy.class_eval do
322
+ has_many :attachments, :class_name => 'Dummy'
323
+ end
324
+
325
+ @dummy = Dummy.new
326
+ @dummy.avatar = File.new(File.join(File.dirname(__FILE__),
327
+ "fixtures",
328
+ "5k.png"), 'rb')
329
+ end
330
+
331
+ should "should not error when saving" do
332
+ assert_nothing_raised do
333
+ @dummy.save!
334
+ end
335
+ end
336
+ end
337
+
338
+ if ENV['S3_TEST_BUCKET']
339
+ def s3_files_for attachment
340
+ [:thumb, :medium, :large, :original].inject({}) do |files, style|
341
+ data = `curl "#{attachment.url(style)}" 2>/dev/null`.chomp
342
+ t = Tempfile.new("paperclip-test")
343
+ t.binmode
344
+ t.write(data)
345
+ t.rewind
346
+ files[style] = t
347
+ files
348
+ end
349
+ end
350
+
351
+ def s3_headers_for attachment, style
352
+ `curl --head "#{attachment.url(style)}" 2>/dev/null`.split("\n").inject({}) do |h,head|
353
+ split_head = head.chomp.split(/\s*:\s*/, 2)
354
+ h[split_head.first.downcase] = split_head.last unless split_head.empty?
355
+ h
356
+ end
357
+ end
358
+
359
+ context "A model with an S3 attachment" do
360
+ setup do
361
+ rebuild_model :styles => { :large => "300x300>",
362
+ :medium => "100x100",
363
+ :thumb => ["32x32#", :gif] },
364
+ :storage => :s3,
365
+ :whiny_thumbnails => true,
366
+ # :s3_options => {:logger => Logger.new(StringIO.new)},
367
+ :s3_credentials => File.new(File.join(File.dirname(__FILE__), "s3.yml")),
368
+ :default_style => :medium,
369
+ :bucket => ENV['S3_TEST_BUCKET'],
370
+ :path => ":class/:attachment/:id/:style/:basename.:extension"
371
+ @dummy = Dummy.new
372
+ @file = File.new(File.join(FIXTURES_DIR, "5k.png"), 'rb')
373
+ @bad_file = File.new(File.join(FIXTURES_DIR, "bad.png"), 'rb')
374
+
375
+ assert @dummy.avatar = @file
376
+ assert @dummy.valid?
377
+ assert @dummy.save
378
+
379
+ @files_on_s3 = s3_files_for @dummy.avatar
380
+ end
381
+
382
+ should "have the same contents as the original" do
383
+ @file.rewind
384
+ assert_equal @file.read, @files_on_s3[:original].read
385
+ end
386
+
387
+ should "write and delete its files" do
388
+ [["434x66", :original],
389
+ ["300x46", :large],
390
+ ["100x15", :medium],
391
+ ["32x32", :thumb]].each do |geo, style|
392
+ cmd = %Q[identify -format "%wx%h" "#{@files_on_s3[style].path}"]
393
+ assert_equal geo, `#{cmd}`.chomp, cmd
394
+ end
395
+
396
+ @d2 = Dummy.find(@dummy.id)
397
+ @d2_files = s3_files_for @d2.avatar
398
+ [["434x66", :original],
399
+ ["300x46", :large],
400
+ ["100x15", :medium],
401
+ ["32x32", :thumb]].each do |geo, style|
402
+ cmd = %Q[identify -format "%wx%h" "#{@d2_files[style].path}"]
403
+ assert_equal geo, `#{cmd}`.chomp, cmd
404
+ end
405
+
406
+ @dummy.avatar = "not a valid file but not nil"
407
+ assert_equal File.basename(@file.path), @dummy.avatar_file_name
408
+ assert @dummy.valid?
409
+ assert @dummy.save
410
+
411
+ [:thumb, :medium, :large, :original].each do |style|
412
+ assert @dummy.avatar.exists?(style)
413
+ end
414
+
415
+ @dummy.avatar.clear
416
+ assert_nil @dummy.avatar_file_name
417
+ assert @dummy.valid?
418
+ assert @dummy.save
419
+
420
+ [:thumb, :medium, :large, :original].each do |style|
421
+ assert ! @dummy.avatar.exists?(style)
422
+ end
423
+
424
+ @d2 = Dummy.find(@dummy.id)
425
+ assert_nil @d2.avatar_file_name
426
+ end
427
+
428
+ should "work exactly the same when new as when reloaded" do
429
+ @d2 = Dummy.find(@dummy.id)
430
+
431
+ assert_equal @dummy.avatar_file_name, @d2.avatar_file_name
432
+ [:thumb, :medium, :large, :original].each do |style|
433
+ assert_equal @dummy.avatar.to_file(style).read, @d2.avatar.to_file(style).read
434
+ end
435
+
436
+ saved_keys = [:thumb, :medium, :large, :original].collect{|s| @dummy.avatar.to_file(s) }
437
+
438
+ @d2.avatar.clear
439
+ assert @d2.save
440
+
441
+ [:thumb, :medium, :large, :original].each do |style|
442
+ assert ! @dummy.avatar.exists?(style)
443
+ end
444
+ end
445
+
446
+ should "know the difference between good files, bad files, not files, and nil" do
447
+ expected = @dummy.avatar.to_file
448
+ @dummy.avatar = "not a file"
449
+ assert @dummy.valid?
450
+ assert_equal expected.read, @dummy.avatar.to_file.read
451
+
452
+ @dummy.avatar = @bad_file
453
+ assert ! @dummy.valid?
454
+ @dummy.avatar = nil
455
+ assert @dummy.valid?
456
+
457
+ Dummy.validates_attachment_presence :avatar
458
+ @d2 = Dummy.find(@dummy.id)
459
+ @d2.avatar = @file
460
+ assert @d2.valid?
461
+ @d2.avatar = @bad_file
462
+ assert ! @d2.valid?
463
+ @d2.avatar = nil
464
+ assert ! @d2.valid?
465
+ end
466
+
467
+ should "be able to reload without saving and not have the file disappear" do
468
+ @dummy.avatar = @file
469
+ assert @dummy.save
470
+ @dummy.avatar = nil
471
+ assert_nil @dummy.avatar_file_name
472
+ @dummy.reload
473
+ assert_equal "5k.png", @dummy.avatar_file_name
474
+ end
475
+
476
+ should "have the right content type" do
477
+ headers = s3_headers_for(@dummy.avatar, :original)
478
+ assert_equal 'image/png', headers['content-type']
479
+ end
480
+ end
481
+ end
482
+ end
483
+