radiant-paperclipped-extension 0.8.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (143) hide show
  1. data/.gitmodules +3 -0
  2. data/HELP_admin.markdown +69 -0
  3. data/LICENSE +21 -0
  4. data/README.md +137 -0
  5. data/Rakefile +110 -0
  6. data/VERSION +1 -0
  7. data/app/controllers/admin/assets_controller.rb +124 -0
  8. data/app/helpers/admin/assets_helper.rb +5 -0
  9. data/app/models/asset.rb +279 -0
  10. data/app/models/asset_page_tags.rb +101 -0
  11. data/app/models/asset_tags.rb +272 -0
  12. data/app/models/old_page_attachment.rb +26 -0
  13. data/app/models/page_attachment.rb +8 -0
  14. data/app/views/admin/assets/_asset.html.haml +19 -0
  15. data/app/views/admin/assets/_asset_table.html.haml +49 -0
  16. data/app/views/admin/assets/_assets_bucket.html.haml +8 -0
  17. data/app/views/admin/assets/_assets_container.html.haml +72 -0
  18. data/app/views/admin/assets/_bucket.html.haml +11 -0
  19. data/app/views/admin/assets/_bucket_asset.html.haml +9 -0
  20. data/app/views/admin/assets/_errors.html.haml +3 -0
  21. data/app/views/admin/assets/_form.html.haml +20 -0
  22. data/app/views/admin/assets/_page_assets.html.haml +12 -0
  23. data/app/views/admin/assets/_search_results.html.haml +17 -0
  24. data/app/views/admin/assets/_show_bucket_link.html.haml +4 -0
  25. data/app/views/admin/assets/_upload_to_page.html.haml +16 -0
  26. data/app/views/admin/assets/edit.html.haml +53 -0
  27. data/app/views/admin/assets/index.html.haml +43 -0
  28. data/app/views/admin/assets/new.html.haml +27 -0
  29. data/app/views/admin/assets/remove.html.haml +21 -0
  30. data/app/views/admin/bucket/_iframe.html.haml +1 -0
  31. data/config/locales/en.yml +60 -0
  32. data/config/locales/nl.yml +60 -0
  33. data/config/locales/pl.yml +60 -0
  34. data/config/routes.rb +20 -0
  35. data/db/migrate/001_create_assets.rb +12 -0
  36. data/db/migrate/002_create_paperclip_attributes.rb +13 -0
  37. data/db/migrate/003_create_user_observer.rb +13 -0
  38. data/db/migrate/004_create_page_attachments.rb +19 -0
  39. data/db/migrate/005_rename_users.rb +13 -0
  40. data/db/migrate/006_add_default_configs.rb +29 -0
  41. data/db/migrate/007_add_default_content_types.rb +29 -0
  42. data/db/migrate/20090316132151_disable_file_types.rb +20 -0
  43. data/lib/assets_admin_ui.rb +38 -0
  44. data/lib/mime_type_ext.rb +7 -0
  45. data/lib/tasks/assets_extension_tasks.rake +123 -0
  46. data/lib/tasks/paperclip_tasks.rake +79 -0
  47. data/lib/url_additions.rb +10 -0
  48. data/paperclipped_extension.rb +56 -0
  49. data/psds/file_type_icons.psd +0 -0
  50. data/psds/file_type_icons_.psd +0 -0
  51. data/public/images/assets/_page_assets.html.haml +26 -0
  52. data/public/images/assets/add-to-bucket.png +0 -0
  53. data/public/images/assets/add.png +0 -0
  54. data/public/images/assets/audio_icon.png +0 -0
  55. data/public/images/assets/audio_thumbnail.png +0 -0
  56. data/public/images/assets/delete.png +0 -0
  57. data/public/images/assets/doc_icon.png +0 -0
  58. data/public/images/assets/doc_thumbnail.png +0 -0
  59. data/public/images/assets/edit.png +0 -0
  60. data/public/images/assets/movie_icon.png +0 -0
  61. data/public/images/assets/movie_thumbnail.png +0 -0
  62. data/public/images/assets/new-asset.png +0 -0
  63. data/public/images/assets/page_edit.png +0 -0
  64. data/public/images/assets/pdf_icon.png +0 -0
  65. data/public/images/assets/pdf_thumbnail.png +0 -0
  66. data/public/images/assets/reorder_assets.png +0 -0
  67. data/public/javascripts/admin/assets.js +173 -0
  68. data/public/stylesheets/admin/assets.css +163 -0
  69. data/spec/controllers/admin/assets_controller_spec.rb +10 -0
  70. data/spec/models/asset_spec.rb +68 -0
  71. data/spec/spec.opts +6 -0
  72. data/spec/spec_helper.rb +27 -0
  73. data/vendor/plugins/acts_as_list/README +23 -0
  74. data/vendor/plugins/acts_as_list/init.rb +3 -0
  75. data/vendor/plugins/acts_as_list/lib/active_record/acts/list.rb +256 -0
  76. data/vendor/plugins/acts_as_list/test/list_test.rb +332 -0
  77. data/vendor/plugins/paperclip/LICENSE +26 -0
  78. data/vendor/plugins/paperclip/README.rdoc +179 -0
  79. data/vendor/plugins/paperclip/Rakefile +76 -0
  80. data/vendor/plugins/paperclip/cucumber/paperclip_steps.rb +6 -0
  81. data/vendor/plugins/paperclip/generators/paperclip/USAGE +5 -0
  82. data/vendor/plugins/paperclip/generators/paperclip/paperclip_generator.rb +27 -0
  83. data/vendor/plugins/paperclip/generators/paperclip/templates/paperclip_migration.rb.erb +19 -0
  84. data/vendor/plugins/paperclip/init.rb +1 -0
  85. data/vendor/plugins/paperclip/lib/generators/paperclip/USAGE +8 -0
  86. data/vendor/plugins/paperclip/lib/generators/paperclip/paperclip_generator.rb +31 -0
  87. data/vendor/plugins/paperclip/lib/generators/paperclip/templates/paperclip_migration.rb.erb +19 -0
  88. data/vendor/plugins/paperclip/lib/paperclip.rb +397 -0
  89. data/vendor/plugins/paperclip/lib/paperclip/attachment.rb +326 -0
  90. data/vendor/plugins/paperclip/lib/paperclip/callback_compatability.rb +61 -0
  91. data/vendor/plugins/paperclip/lib/paperclip/geometry.rb +115 -0
  92. data/vendor/plugins/paperclip/lib/paperclip/interpolations.rb +108 -0
  93. data/vendor/plugins/paperclip/lib/paperclip/iostream.rb +59 -0
  94. data/vendor/plugins/paperclip/lib/paperclip/matchers.rb +33 -0
  95. data/vendor/plugins/paperclip/lib/paperclip/matchers/have_attached_file_matcher.rb +57 -0
  96. data/vendor/plugins/paperclip/lib/paperclip/matchers/validate_attachment_content_type_matcher.rb +74 -0
  97. data/vendor/plugins/paperclip/lib/paperclip/matchers/validate_attachment_presence_matcher.rb +54 -0
  98. data/vendor/plugins/paperclip/lib/paperclip/matchers/validate_attachment_size_matcher.rb +95 -0
  99. data/vendor/plugins/paperclip/lib/paperclip/processor.rb +49 -0
  100. data/vendor/plugins/paperclip/lib/paperclip/railtie.rb +24 -0
  101. data/vendor/plugins/paperclip/lib/paperclip/storage.rb +247 -0
  102. data/vendor/plugins/paperclip/lib/paperclip/style.rb +90 -0
  103. data/vendor/plugins/paperclip/lib/paperclip/thumbnail.rb +78 -0
  104. data/vendor/plugins/paperclip/lib/paperclip/upfile.rb +52 -0
  105. data/vendor/plugins/paperclip/lib/paperclip/version.rb +3 -0
  106. data/vendor/plugins/paperclip/lib/tasks/paperclip.rake +79 -0
  107. data/vendor/plugins/paperclip/paperclip.gemspec +34 -0
  108. data/vendor/plugins/paperclip/rails/init.rb +2 -0
  109. data/vendor/plugins/paperclip/shoulda_macros/paperclip.rb +119 -0
  110. data/vendor/plugins/paperclip/test/.gitignore +1 -0
  111. data/vendor/plugins/paperclip/test/attachment_test.rb +758 -0
  112. data/vendor/plugins/paperclip/test/database.yml +4 -0
  113. data/vendor/plugins/paperclip/test/fixtures/12k.png +0 -0
  114. data/vendor/plugins/paperclip/test/fixtures/50x50.png +0 -0
  115. data/vendor/plugins/paperclip/test/fixtures/5k.png +0 -0
  116. data/vendor/plugins/paperclip/test/fixtures/bad.png +1 -0
  117. data/vendor/plugins/paperclip/test/fixtures/s3.yml +8 -0
  118. data/vendor/plugins/paperclip/test/fixtures/text.txt +0 -0
  119. data/vendor/plugins/paperclip/test/fixtures/twopage.pdf +0 -0
  120. data/vendor/plugins/paperclip/test/geometry_test.rb +177 -0
  121. data/vendor/plugins/paperclip/test/helper.rb +148 -0
  122. data/vendor/plugins/paperclip/test/integration_test.rb +483 -0
  123. data/vendor/plugins/paperclip/test/interpolations_test.rb +124 -0
  124. data/vendor/plugins/paperclip/test/iostream_test.rb +78 -0
  125. data/vendor/plugins/paperclip/test/matchers/have_attached_file_matcher_test.rb +24 -0
  126. data/vendor/plugins/paperclip/test/matchers/validate_attachment_content_type_matcher_test.rb +37 -0
  127. data/vendor/plugins/paperclip/test/matchers/validate_attachment_presence_matcher_test.rb +26 -0
  128. data/vendor/plugins/paperclip/test/matchers/validate_attachment_size_matcher_test.rb +51 -0
  129. data/vendor/plugins/paperclip/test/paperclip_test.rb +317 -0
  130. data/vendor/plugins/paperclip/test/processor_test.rb +10 -0
  131. data/vendor/plugins/paperclip/test/storage_test.rb +343 -0
  132. data/vendor/plugins/paperclip/test/style_test.rb +141 -0
  133. data/vendor/plugins/paperclip/test/thumbnail_test.rb +227 -0
  134. data/vendor/plugins/paperclip/test/upfile_test.rb +36 -0
  135. data/vendor/plugins/responds_to_parent/MIT-LICENSE +20 -0
  136. data/vendor/plugins/responds_to_parent/README +42 -0
  137. data/vendor/plugins/responds_to_parent/Rakefile +22 -0
  138. data/vendor/plugins/responds_to_parent/init.rb +2 -0
  139. data/vendor/plugins/responds_to_parent/lib/parent_selector_assertion.rb +144 -0
  140. data/vendor/plugins/responds_to_parent/lib/responds_to_parent.rb +46 -0
  141. data/vendor/plugins/responds_to_parent/test/assert_select_parent_test.rb +318 -0
  142. data/vendor/plugins/responds_to_parent/test/responds_to_parent_test.rb +115 -0
  143. metadata +226 -0
