carrierwave-base64 2.7.0 → 2.8.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: d46d2f9f96d23a40a168aa3a25f1d79475cac873e7b1e004153199aa12c11962
4
- data.tar.gz: 77d81babfb99831edae4fc362010c13639a957b0ff56377420744f56de5eb029
3
+ metadata.gz: c9da0e56ae6151c3068b895309bd70f9bfba959b28785b8fafde064bfecc9047
4
+ data.tar.gz: b723fcc95905b624fdc54394129bd9629b8bc1aa781061fc992f6f2ec3405d6f
5
5
  SHA512:
6
- metadata.gz: 8707ca4f1b52279e6c1f6a8eec3b576029fd1fd4641ecdf656e824afd697d7e04d94e54e9fc796df648f8626711d3f2a53cb3758f5cd04d8305f1bc20696114b
7
- data.tar.gz: 856803d525d0d2b614ae21688e424de5c9f98ecad8dd1d39feb9509bed63d7f2fa3b68a80377bd0fbdb790f1f4c39a000bba5629aed6cded3f39fd523862988b
6
+ metadata.gz: ccb2e7e949025a1eb37ba1bea875219ea269b561f6783f44203ba3085f9e2455913f9e01eb540e54165d0f4e5cc371cae4009dc367a2a278d9f65b6d26c01dc0
7
+ data.tar.gz: a429b36e280b49ac842bcf26ccf73c096cce0a3a67129ee97a3c92fd391a38ffb3486220c947a9a243e8d7db7a4997b81cfad9b36d0e3c974ad451d6f1b71286
@@ -4,3 +4,4 @@ Documentation:
4
4
  Metrics/BlockLength:
5
5
  Exclude:
6
6
  - "spec/**/*"
7
+ - "*.gemspec"
@@ -1,16 +1,17 @@
1
1
  language: ruby
2
2
 
3
3
  rvm:
4
- - 2.1.10
5
4
  - 2.2.10
6
- - 2.3.7
7
- - 2.4.4
8
- - 2.5.1
5
+ - 2.3.8
6
+ - 2.4.5
7
+ - 2.5.2
9
8
  - ruby-head
9
+ - jruby-head
10
10
 
11
11
  matrix:
12
12
  allow_failures:
13
13
  - rvm: ruby-head
14
+ - rvm: jruby-head
14
15
 
15
16
  gemfile:
16
17
  - Gemfile
@@ -1,5 +1,9 @@
1
1
  # carrierwave-base64 changelog
2
2
 
