class ActiveReport::Hash attr_accessor :datum, :only, :except, :headers, :options def initialize(datum, only: nil, except: nil, headers: nil, options: {}) @datum = datum @only = only @except = except @headers = headers @options = options end def self.export(datum, only: nil, except: nil, headers: nil, options: {}) new(datum, only: only, except: except, headers: headers, options: options).export end def export @datum = [].push(@datum).compact unless @datum.is_a?(Array) @only = [].push(@only).compact unless @only.is_a?(Array) @except = [].push(@except).compact unless @except.is_a?(Array) CSV.generate(@options) do |csv| header = @datum.first.dup.keep_if { |k,v| @only.include?(k) } unless @only.empty? header = @datum.first.dup.delete_if { |k,v| @except.include?(k) } unless @except.empty? csv << (@headers || (header || @datum.first).keys.map { |k| k.to_s.gsub("_", " ").capitalize}) @datum.each do |data| cell = data.dup.keep_if { |k,v| @only.include?(k) } unless @only.empty? cell = data.dup.delete_if { |k,v| @except.include?(k) } unless @except.empty? csv << (cell || data).values end end end def self.import(datum, only: nil, except: nil, headers: nil, options: {}) new(datum, only: only, except: except, headers: headers, options: options).import end def import return(nil) if @headers.nil? @only = [].push(@only).compact unless @only.is_a?(Array) @except = [].push(@except).compact unless @except.is_a?(Array) processed_datum = [] CSV.foreach(@datum, @options) do |data| processed_data = {} @headers.each_with_index do |v,i| processed_data.store(v.gsub(" ", "_").downcase.to_sym, data.fetch(i, nil) ) end processed_data.keep_if { |k,v| @only.include?(k) } unless @only.empty? processed_data.delete_if { |k,v| @except.include?(k) } unless @except.empty? processed_datum.push(processed_data) end processed_datum end end