@@ -0,0 +1,4 @@
1
+ test:
2
+ adapter: sqlite3
3
+ database: ":memory:"
4
+
@@ -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'] %>
@@ -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,148 @@
1
+ require 'rubygems'
2
+ require 'tempfile'
3
+ require 'test/unit'
4
+
5
+ require 'shoulda'
6
+ require 'mocha'
7
+
8
+ case ENV['RAILS_VERSION']
9
+ when '2.1' then
10
+ gem 'activerecord', '~>2.1.0'
11
+ gem 'activesupport', '~>2.1.0'
12
+ gem 'actionpack', '~>2.1.0'
13
+ when '3.0' then
14
+ gem 'activerecord', '~>3.0.0'
15
+ gem 'activesupport', '~>3.0.0'
16
+ gem 'actionpack', '~>3.0.0'
17
+ else
18
+ gem 'activerecord', '~>2.3.0'
19
+ gem 'activesupport', '~>2.3.0'
20
+ gem 'actionpack', '~>2.3.0'
21
+ end
22
+
23
+ require 'active_record'
24
+ require 'active_record/version'
25
+ require 'active_support'
26
+ require 'action_pack'
27
+
28
+ puts "Testing against version #{ActiveRecord::VERSION::STRING}"
29
+
30
+ begin
31
+ require 'ruby-debug'
32
+ rescue LoadError => e
33
+ puts "debugger disabled"
34
+ end
35
+
36
+ ROOT = File.join(File.dirname(__FILE__), '..')
37
+
38
+ def silence_warnings
39
+ old_verbose, $VERBOSE = $VERBOSE, nil
40
+ yield
41
+ ensure
42
+ $VERBOSE = old_verbose
43
+ end
44
+
45
+ class Test::Unit::TestCase
46
+ def setup
47
+ silence_warnings do
48
+ Object.const_set(:Rails, stub('Rails', :root => ROOT, :env => 'test'))
49
+ end
50
+ end
51
+ end
52
+
53
+ $LOAD_PATH << File.join(ROOT, 'lib')
54
+ $LOAD_PATH << File.join(ROOT, 'lib', 'paperclip')
55
+
56
+ require File.join(ROOT, 'lib', 'paperclip.rb')
57
+
58
+ require 'shoulda_macros/paperclip'
59
+
60
+ FIXTURES_DIR = File.join(File.dirname(__FILE__), "fixtures")
61
+ config = YAML::load(IO.read(File.dirname(__FILE__) + '/database.yml'))
62
+ ActiveRecord::Base.logger = Logger.new(File.dirname(__FILE__) + "/debug.log")
63
+ ActiveRecord::Base.establish_connection(config['test'])
64
+
65
+ def reset_class class_name
66
+ ActiveRecord::Base.send(:include, Paperclip)
67
+ Object.send(:remove_const, class_name) rescue nil
68
+ klass = Object.const_set(class_name, Class.new(ActiveRecord::Base))
69
+ klass.class_eval{ include Paperclip }
70
+ klass
71
+ end
72
+
73
+ def reset_table table_name, &block
74
+ block ||= lambda { |table| true }
75
+ ActiveRecord::Base.connection.create_table :dummies, {:force => true}, &block
76
+ end
77
+
78
+ def modify_table table_name, &block
79
+ ActiveRecord::Base.connection.change_table :dummies, &block
80
+ end
81
+
82
+ def rebuild_model options = {}
83
+ ActiveRecord::Base.connection.create_table :dummies, :force => true do |table|
84
+ table.column :other, :string
85
+ table.column :avatar_file_name, :string
86
+ table.column :avatar_content_type, :string
87
+ table.column :avatar_file_size, :integer
88
+ table.column :avatar_updated_at, :datetime
89
+ end
90
+ rebuild_class options
91
+ end
92
+
93
+ def rebuild_class options = {}
94
+ ActiveRecord::Base.send(:include, Paperclip)
95
+ Object.send(:remove_const, "Dummy") rescue nil
96
+ Object.const_set("Dummy", Class.new(ActiveRecord::Base))
97
+ Dummy.class_eval do
98
+ include Paperclip
99
+ has_attached_file :avatar, options
100
+ end
101
+ end
102
+
103
+ class FakeModel
104
+ attr_accessor :avatar_file_name,
105
+ :avatar_file_size,
106
+ :avatar_last_updated,
107
+ :avatar_content_type,
108
+ :id
109
+
110
+ def errors
111
+ @errors ||= []
112
+ end
113
+
114
+ def run_paperclip_callbacks name, *args
115
+ end
116
+
117
+ end
118
+
119
+ def attachment options
120
+ Paperclip::Attachment.new(:avatar, FakeModel.new, options)
121
+ end
122
+
123
+ def silence_warnings
124
+ old_verbose, $VERBOSE = $VERBOSE, nil
125
+ yield
126
+ ensure
127
+ $VERBOSE = old_verbose
128
+ end
129
+
130
+ def should_accept_dummy_class
131
+ should "accept the class" do
132
+ assert_accepts @matcher, @dummy_class
133
+ end
134
+
135
+ should "accept an instance of that class" do
136
+ assert_accepts @matcher, @dummy_class.new
137
+ end
138
+ end
139
+
140
+ def should_reject_dummy_class
141
+ should "reject the class" do
142
+ assert_rejects @matcher, @dummy_class
143
+ end
144
+
145
+ should "reject an instance of that class" do
146
+ assert_rejects @matcher, @dummy_class.new
147
+ end
148
+ 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
+