rsturim-paperclip 2.2.9.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (47) hide show
  1. data/LICENSE +26 -0
  2. data/README.rdoc +174 -0
  3. data/Rakefile +99 -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 +350 -0
  9. data/lib/paperclip/attachment.rb +413 -0
  10. data/lib/paperclip/callback_compatability.rb +33 -0
  11. data/lib/paperclip/geometry.rb +115 -0
  12. data/lib/paperclip/interpolations.rb +105 -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 +241 -0
  21. data/lib/paperclip/thumbnail.rb +70 -0
  22. data/lib/paperclip/upfile.rb +48 -0
  23. data/shoulda_macros/paperclip.rb +68 -0
  24. data/tasks/paperclip_tasks.rake +79 -0
  25. data/test/attachment_test.rb +768 -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 +4 -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 +100 -0
  36. data/test/integration_test.rb +481 -0
  37. data/test/interpolations_test.rb +120 -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 +291 -0
  44. data/test/processor_test.rb +10 -0
  45. data/test/storage_test.rb +293 -0
  46. data/test/thumbnail_test.rb +177 -0
  47. metadata +124 -0
@@ -0,0 +1,10 @@
1
+ require 'test/helper'
2
+
3
+ class ProcessorTest < Test::Unit::TestCase
4
+ should "instantiate and call #make when sent #make to the class" do
5
+ processor = mock
6
+ processor.expects(:make).with()
7
+ Paperclip::Processor.expects(:new).with(:one, :two, :three).returns(processor)
8
+ Paperclip::Processor.make(:one, :two, :three)
9
+ end
10
+ end
@@ -0,0 +1,293 @@
1
+ require 'test/helper'
2
+
3
+ class StorageTest < Test::Unit::TestCase
4
+ context "Parsing S3 credentials" do
5
+ setup do
6
+ rebuild_model :storage => :s3,
7
+ :bucket => "testing",
8
+ :s3_credentials => {:not => :important}
9
+
10
+ @dummy = Dummy.new
11
+ @avatar = @dummy.avatar
12
+
13
+ @current_env = RAILS_ENV
14
+ end
15
+
16
+ teardown do
17
+ Object.const_set("RAILS_ENV", @current_env)
18
+ end
19
+
20
+ should "get the correct credentials when RAILS_ENV is production" do
21
+ Object.const_set('RAILS_ENV', "production")
22
+ assert_equal({:key => "12345"},
23
+ @avatar.parse_credentials('production' => {:key => '12345'},
24
+ :development => {:key => "54321"}))
25
+ end
26
+
27
+ should "get the correct credentials when RAILS_ENV is development" do
28
+ Object.const_set('RAILS_ENV', "development")
29
+ assert_equal({:key => "54321"},
30
+ @avatar.parse_credentials('production' => {:key => '12345'},
31
+ :development => {:key => "54321"}))
32
+ end
33
+
34
+ should "return the argument if the key does not exist" do
35
+ Object.const_set('RAILS_ENV', "not really an env")
36
+ assert_equal({:test => "12345"}, @avatar.parse_credentials(:test => "12345"))
37
+ end
38
+ end
39
+
40
+ context "" do
41
+ setup do
42
+ rebuild_model :storage => :s3,
43
+ :s3_credentials => {},
44
+ :bucket => "bucket",
45
+ :path => ":attachment/:basename.:extension",
46
+ :url => ":s3_path_url"
47
+ @dummy = Dummy.new
48
+ @dummy.avatar = StringIO.new(".")
49
+ end
50
+
51
+ should "return a url based on an S3 path" do
52
+ assert_match %r{^http://s3.amazonaws.com/bucket/avatars/stringio.txt}, @dummy.avatar.url
53
+ end
54
+ end
55
+ context "" do
56
+ setup do
57
+ rebuild_model :storage => :s3,
58
+ :s3_credentials => {},
59
+ :bucket => "bucket",
60
+ :path => ":attachment/:basename.:extension",
61
+ :url => ":s3_domain_url"
62
+ @dummy = Dummy.new
63
+ @dummy.avatar = StringIO.new(".")
64
+ end
65
+
66
+ should "return a url based on an S3 subdomain" do
67
+ assert_match %r{^http://bucket.s3.amazonaws.com/avatars/stringio.txt}, @dummy.avatar.url
68
+ end
69
+ end
70
+ context "" do
71
+ setup do
72
+ rebuild_model :storage => :s3,
73
+ :s3_credentials => {
74
+ :production => { :bucket => "prod_bucket" },
75
+ :development => { :bucket => "dev_bucket" }
76
+ },
77
+ :s3_host_alias => "something.something.com",
78
+ :path => ":attachment/:basename.:extension",
79
+ :url => ":s3_alias_url"
80
+ @dummy = Dummy.new
81
+ @dummy.avatar = StringIO.new(".")
82
+ end
83
+
84
+ should "return a url based on the host_alias" do
85
+ assert_match %r{^http://something.something.com/avatars/stringio.txt}, @dummy.avatar.url
86
+ end
87
+ end
88
+
89
+ context "Parsing S3 credentials with a bucket in them" do
90
+ setup do
91
+ rebuild_model :storage => :s3,
92
+ :s3_credentials => {
93
+ :production => { :bucket => "prod_bucket" },
94
+ :development => { :bucket => "dev_bucket" }
95
+ }
96
+ @dummy = Dummy.new
97
+ @old_env = RAILS_ENV
98
+ end
99
+
100
+ teardown{ Object.const_set("RAILS_ENV", @old_env) }
101
+
102
+ should "get the right bucket in production" do
103
+ Object.const_set("RAILS_ENV", "production")
104
+ assert_equal "prod_bucket", @dummy.avatar.bucket_name
105
+ end
106
+
107
+ should "get the right bucket in development" do
108
+ Object.const_set("RAILS_ENV", "development")
109
+ assert_equal "dev_bucket", @dummy.avatar.bucket_name
110
+ end
111
+ end
112
+
113
+ context "An attachment with S3 storage" do
114
+ setup do
115
+ rebuild_model :storage => :s3,
116
+ :bucket => "testing",
117
+ :path => ":attachment/:style/:basename.:extension",
118
+ :s3_credentials => {
119
+ 'access_key_id' => "12345",
120
+ 'secret_access_key' => "54321"
121
+ }
122
+ end
123
+
124
+ should "be extended by the S3 module" do
125
+ assert Dummy.new.avatar.is_a?(Paperclip::Storage::S3)
126
+ end
127
+
128
+ should "not be extended by the Filesystem module" do
129
+ assert ! Dummy.new.avatar.is_a?(Paperclip::Storage::Filesystem)
130
+ end
131
+
132
+ context "'to file'" do
133
+ setup do
134
+ @dummy = Dummy.new.avatar
135
+ @dummy.stubs(:s3_bucket).returns(stub(:key => stub(:data => "giraffe")))
136
+ end
137
+
138
+ should "return a File like object" do
139
+ assert @dummy.to_file.respond_to?(:read)
140
+ end
141
+ end
142
+
143
+ context "when assigned" do
144
+ setup do
145
+ @file = File.new(File.join(File.dirname(__FILE__), 'fixtures', '5k.png'), 'rb')
146
+ @dummy = Dummy.new
147
+ @dummy.avatar = @file
148
+ end
149
+
150
+ teardown { @file.close }
151
+
152
+ should "not get a bucket to get a URL" do
153
+ @dummy.avatar.expects(:s3).never
154
+ @dummy.avatar.expects(:s3_bucket).never
155
+ assert_match %r{^http://s3\.amazonaws\.com/testing/avatars/original/5k\.png}, @dummy.avatar.url
156
+ end
157
+
158
+ context "and saved" do
159
+ setup do
160
+ @s3_mock = stub
161
+ @bucket_mock = stub
162
+ RightAws::S3.expects(:new).with("12345", "54321", {}).returns(@s3_mock)
163
+ @s3_mock.expects(:bucket).with("testing", true, "public-read").returns(@bucket_mock)
164
+ @key_mock = stub
165
+ @bucket_mock.expects(:key).returns(@key_mock)
166
+ @key_mock.expects(:data=)
167
+ @key_mock.expects(:put).with(nil, 'public-read', 'Content-type' => 'image/png')
168
+ @dummy.save
169
+ end
170
+
171
+ should "succeed" do
172
+ assert true
173
+ end
174
+ end
175
+
176
+ context "and remove" do
177
+ setup do
178
+ @s3_mock = stub
179
+ @bucket_mock = stub
180
+ RightAws::S3.expects(:new).with("12345", "54321", {}).returns(@s3_mock)
181
+ @s3_mock.expects(:bucket).with("testing", true, "public-read").returns(@bucket_mock)
182
+ @key_mock = stub
183
+ @bucket_mock.expects(:key).at_least(2).returns(@key_mock)
184
+ @key_mock.expects(:delete)
185
+ @dummy.destroy_attached_files
186
+ end
187
+
188
+ should "succeed" do
189
+ assert true
190
+ end
191
+ end
192
+ end
193
+ end
194
+
195
+ context "An attachment with S3 storage and bucket defined as a Proc" do
196
+ setup do
197
+ rebuild_model :storage => :s3,
198
+ :bucket => lambda { |attachment| "bucket_#{attachment.instance.other}" },
199
+ :s3_credentials => {:not => :important}
200
+ end
201
+
202
+ should "get the right bucket name" do
203
+ assert "bucket_a", Dummy.new(:other => 'a').avatar.bucket_name
204
+ assert "bucket_b", Dummy.new(:other => 'b').avatar.bucket_name
205
+ end
206
+ end
207
+
208
+ context "An attachment with S3 storage and specific s3 headers set" do
209
+ setup do
210
+ rebuild_model :storage => :s3,
211
+ :bucket => "testing",
212
+ :path => ":attachment/:style/:basename.:extension",
213
+ :s3_credentials => {
214
+ 'access_key_id' => "12345",
215
+ 'secret_access_key' => "54321"
216
+ },
217
+ :s3_headers => {'Cache-Control' => 'max-age=31557600'}
218
+ end
219
+
220
+ context "when assigned" do
221
+ setup do
222
+ @file = File.new(File.join(File.dirname(__FILE__), 'fixtures', '5k.png'), 'rb')
223
+ @dummy = Dummy.new
224
+ @dummy.avatar = @file
225
+ end
226
+
227
+ teardown { @file.close }
228
+
229
+ context "and saved" do
230
+ setup do
231
+ @s3_mock = stub
232
+ @bucket_mock = stub
233
+ RightAws::S3.expects(:new).with("12345", "54321", {}).returns(@s3_mock)
234
+ @s3_mock.expects(:bucket).with("testing", true, "public-read").returns(@bucket_mock)
235
+ @key_mock = stub
236
+ @bucket_mock.expects(:key).returns(@key_mock)
237
+ @key_mock.expects(:data=)
238
+ @key_mock.expects(:put).with(nil,
239
+ 'public-read',
240
+ 'Content-type' => 'image/png',
241
+ 'Cache-Control' => 'max-age=31557600')
242
+ @dummy.save
243
+ end
244
+
245
+ should "succeed" do
246
+ assert true
247
+ end
248
+ end
249
+ end
250
+ end
251
+
252
+ unless ENV["S3_TEST_BUCKET"].blank?
253
+ context "Using S3 for real, an attachment with S3 storage" do
254
+ setup do
255
+ rebuild_model :styles => { :thumb => "100x100", :square => "32x32#" },
256
+ :storage => :s3,
257
+ :bucket => ENV["S3_TEST_BUCKET"],
258
+ :path => ":class/:attachment/:id/:style.:extension",
259
+ :s3_credentials => File.new(File.join(File.dirname(__FILE__), "s3.yml"))
260
+
261
+ Dummy.delete_all
262
+ @dummy = Dummy.new
263
+ end
264
+
265
+ should "be extended by the S3 module" do
266
+ assert Dummy.new.avatar.is_a?(Paperclip::Storage::S3)
267
+ end
268
+
269
+ context "when assigned" do
270
+ setup do
271
+ @file = File.new(File.join(File.dirname(__FILE__), 'fixtures', '5k.png'), 'rb')
272
+ @dummy.avatar = @file
273
+ end
274
+
275
+ teardown { @file.close }
276
+
277
+ should "still return a Tempfile when sent #to_io" do
278
+ assert_equal Tempfile, @dummy.avatar.to_io.class
279
+ end
280
+
281
+ context "and saved" do
282
+ setup do
283
+ @dummy.save
284
+ end
285
+
286
+ should "be on S3" do
287
+ assert true
288
+ end
289
+ end
290
+ end
291
+ end
292
+ end
293
+ end
@@ -0,0 +1,177 @@
1
+ require 'test/helper'
2
+
3
+ class ThumbnailTest < Test::Unit::TestCase
4
+
5
+ context "A Paperclip Tempfile" do
6
+ setup do
7
+ @tempfile = Paperclip::Tempfile.new("file.jpg")
8
+ end
9
+
10
+ should "have its path contain a real extension" do
11
+ assert_equal ".jpg", File.extname(@tempfile.path)
12
+ end
13
+
14
+ should "be a real Tempfile" do
15
+ assert @tempfile.is_a?(::Tempfile)
16
+ end
17
+ end
18
+
19
+ context "Another Paperclip Tempfile" do
20
+ setup do
21
+ @tempfile = Paperclip::Tempfile.new("file")
22
+ end
23
+
24
+ should "not have an extension if not given one" do
25
+ assert_equal "", File.extname(@tempfile.path)
26
+ end
27
+
28
+ should "still be a real Tempfile" do
29
+ assert @tempfile.is_a?(::Tempfile)
30
+ end
31
+ end
32
+
33
+ context "An image" do
34
+ setup do
35
+ @file = File.new(File.join(File.dirname(__FILE__), "fixtures", "5k.png"), 'rb')
36
+ end
37
+
38
+ teardown { @file.close }
39
+
40
+ [["600x600>", "434x66"],
41
+ ["400x400>", "400x61"],
42
+ ["32x32<", "434x66"]
43
+ ].each do |args|
44
+ context "being thumbnailed with a geometry of #{args[0]}" do
45
+ setup do
46
+ @thumb = Paperclip::Thumbnail.new(@file, :geometry => args[0])
47
+ end
48
+
49
+ should "start with dimensions of 434x66" do
50
+ cmd = %Q[identify -format "%wx%h" "#{@file.path}"]
51
+ assert_equal "434x66", `#{cmd}`.chomp
52
+ end
53
+
54
+ should "report the correct target geometry" do
55
+ assert_equal args[0], @thumb.target_geometry.to_s
56
+ end
57
+
58
+ context "when made" do
59
+ setup do
60
+ @thumb_result = @thumb.make
61
+ end
62
+
63
+ should "be the size we expect it to be" do
64
+ cmd = %Q[identify -format "%wx%h" "#{@thumb_result.path}"]
65
+ assert_equal args[1], `#{cmd}`.chomp
66
+ end
67
+ end
68
+ end
69
+ end
70
+
71
+ context "being thumbnailed at 100x50 with cropping" do
72
+ setup do
73
+ @thumb = Paperclip::Thumbnail.new(@file, :geometry => "100x50#")
74
+ end
75
+
76
+ should "report its correct current and target geometries" do
77
+ assert_equal "100x50#", @thumb.target_geometry.to_s
78
+ assert_equal "434x66", @thumb.current_geometry.to_s
79
+ end
80
+
81
+ should "report its correct format" do
82
+ assert_nil @thumb.format
83
+ end
84
+
85
+ should "have whiny turned on by default" do
86
+ assert @thumb.whiny
87
+ end
88
+
89
+ should "have convert_options set to nil by default" do
90
+ assert_equal nil, @thumb.convert_options
91
+ end
92
+
93
+ should "send the right command to convert when sent #make" do
94
+ Paperclip.expects(:"`").with do |arg|
95
+ arg.match %r{convert\s+"#{File.expand_path(@thumb.file.path)}\[0\]"\s+-resize\s+\"x50\"\s+-crop\s+\"100x50\+114\+0\"\s+\+repage\s+".*?"}
96
+ end
97
+ @thumb.make
98
+ end
99
+
100
+ should "create the thumbnail when sent #make" do
101
+ dst = @thumb.make
102
+ assert_match /100x50/, `identify "#{dst.path}"`
103
+ end
104
+ end
105
+
106
+ context "being thumbnailed with convert options set" do
107
+ setup do
108
+ @thumb = Paperclip::Thumbnail.new(@file,
109
+ :geometry => "100x50#",
110
+ :convert_options => "-strip -depth 8")
111
+ end
112
+
113
+ should "have convert_options value set" do
114
+ assert_equal "-strip -depth 8", @thumb.convert_options
115
+ end
116
+
117
+ should "send the right command to convert when sent #make" do
118
+ Paperclip.expects(:"`").with do |arg|
119
+ arg.match %r{convert\s+"#{File.expand_path(@thumb.file.path)}\[0\]"\s+-resize\s+"x50"\s+-crop\s+"100x50\+114\+0"\s+\+repage\s+-strip\s+-depth\s+8\s+".*?"}
120
+ end
121
+ @thumb.make
122
+ end
123
+
124
+ should "create the thumbnail when sent #make" do
125
+ dst = @thumb.make
126
+ assert_match /100x50/, `identify "#{dst.path}"`
127
+ end
128
+
129
+ context "redefined to have bad convert_options setting" do
130
+ setup do
131
+ @thumb = Paperclip::Thumbnail.new(@file,
132
+ :geometry => "100x50#",
133
+ :convert_options => "-this-aint-no-option")
134
+ end
135
+
136
+ should "error when trying to create the thumbnail" do
137
+ assert_raises(Paperclip::PaperclipError) do
138
+ @thumb.make
139
+ end
140
+ end
141
+ end
142
+ end
143
+ end
144
+
145
+ context "A multipage PDF" do
146
+ setup do
147
+ @file = File.new(File.join(File.dirname(__FILE__), "fixtures", "twopage.pdf"), 'rb')
148
+ end
149
+
150
+ teardown { @file.close }
151
+
152
+ should "start with two pages with dimensions 612x792" do
153
+ cmd = %Q[identify -format "%wx%h" "#{@file.path}"]
154
+ assert_equal "612x792"*2, `#{cmd}`.chomp
155
+ end
156
+
157
+ context "being thumbnailed at 100x100 with cropping" do
158
+ setup do
159
+ @thumb = Paperclip::Thumbnail.new(@file, :geometry => "100x100#", :format => :png)
160
+ end
161
+
162
+ should "report its correct current and target geometries" do
163
+ assert_equal "100x100#", @thumb.target_geometry.to_s
164
+ assert_equal "612x792", @thumb.current_geometry.to_s
165
+ end
166
+
167
+ should "report its correct format" do
168
+ assert_equal :png, @thumb.format
169
+ end
170
+
171
+ should "create the thumbnail when sent #make" do
172
+ dst = @thumb.make
173
+ assert_match /100x100/, `identify "#{dst.path}"`
174
+ end
175
+ end
176
+ end
177
+ end