3
+ ## 2.8.0
4
+
5
+ - Mime Type is now detected from file contents, falling back to the type, specified by the user. (#73, @justisb)
6
+
3
7
  ## 2.7.0
4
8
 
5
9
  - Railtie is not required by default, so the gem can be used in non Rails app (@sleepingstu)
data/README.md CHANGED
@@ -1,8 +1,9 @@
1
1
  # Carrierwave::Base64
2
2
 
3
3
  [![Gem Version](https://badge.fury.io/rb/carrierwave-base64.svg)](http://badge.fury.io/rb/carrierwave-base64)
4
- [![Build Status](https://travis-ci.org/lebedev-yury/carrierwave-base64.svg?branch=master)](https://travis-ci.org/lebedev-yury/carrierwave-base64)
5
- [![Code Climate](https://codeclimate.com/github/lebedev-yury/carrierwave-base64/badges/gpa.svg)](https://codeclimate.com/github/lebedev-yury/carrierwave-base64)
4
+ [![Build Status](https://travis-ci.org/y9v/carrierwave-base64.svg?branch=master)](https://travis-ci.org/y9v/carrierwave-base64)
5
+ [![Code Climate](https://codeclimate.com/github/y9v/carrierwave-base64/badges/gpa.svg)](https://codeclimate.com/github/y9v/carrierwave-base64)
6
+ [![Inch CI](https://inch-ci.org/github/y9v/carrierwave-base64.svg?branch=master)](https://inch-ci.org/github/y9v/carrierwave-base64)
6
7
 
7
8
  Upload files encoded as base64 to carrierwave.
8
9
 
@@ -21,15 +21,17 @@ Gem::Specification.new do |spec|
21
21
 
22
22
  spec.add_dependency 'carrierwave', '>= 0.8.0'
23
23
  spec.add_dependency 'mime-types', '~> 3.0'
24
+ spec.add_dependency 'mimemagic', '~> 0.3.2'
24
25
 
25
26
  spec.add_development_dependency 'bundler', '~> 1.7'
26
27
  spec.add_development_dependency 'carrierwave-mongoid'
27
28
  spec.add_development_dependency 'mongoid'
28
29
  spec.add_development_dependency 'pry'
29
- spec.add_development_dependency 'rails', '~> 4'
30
+ spec.add_development_dependency 'rails', '~> 5'
30
31
  spec.add_development_dependency 'rake', '~> 10.0'
31
32
  spec.add_development_dependency 'rspec', '~> 2.14'
32
33
  spec.add_development_dependency 'rubocop'
33
34
  spec.add_development_dependency 'sham_rack'
34
35
  spec.add_development_dependency 'sqlite3'
36
+ spec.add_development_dependency 'yard'
35
37
  end
@@ -2,6 +2,7 @@ require 'mime/types/full'
2
2
  require 'carrierwave/base64/version'
3
3
  require 'carrierwave/base64/exceptions'
4
4
  require 'carrierwave/base64/base64_string_io'
5
+ require 'carrierwave/base64/mounting_helper'
5
6
  require 'carrierwave/base64/adapter'
6
7
  require 'carrierwave/base64/railtie' if defined?(Rails)
7
8
 
@@ -1,42 +1,32 @@
1
1
  module Carrierwave
2
2
  module Base64
3
+ # Module with .mount_base64_uploader method, that is mixed in to
4
+ # ActiveRecord::Base or Mongoid::Document::ClassMethods
3
5
  module Adapter
4
- # rubocop:disable Metrics/MethodLength, Metrics/AbcSize
5
- # rubocop:disable Metrics/CyclomaticComplexity
6
- # rubocop:disable Metrics/PerceivedComplexity
6
+ # Mounts the carrierwave uploader that can accept a base64 encoded
7
+ # string as input. It also accepts regular file uploads.
8
+ #
9
+ # @param attribute [Symbol] the attribute to mount this uploader on
10
+ # @param uploader_class [Carrierwave::Uploader] the uploader class to
11
+ # mount
12
+ # @param options [Hash{Symbol => Object}] a set of options
13
+ # @option options [Proc] :file_name Proc that must return a file name
14
+ # without extension
15
+ #
16
+ # @example Mount the uploader and specify the file name
17
+ # mount_base64_uploader :image, ImageUploader,
18
+ # file_name: -> (u) { u.username }
19
+ #
20
+ # @return [Symbol] the defined writer method name
7
21
  def mount_base64_uploader(attribute, uploader_class, options = {})
8
22
  mount_uploader attribute, uploader_class, options
9
23
  options[:file_name] ||= proc { attribute }
10
24
 
11
- if options[:file_name].is_a?(String)
12
- warn(
13
- '[Deprecation warning] Setting `file_name` option to a string is '\
14
- 'deprecated and will be removed in 3.0.0. If you want to keep the '\
15
- 'existing behaviour, wrap the string in a Proc'
16
- )
17
- end
25
+ Carrierwave::Base64::MountingHelper.check_for_deprecations(options)
18
26
 
19
- define_method "#{attribute}=" do |data|
20
- return if data == send(attribute).to_s
21
-
22
- if respond_to?("#{attribute}_will_change!") && data.present?
23
- send "#{attribute}_will_change!"
24
- end
25
-
26
- return super(data) unless data.is_a?(String) &&
27
- data.strip.start_with?('data')
28
-
29
- filename = if options[:file_name].respond_to?(:call)
30
- options[:file_name].call(self)
31
- else
32
- options[:file_name]
33
- end.to_s
34
-
35
- super Carrierwave::Base64::Base64StringIO.new(data.strip, filename)
36
- end
37
- # rubocop:enable Metrics/MethodLength, Metrics/AbcSize
38
- # rubocop:enable Metrics/CyclomaticComplexity
39
- # rubocop:enable Metrics/PerceivedComplexity
27
+ Carrierwave::Base64::MountingHelper.define_writer(
28
+ self, attribute, options
29
+ )
40
30
  end
41
31
  end
42
32
  end
@@ -1,8 +1,28 @@
1
+ require 'mimemagic'
2
+ require 'mimemagic/overlay'
3
+
1
4
  module Carrierwave
2
5
  module Base64
6
+ # Class that decodes a base64 string, builds a StringIO for the
7
+ # decoded bytes, and extracts the file MIME type to build a file
8
+ # name with extension.
3
9
  class Base64StringIO < StringIO
4
- attr_accessor :file_extension, :file_name
10
+ # @return [String] the file name without extension
11
+ attr_reader :file_name
12
+
13
+ # @return [String] the file extension for the uploaded file
14
+ attr_reader :file_extension
5
15
 
16
+ # Returns a StringIO with decoded bytes from the base64 encoded
17
+ # string and builds a file name with extension for the uploaded file,
18
+ # based on the MIME type specified in the base64 encoded string.
19
+ #
20
+ # @param encoded_file [String] the base64 encoded file contents
21
+ # @param file_name [String] the file name without extention
22
+ #
23
+ # @raise [ArgumentError] If the base64 encoded string is empty
24
+ #
25
+ # @return [StringIO] StringIO with decoded bytes
6
26
  def initialize(encoded_file, file_name)
7
27
  description, encoded_bytes = encoded_file.split(',')
8
28
 
@@ -10,21 +30,28 @@ module Carrierwave
10
30
  raise ArgumentError if encoded_bytes.eql?('(null)')
11
31
 
12
32
  @file_name = file_name
13
- @file_extension = get_file_extension description
14
33
  bytes = ::Base64.decode64 encoded_bytes
34
+ @file_extension = get_file_extension description, bytes
15
35
 
16
36
  super bytes
17
37
  end
18
38
 
39
+ # Returns a file name with extension, based on the MIME type specified
40
+ # in the base64 encoded string.
41
+ #
42
+ # @return [String] File name with extention
19
43
  def original_filename
20
44
  File.basename("#{@file_name}.#{@file_extension}")
21
45
  end
22
46
 
23
47
  private
24
48
 
25
- def get_file_extension(description)
26
- content_type = description.split(';base64').first
27
- mime_type = MIME::Types[content_type].first
49
+ # Determine content type from input, with provided type as fallback
50
+ def get_file_extension(description, bytes)
51
+ detected_type = MimeMagic.by_magic(bytes)
52
+ content_type = (detected_type && detected_type.type) ||
53
+ description.split(';base64').first
54
+ mime_type = MIME::Types[content_type].last
28
55
  unless mime_type
29
56
  raise Carrierwave::Base64::UnknownMimeTypeError,
30
57
  "Unknown MIME type: #{content_type}"
@@ -1,5 +1,7 @@
1
1
  module Carrierwave
2
2
  module Base64
3
+ # Exception that gets raised when base64 encoded string
4
+ # is of unknown MIME type
3
5
  class UnknownMimeTypeError < ::ArgumentError; end
4
6
  end
5
7
  end
@@ -0,0 +1,63 @@
1
+ module Carrierwave
2
+ module Base64
3
+ # Module with helper functions for mounting uploaders
4
+ module MountingHelper
5
+ module_function
6
+
7
+ # Checks for deprecations and prints a warning if found any.
8
+ # @private
9
+ #
10
+ # @param options [Hash{Symbol => Object}] the uploader options
11
+ # @return [void]
12
+ def check_for_deprecations(options)
13
+ return unless options[:file_name].is_a?(String)
14
+
15
+ warn(
16
+ '[Deprecation warning] Setting `file_name` option to a string is '\
17
+ 'deprecated and will be removed in 3.0.0. If you want to keep the '\
18
+ 'existing behaviour, wrap the string in a Proc'
19
+ )
20
+ end
21
+
22
+ # Returns a file name for the uploaded file.
23
+ # @private
24
+ #
25
+ # @param model_instance [Object] the model instance object
26
+ # @param options [Hash{Symbol => Object}] the uploader options
27
+ # @return [String] File name without extension
28
+ def file_name(model_instance, options)
29
+ if options[:file_name].respond_to?(:call)
30
+ options[:file_name].call(model_instance)
31
+ else
32
+ options[:file_name]
33
+ end.to_s
34
+ end
35
+
36
+ # Defines an attribute writer method on the class with mounted uploader.
37
+ # @private
38
+ #
39
+ # @param klass [Class] the class with mounted uploader
40
+ # @param attr [Symbol] the attribute for which the writer will be defined
41
+ # @param options [Hash{Symbol => Object}] a set of options
42
+ # @return [Symbol] the defined writer method name
43
+ def define_writer(klass, attr, options)
44
+ klass.send(:define_method, "#{attr}=") do |data|
45
+ # rubocop:disable Lint/NonLocalExitFromIterator
46
+ return if data.to_s.empty? || data == send(attr).to_s
47
+
48
+ # rubocop:enable Lint/NonLocalExitFromIterator
49
+
50
+ send("#{attr}_will_change!") if respond_to?("#{attr}_will_change!")
51
+
52
+ return super(data) unless data.is_a?(String) &&
53
+ data.strip.start_with?('data')
54
+
55
+ super Carrierwave::Base64::Base64StringIO.new(
56
+ data.strip,
57
+ Carrierwave::Base64::MountingHelper.file_name(self, options)
58
+ )
59
+ end
60
+ end
61
+ end
62
+ end
63
+ end
@@ -1,5 +1,7 @@
1
1
  module Carrierwave
2
2
  module Base64
3
+ # Railtie class to load the carrierwave-base64 adapters
4
+ # Loads adapters for ActiveRecord and Mongoid
3
5
  class Railtie < Rails::Railtie
4
6
  ActiveSupport.on_load :active_record do
5
7
  ActiveRecord::Base.extend Carrierwave::Base64::Adapter
@@ -1,5 +1,5 @@
1
1
  module Carrierwave
2
2
  module Base64
3
- VERSION = '2.7.0'.freeze
3
+ VERSION = '2.8.0'.freeze
4
4
  end
5
5
  end
@@ -30,11 +30,10 @@ RSpec.describe Carrierwave::Base64::Adapter do
30
30
 
31
31
  it 'creates a file' do
32
32
  subject.save!
33
- subject.reload
34
33
 
35
34
  expect(
36
35
  subject.image.current_path
37
- ).to eq file_path('../uploads', 'image.jpeg')
36
+ ).to eq file_path('../uploads', 'image.png')
38
37
  end
39
38
  end
40
39
  end
@@ -86,7 +85,6 @@ RSpec.describe Carrierwave::Base64::Adapter do
86
85
 
87
86
  it 'saves the file' do
88
87
  subject.save!
89
- subject.reload
90
88
 
91
89
  expect(
92
90
  subject.image.current_path
@@ -103,11 +101,10 @@ RSpec.describe Carrierwave::Base64::Adapter do
103
101
 
104
102
  it 'creates a file' do
105
103
  subject.save!
106
- subject.reload
107
104
 
108
105
  expect(
109
106
  subject.image.current_path
110
- ).to eq file_path('../uploads', 'batman.jpeg')
107
+ ).to eq file_path('../uploads', 'batman.png')
111
108
  end
112
109
 
113
110
  it 'sets will_change for the attribute' do
@@ -126,10 +123,9 @@ RSpec.describe Carrierwave::Base64::Adapter do
126
123
  it 'should invoke the file_name proc upon each upload' do
127
124
  subject.save!
128
125
  another_subject.save!
129
- another_subject.reload
130
126
  expect(
131
127
  another_subject.image.current_path
132
- ).to eq file_path('../uploads', 'robin.jpeg')
128
+ ).to eq file_path('../uploads', 'robin.png')
133
129
  end
134
130
  end
135
131
  end
@@ -140,21 +136,39 @@ RSpec.describe Carrierwave::Base64::Adapter do
140
136
  file_path('fixtures', 'base64_image.fixture')
141
137
  ).strip
142
138
  subject.save!
143
- subject.reload
144
139
  end
145
140
 
146
141
  it 'keeps the file when setting the attribute to existing value' do
147
- expect(File.exist?(subject.reload.image.file.file)).to be_truthy
142
+ expect(File.exist?(subject.image.file.file)).to be_truthy
148
143
  subject.update!(image: subject.image.to_s)
149
- expect(File.exist?(subject.reload.image.file.file)).to be_truthy
144
+ expect(File.exist?(subject.image.file.file)).to be_truthy
150
145
  end
151
146
 
152
147
  it 'removes files when remove_* is set to true' do
153
148
  subject.remove_image = true
154
149
  subject.save!
155
- expect(subject.reload.image.file).to be_nil
150
+ expect(subject.image.file).to be_nil
156
151
  end
157
152
  end
158
153
  end
154
+
155
+ context 'models with presence validation on attribute with uploader' do
156
+ subject do
157
+ User.validates(:image, presence: true)
158
+ User.mount_base64_uploader(:image, uploader)
159
+ User.new
160
+ end
161
+
162
+ before(:each) do
163
+ subject.image = File.read(
164
+ file_path('fixtures', 'base64_image.fixture')
165
+ ).strip
166
+ subject.save!
167
+ end
168
+
169
+ it 'gives no false positive on presence validation' do
170
+ expect { subject.update!(username: 'new-username') }.not_to raise_error
171
+ end
172
+ end
159
173
  end
160
174
  end
@@ -2,17 +2,19 @@ RSpec.describe Carrierwave::Base64::Base64StringIO do
2
2
  %w[application/vnd.openxmlformats-officedocument.wordprocessingml.document
3
3
  image/jpeg application/pdf audio/mpeg].each do |content_type|
4
4
  context "correct #{content_type} data" do
5
- let(:data) do
6
- "data:#{content_type};base64,/9j/4AAQSkZJRgABAQEASABKdhH//2Q=="
7
- end
8
-
9
5
  let(:file_extension) do
10
6
  MIME::Types[content_type].last.preferred_extension
11
7
  end
12
8
 
9
+ let(:data) do
10
+ file_name = "test.#{file_extension}"
11
+ bytes = File.read(file_path('fixtures', file_name))
12
+ "data:#{content_type};base64,#{::Base64.encode64(bytes)}"
13
+ end
14
+
13
15
  subject { described_class.new data, 'file' }
14
16
 
15
- it 'determines the file format from the Data URI content type' do
17
+ it 'determines the file format from the content type' do
16
18
  expect(subject.file_extension).to eql(file_extension)
17
19
  end
18
20
 
@@ -1 +1 @@
1
- 
1
+ 
Binary file
Binary file
Binary file
Binary file
File without changes
@@ -14,7 +14,6 @@ require 'carrierwave/mongoid'
14
14
 
15
15
  require 'carrierwave/base64'
16
16
 
17
- ActiveRecord::Base.raise_in_transactional_callbacks = true
18
17
  ActiveRecord::Base.establish_connection(
19
18
  adapter: 'sqlite3',
20
19
  database: ':memory:'
@@ -29,3 +28,15 @@ def file_path(*paths)
29
28
  end
30
29
 
31
30
  CarrierWave.root = ''
31
+
32
+ # Add preferred file types similar to how users would implement custom types
33
+ content_types = {
34
+ 'audio/mpeg' => 'mp3'
35
+ }
36
+ content_types.each do |content_type, extension|
37
+ MIME::Types.add(
38
+ MIME::Type.new(content_type).tap do |type|
39
+ type.preferred_extension = extension
40
+ end
41
+ )
42
+ end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: carrierwave-base64
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.7.0
4
+ version: 2.8.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Yury Lebedev
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2018-04-12 00:00:00.000000000 Z
11
+ date: 2018-11-29 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: carrierwave
@@ -38,6 +38,20 @@ dependencies:
38
38
  - - "~>"
39
39
  - !ruby/object:Gem::Version
40
40
  version: '3.0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: mimemagic
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: 0.3.2
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: 0.3.2
41
55
  - !ruby/object:Gem::Dependency
42
56
  name: bundler
43
57
  requirement: !ruby/object:Gem::Requirement
@@ -100,14 +114,14 @@ dependencies:
100
114
  requirements:
101
115
  - - "~>"
102
116
  - !ruby/object:Gem::Version
103
- version: '4'
117
+ version: '5'
104
118
  type: :development
105
119
  prerelease: false
106
120
  version_requirements: !ruby/object:Gem::Requirement
107
121
  requirements:
108
122
  - - "~>"
109
123
  - !ruby/object:Gem::Version
110
- version: '4'
124
+ version: '5'
111
125
  - !ruby/object:Gem::Dependency
112
126
  name: rake
113
127
  requirement: !ruby/object:Gem::Requirement
@@ -178,6 +192,20 @@ dependencies:
178
192
  - - ">="
179
193
  - !ruby/object:Gem::Version
180
194
  version: '0'
195
+ - !ruby/object:Gem::Dependency
196
+ name: yard
197
+ requirement: !ruby/object:Gem::Requirement
198
+ requirements:
199
+ - - ">="
200
+ - !ruby/object:Gem::Version
201
+ version: '0'
202
+ type: :development
203
+ prerelease: false
204
+ version_requirements: !ruby/object:Gem::Requirement
205
+ requirements:
206
+ - - ">="
207
+ - !ruby/object:Gem::Version
208
+ version: '0'
181
209
  description: This gem can be useful, if you need to upload files to your API from
182
210
  mobile devises.
183
211
  email:
@@ -200,12 +228,17 @@ files:
200
228
  - lib/carrierwave/base64/adapter.rb
201
229
  - lib/carrierwave/base64/base64_string_io.rb
202
230
  - lib/carrierwave/base64/exceptions.rb
231
+ - lib/carrierwave/base64/mounting_helper.rb
203
232
  - lib/carrierwave/base64/railtie.rb
204
233
  - lib/carrierwave/base64/version.rb
205
234
  - spec/adapter_spec.rb
206
235
  - spec/base64_string_io_spec.rb
207
236
  - spec/fixtures/base64_image.fixture
208
- - spec/fixtures/test.jpg
237
+ - spec/fixtures/test.docx
238
+ - spec/fixtures/test.jpeg
239
+ - spec/fixtures/test.mp3
240
+ - spec/fixtures/test.pdf
241
+ - spec/fixtures/test.png
209
242
  - spec/spec_helper.rb
210
243
  - spec/support/custom_expectations/warn_expectation.rb
211
244
  - spec/support/models.rb
@@ -230,7 +263,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
230
263
  version: '0'
231
264
  requirements: []
232
265
  rubyforge_project:
233
- rubygems_version: 2.7.3
266
+ rubygems_version: 2.7.6
234
267
  signing_key:
235
268
  specification_version: 4
236
269
  summary: Upload images encoded as base64 to carrierwave.
@@ -238,7 +271,11 @@ test_files:
238
271
  - spec/adapter_spec.rb
239
272
  - spec/base64_string_io_spec.rb
240
273
  - spec/fixtures/base64_image.fixture
241
- - spec/fixtures/test.jpg
274
+ - spec/fixtures/test.docx
275
+ - spec/fixtures/test.jpeg
276
+ - spec/fixtures/test.mp3
277
+ - spec/fixtures/test.pdf
278
+ - spec/fixtures/test.png
242
279
  - spec/spec_helper.rb
243
280
  - spec/support/custom_expectations/warn_expectation.rb
244
281
  - spec/support/models.rb