2015-09-21 08:31:31 +00:00
|
|
|
# encoding: utf-8
|
|
|
|
|
|
|
|
# Parses a csv document
|
|
|
|
# This implementation was inspired by a blog post
|
|
|
|
# @see http://technicalpickles.com/posts/parsing-csv-with-ruby
|
2016-03-08 18:06:55 +00:00
|
|
|
module Inspec::Resources
|
|
|
|
class CsvConfig < JsonConfig
|
|
|
|
name 'csv'
|
2018-02-19 14:26:49 +00:00
|
|
|
supports platform: 'unix'
|
|
|
|
supports platform: 'windows'
|
|
|
|
supports platform: 'esx'
|
|
|
|
supports platform: 'cisco'
|
2016-03-08 18:06:55 +00:00
|
|
|
desc 'Use the csv InSpec audit resource to test configuration data in a CSV file.'
|
|
|
|
example "
|
|
|
|
describe csv('example.csv') do
|
|
|
|
its('name') { should eq(['John', 'Alice']) }
|
|
|
|
end
|
|
|
|
"
|
2015-09-21 08:31:31 +00:00
|
|
|
|
2017-03-14 16:33:54 +00:00
|
|
|
# override the parse method from JsonConfig
|
|
|
|
# Assuming a header row of name,col1,col2, it will output an array of hashes like so:
|
|
|
|
# [
|
|
|
|
# { 'name' => 'row1', 'col1' => 'value1', 'col2' => 'value2' },
|
|
|
|
# { 'name' => 'row2', 'col1' => 'value3', 'col2' => 'value4' }
|
|
|
|
# ]
|
2016-03-08 18:06:55 +00:00
|
|
|
def parse(content)
|
|
|
|
require 'csv'
|
2017-03-14 16:33:54 +00:00
|
|
|
|
2016-03-08 18:06:55 +00:00
|
|
|
# convert empty field to nil
|
|
|
|
CSV::Converters[:blank_to_nil] = lambda do |field|
|
|
|
|
field && field.empty? ? nil : field
|
|
|
|
end
|
2017-03-14 16:33:54 +00:00
|
|
|
|
2016-03-08 18:06:55 +00:00
|
|
|
# implicit conversion of values
|
|
|
|
csv = CSV.new(content, headers: true, converters: [:all, :blank_to_nil])
|
2017-03-14 16:33:54 +00:00
|
|
|
|
2016-03-08 18:06:55 +00:00
|
|
|
# convert to hash
|
|
|
|
csv.to_a.map(&:to_hash)
|
2017-11-27 16:13:02 +00:00
|
|
|
rescue => e
|
|
|
|
raise Inspec::Exceptions::ResourceFailed, "Unable to parse CSV: #{e.message}"
|
2015-09-21 08:31:31 +00:00
|
|
|
end
|
2015-10-12 11:01:58 +00:00
|
|
|
|
2017-03-14 16:33:54 +00:00
|
|
|
# override the value method from JsonConfig
|
|
|
|
# The format of the CSV hash as created by #parse is very different
|
|
|
|
# than what the YAML, JSON, and INI resources create, so using the
|
|
|
|
# #value method from JsonConfig (which uses ObjectTraverser.extract_value)
|
|
|
|
# doesn't make sense here.
|
|
|
|
def value(key)
|
|
|
|
@params.map { |x| x[key.first.to_s] }.compact
|
|
|
|
end
|
|
|
|
|
2017-11-27 16:13:02 +00:00
|
|
|
private
|
|
|
|
|
|
|
|
# used by JsonConfig to build up a full to_s method
|
|
|
|
# based on whether a file path, content, or command was supplied.
|
|
|
|
def resource_base_name
|
|
|
|
'CSV'
|
2016-03-08 18:06:55 +00:00
|
|
|
end
|
2015-10-12 11:01:58 +00:00
|
|
|
end
|
2015-09-21 08:31:31 +00:00
|
|
|
end
|