fog-dragonfly 0.8.1
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of fog-dragonfly might be problematic. Click here for more details.
- data/.specopts +2 -0
- data/.yardopts +23 -0
- data/Gemfile +23 -0
- data/Gemfile.rails.2.3.5 +14 -0
- data/History.md +266 -0
- data/LICENSE +20 -0
- data/README.md +88 -0
- data/Rakefile +92 -0
- data/VERSION +1 -0
- data/config.ru +13 -0
- data/docs.watchr +1 -0
- data/dragonfly.gemspec +293 -0
- data/extra_docs/Analysers.md +108 -0
- data/extra_docs/Caching.md +23 -0
- data/extra_docs/Configuration.md +138 -0
- data/extra_docs/DataStorage.md +136 -0
- data/extra_docs/Encoding.md +96 -0
- data/extra_docs/GeneralUsage.md +121 -0
- data/extra_docs/Generators.md +102 -0
- data/extra_docs/Heroku.md +50 -0
- data/extra_docs/Index.md +36 -0
- data/extra_docs/MimeTypes.md +40 -0
- data/extra_docs/Models.md +266 -0
- data/extra_docs/Mongo.md +45 -0
- data/extra_docs/Processing.md +130 -0
- data/extra_docs/Rack.md +52 -0
- data/extra_docs/Rails2.md +55 -0
- data/extra_docs/Rails3.md +62 -0
- data/extra_docs/Sinatra.md +25 -0
- data/extra_docs/URLs.md +169 -0
- data/features/3.0.3.feature +8 -0
- data/features/images.feature +47 -0
- data/features/no_processing.feature +14 -0
- data/features/rails_2.3.5.feature +7 -0
- data/features/steps/common_steps.rb +8 -0
- data/features/steps/dragonfly_steps.rb +66 -0
- data/features/steps/rails_steps.rb +39 -0
- data/features/support/env.rb +40 -0
- data/fixtures/files/app/models/album.rb +3 -0
- data/fixtures/files/app/views/albums/new.html.erb +4 -0
- data/fixtures/files/app/views/albums/show.html.erb +4 -0
- data/fixtures/files/config/initializers/dragonfly.rb +4 -0
- data/fixtures/files/features/manage_album_images.feature +12 -0
- data/fixtures/files/features/step_definitions/image_steps.rb +15 -0
- data/fixtures/files/features/support/paths.rb +15 -0
- data/fixtures/files/features/text_images.feature +7 -0
- data/fixtures/rails_2.3.5/template.rb +10 -0
- data/fixtures/rails_3.0.3/template.rb +20 -0
- data/irbrc.rb +17 -0
- data/lib/dragonfly.rb +45 -0
- data/lib/dragonfly/active_model_extensions.rb +13 -0
- data/lib/dragonfly/active_model_extensions/attachment.rb +169 -0
- data/lib/dragonfly/active_model_extensions/class_methods.rb +45 -0
- data/lib/dragonfly/active_model_extensions/instance_methods.rb +28 -0
- data/lib/dragonfly/active_model_extensions/validations.rb +37 -0
- data/lib/dragonfly/analyser.rb +59 -0
- data/lib/dragonfly/analysis/file_command_analyser.rb +32 -0
- data/lib/dragonfly/analysis/image_magick_analyser.rb +47 -0
- data/lib/dragonfly/analysis/r_magick_analyser.rb +63 -0
- data/lib/dragonfly/app.rb +182 -0
- data/lib/dragonfly/config/heroku.rb +19 -0
- data/lib/dragonfly/config/image_magick.rb +41 -0
- data/lib/dragonfly/config/r_magick.rb +46 -0
- data/lib/dragonfly/config/rails.rb +17 -0
- data/lib/dragonfly/configurable.rb +119 -0
- data/lib/dragonfly/core_ext/object.rb +8 -0
- data/lib/dragonfly/core_ext/string.rb +9 -0
- data/lib/dragonfly/core_ext/symbol.rb +9 -0
- data/lib/dragonfly/data_storage.rb +9 -0
- data/lib/dragonfly/data_storage/file_data_store.rb +114 -0
- data/lib/dragonfly/data_storage/mongo_data_store.rb +82 -0
- data/lib/dragonfly/data_storage/s3data_store.rb +115 -0
- data/lib/dragonfly/encoder.rb +13 -0
- data/lib/dragonfly/encoding/image_magick_encoder.rb +57 -0
- data/lib/dragonfly/encoding/r_magick_encoder.rb +61 -0
- data/lib/dragonfly/function_manager.rb +69 -0
- data/lib/dragonfly/generation/hash_with_css_style_keys.rb +23 -0
- data/lib/dragonfly/generation/image_magick_generator.rb +140 -0
- data/lib/dragonfly/generation/r_magick_generator.rb +155 -0
- data/lib/dragonfly/generator.rb +9 -0
- data/lib/dragonfly/image_magick_utils.rb +81 -0
- data/lib/dragonfly/job.rb +371 -0
- data/lib/dragonfly/job_builder.rb +39 -0
- data/lib/dragonfly/job_definitions.rb +26 -0
- data/lib/dragonfly/job_endpoint.rb +15 -0
- data/lib/dragonfly/loggable.rb +28 -0
- data/lib/dragonfly/middleware.rb +34 -0
- data/lib/dragonfly/processing/image_magick_processor.rb +99 -0
- data/lib/dragonfly/processing/r_magick_processor.rb +126 -0
- data/lib/dragonfly/processor.rb +9 -0
- data/lib/dragonfly/r_magick_utils.rb +48 -0
- data/lib/dragonfly/rails/images.rb +22 -0
- data/lib/dragonfly/response.rb +82 -0
- data/lib/dragonfly/routed_endpoint.rb +40 -0
- data/lib/dragonfly/serializer.rb +32 -0
- data/lib/dragonfly/simple_cache.rb +23 -0
- data/lib/dragonfly/simple_endpoint.rb +63 -0
- data/lib/dragonfly/temp_object.rb +220 -0
- data/samples/beach.png +0 -0
- data/samples/egg.png +0 -0
- data/samples/round.gif +0 -0
- data/samples/sample.docx +0 -0
- data/samples/taj.jpg +0 -0
- data/spec/argument_matchers.rb +19 -0
- data/spec/dragonfly/active_model_extensions/active_model_setup.rb +97 -0
- data/spec/dragonfly/active_model_extensions/active_record_setup.rb +85 -0
- data/spec/dragonfly/active_model_extensions/model_spec.rb +723 -0
- data/spec/dragonfly/active_model_extensions/spec_helper.rb +11 -0
- data/spec/dragonfly/analyser_spec.rb +123 -0
- data/spec/dragonfly/analysis/file_command_analyser_spec.rb +57 -0
- data/spec/dragonfly/analysis/image_magick_analyser_spec.rb +15 -0
- data/spec/dragonfly/analysis/r_magick_analyser_spec.rb +27 -0
- data/spec/dragonfly/analysis/shared_analyser_spec.rb +51 -0
- data/spec/dragonfly/app_spec.rb +280 -0
- data/spec/dragonfly/config/r_magick_spec.rb +25 -0
- data/spec/dragonfly/configurable_spec.rb +220 -0
- data/spec/dragonfly/core_ext/string_spec.rb +17 -0
- data/spec/dragonfly/core_ext/symbol_spec.rb +17 -0
- data/spec/dragonfly/data_storage/data_store_spec.rb +76 -0
- data/spec/dragonfly/data_storage/file_data_store_spec.rb +169 -0
- data/spec/dragonfly/data_storage/mongo_data_store_spec.rb +38 -0
- data/spec/dragonfly/data_storage/s3_data_store_spec.rb +94 -0
- data/spec/dragonfly/deprecation_spec.rb +20 -0
- data/spec/dragonfly/encoding/image_magick_encoder_spec.rb +41 -0
- data/spec/dragonfly/encoding/r_magick_encoder_spec.rb +37 -0
- data/spec/dragonfly/function_manager_spec.rb +154 -0
- data/spec/dragonfly/generation/hash_with_css_style_keys_spec.rb +24 -0
- data/spec/dragonfly/generation/image_magick_generator_spec.rb +12 -0
- data/spec/dragonfly/generation/r_magick_generator_spec.rb +24 -0
- data/spec/dragonfly/generation/shared_generator_spec.rb +91 -0
- data/spec/dragonfly/image_magick_utils_spec.rb +16 -0
- data/spec/dragonfly/job_builder_spec.rb +37 -0
- data/spec/dragonfly/job_definitions_spec.rb +35 -0
- data/spec/dragonfly/job_endpoint_spec.rb +120 -0
- data/spec/dragonfly/job_spec.rb +773 -0
- data/spec/dragonfly/loggable_spec.rb +80 -0
- data/spec/dragonfly/middleware_spec.rb +68 -0
- data/spec/dragonfly/processing/image_magick_processor_spec.rb +29 -0
- data/spec/dragonfly/processing/r_magick_processor_spec.rb +26 -0
- data/spec/dragonfly/processing/shared_processing_spec.rb +215 -0
- data/spec/dragonfly/routed_endpoint_spec.rb +48 -0
- data/spec/dragonfly/serializer_spec.rb +61 -0
- data/spec/dragonfly/simple_cache_spec.rb +27 -0
- data/spec/dragonfly/simple_endpoint_spec.rb +89 -0
- data/spec/dragonfly/temp_object_spec.rb +352 -0
- data/spec/image_matchers.rb +47 -0
- data/spec/simple_matchers.rb +44 -0
- data/spec/spec_helper.rb +58 -0
- data/yard/handlers/configurable_attr_handler.rb +38 -0
- data/yard/setup.rb +15 -0
- data/yard/templates/default/fulldoc/html/css/common.css +107 -0
- data/yard/templates/default/layout/html/layout.erb +87 -0
- data/yard/templates/default/module/html/configuration_summary.erb +31 -0
- data/yard/templates/default/module/setup.rb +17 -0
- metadata +550 -0
@@ -0,0 +1,39 @@
|
|
1
|
+
module Dragonfly
|
2
|
+
class JobBuilder
|
3
|
+
|
4
|
+
def initialize(&block)
|
5
|
+
@block = block
|
6
|
+
end
|
7
|
+
|
8
|
+
def build(job, *args)
|
9
|
+
evaluate_block(job, false, *args)
|
10
|
+
end
|
11
|
+
|
12
|
+
def build!(job, *args)
|
13
|
+
evaluate_block(job, true, *args)
|
14
|
+
end
|
15
|
+
|
16
|
+
Job.step_names.each do |step|
|
17
|
+
|
18
|
+
# fetch, process, etc.
|
19
|
+
define_method step do |*args|
|
20
|
+
if @perform_with_bangs
|
21
|
+
@job.send("#{step}!", *args)
|
22
|
+
else
|
23
|
+
@job = @job.send(step, *args)
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
end
|
28
|
+
|
29
|
+
private
|
30
|
+
|
31
|
+
def evaluate_block(job, perform_with_bangs, *args)
|
32
|
+
@job = job
|
33
|
+
@perform_with_bangs = perform_with_bangs
|
34
|
+
instance_exec(*args, &@block)
|
35
|
+
@job
|
36
|
+
end
|
37
|
+
|
38
|
+
end
|
39
|
+
end
|
@@ -0,0 +1,26 @@
|
|
1
|
+
module Dragonfly
|
2
|
+
class JobDefinitions < Module
|
3
|
+
|
4
|
+
def initialize
|
5
|
+
@job_definitions = {}
|
6
|
+
end
|
7
|
+
|
8
|
+
def add(name, &definition_proc)
|
9
|
+
job_definitions[name] = JobBuilder.new(&definition_proc)
|
10
|
+
jd = job_definitions # Needed because we're about to change 'self'
|
11
|
+
|
12
|
+
define_method name do |*args|
|
13
|
+
jd[name].build(self, *args)
|
14
|
+
end
|
15
|
+
|
16
|
+
define_method "#{name}!" do |*args|
|
17
|
+
jd[name].build!(self, *args)
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
private
|
22
|
+
|
23
|
+
attr_reader :job_definitions
|
24
|
+
|
25
|
+
end
|
26
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
require 'logger'
|
2
|
+
|
3
|
+
module Dragonfly
|
4
|
+
module Loggable
|
5
|
+
|
6
|
+
def log
|
7
|
+
case @log_object
|
8
|
+
when nil
|
9
|
+
@log_object = Logger.new($stdout)
|
10
|
+
when Proc
|
11
|
+
@log_object[]
|
12
|
+
when Logger
|
13
|
+
@log_object
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
def log=(object)
|
18
|
+
@log_object = object
|
19
|
+
end
|
20
|
+
|
21
|
+
attr_reader :log_object
|
22
|
+
|
23
|
+
def use_same_log_as(object)
|
24
|
+
self.log = proc{ object.log }
|
25
|
+
end
|
26
|
+
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
module Dragonfly
|
2
|
+
class Middleware
|
3
|
+
|
4
|
+
def initialize(app, dragonfly_app_name, path_prefix)
|
5
|
+
@app = app
|
6
|
+
@endpoint = Rack::Builder.new {
|
7
|
+
map path_prefix do
|
8
|
+
run Dragonfly[dragonfly_app_name]
|
9
|
+
end
|
10
|
+
}.to_app
|
11
|
+
end
|
12
|
+
|
13
|
+
def call(env)
|
14
|
+
response = @endpoint.call(env)
|
15
|
+
if route_not_found?(response)
|
16
|
+
@app.call(env)
|
17
|
+
else
|
18
|
+
response
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
private
|
23
|
+
|
24
|
+
def route_not_found?(response)
|
25
|
+
response[1]['X-Cascade'] == 'pass' ||
|
26
|
+
(rack_version_doesnt_support_x_cascade? && response[0] == 404)
|
27
|
+
end
|
28
|
+
|
29
|
+
def rack_version_doesnt_support_x_cascade?
|
30
|
+
Rack.version < '1.1'
|
31
|
+
end
|
32
|
+
|
33
|
+
end
|
34
|
+
end
|
@@ -0,0 +1,99 @@
|
|
1
|
+
module Dragonfly
|
2
|
+
module Processing
|
3
|
+
class ImageMagickProcessor
|
4
|
+
|
5
|
+
GRAVITIES = {
|
6
|
+
'nw' => 'NorthWest',
|
7
|
+
'n' => 'North',
|
8
|
+
'ne' => 'NorthEast',
|
9
|
+
'w' => 'West',
|
10
|
+
'c' => 'Center',
|
11
|
+
'e' => 'East',
|
12
|
+
'sw' => 'SouthWest',
|
13
|
+
's' => 'South',
|
14
|
+
'se' => 'SouthEast'
|
15
|
+
}
|
16
|
+
|
17
|
+
# Geometry string patterns
|
18
|
+
RESIZE_GEOMETRY = /^\d*x\d*[><%^!]?$|^\d+@$/ # e.g. '300x200!'
|
19
|
+
CROPPED_RESIZE_GEOMETRY = /^(\d+)x(\d+)#(\w{1,2})?$/ # e.g. '20x50#ne'
|
20
|
+
CROP_GEOMETRY = /^(\d+)x(\d+)([+-]\d+)?([+-]\d+)?(\w{1,2})?$/ # e.g. '30x30+10+10'
|
21
|
+
THUMB_GEOMETRY = Regexp.union RESIZE_GEOMETRY, CROPPED_RESIZE_GEOMETRY, CROP_GEOMETRY
|
22
|
+
|
23
|
+
include ImageMagickUtils
|
24
|
+
|
25
|
+
def resize(temp_object, geometry)
|
26
|
+
convert(temp_object, "-resize '#{geometry}'")
|
27
|
+
end
|
28
|
+
|
29
|
+
def crop(temp_object, opts={})
|
30
|
+
width = opts[:width]
|
31
|
+
height = opts[:height]
|
32
|
+
gravity = GRAVITIES[opts[:gravity]]
|
33
|
+
x = "#{opts[:x] || 0}"
|
34
|
+
x = '+' + x unless x[/^[+-]/]
|
35
|
+
y = "#{opts[:y] || 0}"
|
36
|
+
y = '+' + y unless y[/^[+-]/]
|
37
|
+
|
38
|
+
convert(temp_object, "-crop #{width}x#{height}#{x}#{y}#{" -gravity #{gravity}" if gravity}")
|
39
|
+
end
|
40
|
+
|
41
|
+
def flip(temp_object)
|
42
|
+
convert(temp_object, "-flip")
|
43
|
+
end
|
44
|
+
|
45
|
+
def flop(temp_object)
|
46
|
+
convert(temp_object, "-flop")
|
47
|
+
end
|
48
|
+
|
49
|
+
def greyscale(temp_object)
|
50
|
+
convert(temp_object, "-colorspace Gray")
|
51
|
+
end
|
52
|
+
alias grayscale greyscale
|
53
|
+
|
54
|
+
def resize_and_crop(temp_object, opts={})
|
55
|
+
attrs = identify(temp_object)
|
56
|
+
current_width = attrs[:width].to_i
|
57
|
+
current_height = attrs[:height].to_i
|
58
|
+
|
59
|
+
width = opts[:width] ? opts[:width].to_i : current_width
|
60
|
+
height = opts[:height] ? opts[:height].to_i : current_height
|
61
|
+
gravity = opts[:gravity] || 'c'
|
62
|
+
|
63
|
+
if width != current_width || height != current_height
|
64
|
+
scale = [width.to_f / current_width, height.to_f / current_height].max
|
65
|
+
temp_object = TempObject.new(resize(temp_object, "#{(scale * current_width).ceil}x#{(scale * current_height).ceil}"))
|
66
|
+
end
|
67
|
+
|
68
|
+
crop(temp_object, :width => width, :height => height, :gravity => gravity)
|
69
|
+
end
|
70
|
+
|
71
|
+
def rotate(temp_object, amount, opts={})
|
72
|
+
convert(temp_object, "-rotate '#{amount}#{opts[:qualifier]}'")
|
73
|
+
end
|
74
|
+
|
75
|
+
def thumb(temp_object, geometry)
|
76
|
+
case geometry
|
77
|
+
when RESIZE_GEOMETRY
|
78
|
+
resize(temp_object, geometry)
|
79
|
+
when CROPPED_RESIZE_GEOMETRY
|
80
|
+
resize_and_crop(temp_object, :width => $1, :height => $2, :gravity => $3)
|
81
|
+
when CROP_GEOMETRY
|
82
|
+
crop(temp_object,
|
83
|
+
:width => $1,
|
84
|
+
:height => $2,
|
85
|
+
:x => $3,
|
86
|
+
:y => $4,
|
87
|
+
:gravity => $5
|
88
|
+
)
|
89
|
+
else raise ArgumentError, "Didn't recognise the geometry string #{geometry}"
|
90
|
+
end
|
91
|
+
end
|
92
|
+
|
93
|
+
def convert(temp_object, args='', format=nil)
|
94
|
+
format ? [super, {:format => format.to_sym}] : super
|
95
|
+
end
|
96
|
+
|
97
|
+
end
|
98
|
+
end
|
99
|
+
end
|
@@ -0,0 +1,126 @@
|
|
1
|
+
require 'RMagick'
|
2
|
+
|
3
|
+
module Dragonfly
|
4
|
+
module Processing
|
5
|
+
class RMagickProcessor
|
6
|
+
|
7
|
+
GRAVITIES = {
|
8
|
+
'nw' => Magick::NorthWestGravity,
|
9
|
+
'n' => Magick::NorthGravity,
|
10
|
+
'ne' => Magick::NorthEastGravity,
|
11
|
+
'w' => Magick::WestGravity,
|
12
|
+
'c' => Magick::CenterGravity,
|
13
|
+
'e' => Magick::EastGravity,
|
14
|
+
'sw' => Magick::SouthWestGravity,
|
15
|
+
's' => Magick::SouthGravity,
|
16
|
+
'se' => Magick::SouthEastGravity
|
17
|
+
}
|
18
|
+
|
19
|
+
# Geometry string patterns
|
20
|
+
RESIZE_GEOMETRY = /^\d*x\d*[><%^!]?$|^\d+@$/ # e.g. '300x200!'
|
21
|
+
CROPPED_RESIZE_GEOMETRY = /^(\d+)x(\d+)#(\w{1,2})?$/ # e.g. '20x50#ne'
|
22
|
+
CROP_GEOMETRY = /^(\d+)x(\d+)([+-]\d+)?([+-]\d+)?(\w{1,2})?$/ # e.g. '30x30+10+10'
|
23
|
+
THUMB_GEOMETRY = Regexp.union RESIZE_GEOMETRY, CROPPED_RESIZE_GEOMETRY, CROP_GEOMETRY
|
24
|
+
|
25
|
+
include RMagickUtils
|
26
|
+
include Configurable
|
27
|
+
|
28
|
+
configurable_attr :use_filesystem, true
|
29
|
+
|
30
|
+
def crop(temp_object, opts={})
|
31
|
+
x = opts[:x].to_i
|
32
|
+
y = opts[:y].to_i
|
33
|
+
gravity = GRAVITIES[opts[:gravity]] || Magick::ForgetGravity
|
34
|
+
width = opts[:width].to_i
|
35
|
+
height = opts[:height].to_i
|
36
|
+
|
37
|
+
rmagick_image(temp_object) do |image|
|
38
|
+
# RMagick throws an error if the cropping area is bigger than the image,
|
39
|
+
# when the gravity is something other than nw
|
40
|
+
width = image.columns - x if x + width > image.columns
|
41
|
+
height = image.rows - y if y + height > image.rows
|
42
|
+
image.crop(gravity, x, y, width, height)
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
def flip(temp_object)
|
47
|
+
rmagick_image(temp_object) do |image|
|
48
|
+
image.flip!
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
def flop(temp_object)
|
53
|
+
rmagick_image(temp_object) do |image|
|
54
|
+
image.flop!
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
def greyscale(temp_object, opts={})
|
59
|
+
depth = opts[:depth] || 256
|
60
|
+
rmagick_image(temp_object) do |image|
|
61
|
+
image.quantize(depth, Magick::GRAYColorspace)
|
62
|
+
end
|
63
|
+
end
|
64
|
+
alias grayscale greyscale
|
65
|
+
|
66
|
+
def resize(temp_object, geometry)
|
67
|
+
rmagick_image(temp_object) do |image|
|
68
|
+
image.change_geometry!(geometry) do |cols, rows, img|
|
69
|
+
img.resize!(cols, rows)
|
70
|
+
end
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
def resize_and_crop(temp_object, opts={})
|
75
|
+
rmagick_image(temp_object) do |image|
|
76
|
+
|
77
|
+
width = opts[:width] ? opts[:width].to_i : image.columns
|
78
|
+
height = opts[:height] ? opts[:height].to_i : image.rows
|
79
|
+
gravity = GRAVITIES[opts[:gravity]] || Magick::CenterGravity
|
80
|
+
|
81
|
+
image.crop_resized(width, height, gravity)
|
82
|
+
end
|
83
|
+
end
|
84
|
+
|
85
|
+
def rotate(temp_object, amount, opts={})
|
86
|
+
args = [amount.to_f]
|
87
|
+
args << opts[:qualifier] if opts[:qualifier]
|
88
|
+
rmagick_image(temp_object) do |image|
|
89
|
+
image.background_color = opts[:background_colour] if opts[:background_colour]
|
90
|
+
image.background_color = opts[:background_color] if opts[:background_color]
|
91
|
+
image.rotate(*args) || temp_object
|
92
|
+
end
|
93
|
+
end
|
94
|
+
|
95
|
+
def thumb(temp_object, geometry)
|
96
|
+
case geometry
|
97
|
+
when RESIZE_GEOMETRY
|
98
|
+
resize(temp_object, geometry)
|
99
|
+
when CROPPED_RESIZE_GEOMETRY
|
100
|
+
resize_and_crop(temp_object, :width => $1, :height => $2, :gravity => $3)
|
101
|
+
when CROP_GEOMETRY
|
102
|
+
crop(temp_object,
|
103
|
+
:width => $1,
|
104
|
+
:height => $2,
|
105
|
+
:x => $3,
|
106
|
+
:y => $4,
|
107
|
+
:gravity => $5
|
108
|
+
)
|
109
|
+
else raise ArgumentError, "Didn't recognise the geometry string #{geometry}"
|
110
|
+
end
|
111
|
+
end
|
112
|
+
|
113
|
+
def vignette(temp_object, opts={})
|
114
|
+
x = opts[:x].to_f || temp_object.width * 0.1
|
115
|
+
y = opts[:y].to_f || temp_object.height * 0.1
|
116
|
+
radius = opts[:radius].to_f || 0.0
|
117
|
+
sigma = opts[:sigma].to_f || 10.0
|
118
|
+
|
119
|
+
rmagick_image(temp_object) do |image|
|
120
|
+
image.vignette(x, y, radius, sigma)
|
121
|
+
end
|
122
|
+
end
|
123
|
+
|
124
|
+
end
|
125
|
+
end
|
126
|
+
end
|
@@ -0,0 +1,48 @@
|
|
1
|
+
require 'tempfile'
|
2
|
+
|
3
|
+
module Dragonfly
|
4
|
+
module RMagickUtils
|
5
|
+
|
6
|
+
include Loggable
|
7
|
+
|
8
|
+
private
|
9
|
+
|
10
|
+
# Requires the extended object to respond to 'use_filesystem'
|
11
|
+
def rmagick_image(temp_object, &block)
|
12
|
+
imagelist = use_filesystem ? Magick::Image.read(temp_object.path) : Magick::Image.from_blob(temp_object.data)
|
13
|
+
image = imagelist.first
|
14
|
+
result = block.call(image)
|
15
|
+
case result
|
16
|
+
when Magick::Image, Magick::ImageList
|
17
|
+
content = use_filesystem ? write_to_tempfile(result) : result.to_blob
|
18
|
+
result.destroy!
|
19
|
+
else
|
20
|
+
content = result
|
21
|
+
end
|
22
|
+
image.destroy!
|
23
|
+
content
|
24
|
+
rescue Magick::ImageMagickError => e
|
25
|
+
log.warn("Unable to handle content in #{self.class} - got:\n#{e}")
|
26
|
+
throw :unable_to_handle
|
27
|
+
end
|
28
|
+
|
29
|
+
def ping_rmagick_image(temp_object, &block)
|
30
|
+
imagelist = use_filesystem ? Magick::Image.ping(temp_object.path) : Magick::Image.from_blob(temp_object.data)
|
31
|
+
image = imagelist.first
|
32
|
+
result = block.call(image)
|
33
|
+
image.destroy!
|
34
|
+
result
|
35
|
+
rescue Magick::ImageMagickError => e
|
36
|
+
log.warn("Unable to handle content in #{self.class} - got:\n#{e}")
|
37
|
+
throw :unable_to_handle
|
38
|
+
end
|
39
|
+
|
40
|
+
def write_to_tempfile(rmagick_image)
|
41
|
+
tempfile = Tempfile.new('dragonfly')
|
42
|
+
tempfile.close
|
43
|
+
rmagick_image.write(tempfile.path)
|
44
|
+
tempfile
|
45
|
+
end
|
46
|
+
|
47
|
+
end
|
48
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
require 'dragonfly'
|
2
|
+
require 'rack/cache'
|
3
|
+
require 'uri'
|
4
|
+
|
5
|
+
### The dragonfly app ###
|
6
|
+
app = Dragonfly[:images]
|
7
|
+
app.configure_with(:rails)
|
8
|
+
app.configure_with(:imagemagick)
|
9
|
+
|
10
|
+
### Extend active record ###
|
11
|
+
app.define_macro(ActiveRecord::Base, :image_accessor)
|
12
|
+
|
13
|
+
### Insert the middleware ###
|
14
|
+
# Where the middleware is depends on the version of Rails
|
15
|
+
middleware = Rails.respond_to?(:application) ? Rails.application.middleware : ActionController::Dispatcher.middleware
|
16
|
+
|
17
|
+
middleware.insert_after 'Rack::Lock', 'Dragonfly::Middleware', :images, app.url_path_prefix
|
18
|
+
middleware.insert_before 'Dragonfly::Middleware', 'Rack::Cache', {
|
19
|
+
:verbose => true,
|
20
|
+
:metastore => URI.encode("file:#{Rails.root}/tmp/dragonfly/cache/meta"), # URI encoded because Windows
|
21
|
+
:entitystore => URI.encode("file:#{Rails.root}/tmp/dragonfly/cache/body") # has problems with spaces
|
22
|
+
}
|