Skip to content

Commit 3c9d9b2

Browse files
authored
to supports anything that responds to call (#72)
1 parent 1a910d9 commit 3c9d9b2

File tree

3 files changed

+37
-12
lines changed

3 files changed

+37
-12
lines changed

README.md

+21-1
Original file line numberDiff line numberDiff line change
@@ -187,6 +187,27 @@ Like very advanced stuff? We grant you access to the [`column`](https://github.c
187187
end
188188
```
189189

190+
Note that `to:` accepts anything that responds to call and take 1, 2 or
191+
3 arguments.
192+
193+
```ruby
194+
class ImportUserCSV
195+
include CSVImporter
196+
197+
model User
198+
199+
column :birth_date, to: DateTransformer
200+
column :renewal_date, to: DateTransformer
201+
column :next_renewal_at, to: ->(value) { Time.at(value.to_i) }
202+
end
203+
204+
class DateTransformer
205+
def self.call(date)
206+
Date.strptime(date, '%m/%d/%y')
207+
end
208+
end
209+
```
210+
190211
Now, what if the user does not provide the email column? It's not worth
191212
running the import, we should just reject the CSV file right away.
192213
That's easy:
@@ -206,7 +227,6 @@ import.report.status # => :invalid_header
206227
import.report.message # => "The following columns are required: 'email'"
207228
```
208229

209-
210230
### Update or Create
211231

212232
You often want to find-and-update-or-create when importing a CSV file.

lib/csv_importer/row.rb

+8-7
Original file line numberDiff line numberDiff line change
@@ -52,18 +52,19 @@ def set_attributes(model)
5252
# Set the attribute using the column_definition and the csv_value
5353
def set_attribute(model, column, csv_value)
5454
column_definition = column.definition
55-
if column_definition.to && column_definition.to.is_a?(Proc)
56-
to_proc = column_definition.to
55+
transformer = column_definition.to
56+
if transformer.respond_to?(:call)
57+
arity = transformer.is_a?(Proc) ? transformer.arity : transformer.method(:call).arity
5758

58-
case to_proc.arity
59+
case arity
5960
when 1 # to: ->(email) { email.downcase }
60-
model.public_send("#{column_definition.name}=", to_proc.call(csv_value))
61+
model.public_send("#{column_definition.name}=", transformer.call(csv_value))
6162
when 2 # to: ->(published, post) { post.published_at = Time.now if published == "true" }
62-
to_proc.call(csv_value, model)
63+
transformer.call(csv_value, model)
6364
when 3 # to: ->(field_value, post, column) { post.hash_field[column.name] = field_value }
64-
to_proc.call(csv_value, model, column)
65+
transformer.call(csv_value, model, column)
6566
else
66-
raise ArgumentError, "`to` proc can only have 1, 2 or 3 arguments"
67+
raise ArgumentError, "arity: #{transformer.arity.inspect} - `to` can only have 1, 2 or 3 arguments"
6768
end
6869
else
6970
attribute = column_definition.attribute

spec/csv_importer_spec.rb

+8-4
Original file line numberDiff line numberDiff line change
@@ -61,14 +61,18 @@ def self.store
6161
class ImportUserCSV
6262
include CSVImporter
6363

64+
class ConfirmedProcessor
65+
def self.call(confirmed, model)
66+
model.confirmed_at = confirmed == "true" ? Time.new(2012) : nil
67+
end
68+
end
69+
6470
model User
6571

6672
column :email, required: true, as: /email/i, to: ->(email) { email.downcase }
6773
column :f_name, as: :first_name, required: true
6874
column :last_name, to: :l_name
69-
column :confirmed, to: ->(confirmed, model) do
70-
model.confirmed_at = confirmed == "true" ? Time.new(2012) : nil
71-
end
75+
column :confirmed, to: ConfirmedProcessor
7276
column :extra, as: /extra/i, to: ->(value, model, column) do
7377
model.custom_fields[column.name] = value
7478
end
@@ -495,7 +499,7 @@ class ImportUserCSVByFirstName
495499
import = ImportUserCSV.new(content: csv_content).run!
496500

497501
expect(import).to_not be_success
498-
expect(import.message).to eq "Unclosed quoted field on line 3."
502+
expect(import.message).to include "Unclosed quoted field"
499503
end
500504

501505
it "matches columns via regexp" do

0 commit comments

Comments
 (0)