inspec/test/functional/inspec_json_profile_test.rb

212 lines
5.8 KiB
Ruby
Raw Normal View History

require "functional/helper"
require "mixlib/shellout"
require "json_schemer"
2016-03-25 00:31:19 +00:00
describe "inspec json" do
2016-03-25 00:31:19 +00:00
include FunctionalHelper
parallelize_me!
it "read the profile json" do
out = inspec("json " + example_profile)
_(out.stderr).must_equal ""
assert_exit_code 0, out
2016-03-25 00:31:19 +00:00
s = out.stdout
_(JSON.load(s)).must_be_kind_of Hash
2016-03-25 00:31:19 +00:00
end
describe "json profile data" do
let(:json) { JSON.load(inspec("json " + example_profile).stdout) }
2016-03-25 00:31:19 +00:00
it "has a generator name" do
_(json["generator"]["name"]).must_equal "inspec"
end
it "has a generator inspec version" do
_(json["generator"]["version"]).must_equal Inspec::VERSION
end
it "has a name" do
_(json["name"]).must_equal "profile"
2016-03-25 00:31:19 +00:00
end
it "has a title" do
_(json["title"]).must_equal "InSpec Example Profile"
2016-03-25 00:31:19 +00:00
end
it "has a summary" do
_(json["summary"]).must_equal "Demonstrates the use of InSpec Compliance Profile"
2016-03-25 00:31:19 +00:00
end
it "has a version" do
_(json["version"]).must_equal "1.0.0"
2016-03-25 00:31:19 +00:00
end
it "has a maintainer" do
_(json["maintainer"]).must_equal "Chef Software, Inc."
2016-03-25 00:31:19 +00:00
end
it "has a copyright" do
_(json["copyright"]).must_equal "Chef Software, Inc."
2016-03-25 00:31:19 +00:00
end
it "has controls" do
_(json["controls"].length).must_equal 4
2016-03-25 00:31:19 +00:00
end
describe "a control" do
let(:control) { json["controls"].find { |x| x["id"] == "tmp-1.0" } }
2016-03-25 00:31:19 +00:00
it "has a title" do
_(control["title"]).must_equal "Create / directory"
2016-03-25 00:31:19 +00:00
end
it "has a description" do
_(control["desc"]).must_equal "An optional description..."
2016-03-25 00:31:19 +00:00
end
it "has an impact" do
_(control["impact"]).must_equal 0.7
2016-03-25 00:31:19 +00:00
end
it "has a ref" do
_(control["refs"]).must_equal([{ "ref" => "Document A-12", "url" => "http://..." }])
2016-03-25 00:31:19 +00:00
end
it "has a source location" do
loc = File.join(example_profile, "/controls/example-tmp.rb")
_(control["source_location"]["ref"]).must_equal loc
_(control["source_location"]["line"]).must_equal 6
2016-03-25 00:31:19 +00:00
end
it "has a the source code" do
_(control["code"]).must_match(/\Acontrol 'tmp-1.0' do.*end\n\Z/m)
2016-03-25 00:31:19 +00:00
end
end
end
describe "filter with --controls" do
let(:out) { inspec("json " + example_profile + " --controls tmp-1.0") }
2016-03-25 00:58:59 +00:00
it "still succeeds" do
_(out.stderr).must_equal ""
assert_exit_code 0, out
2016-03-25 00:58:59 +00:00
end
it "only has one control included" do
2016-03-25 00:58:59 +00:00
json = JSON.load(out.stdout)
_(json["controls"].length).must_equal 1
_(json["controls"][0]["id"]).must_equal "tmp-1.0"
_(json["groups"].length).must_equal 1
_(json["groups"][0]["id"]).must_equal "controls/example-tmp.rb"
2016-03-25 00:58:59 +00:00
end
end
it "writes json to file" do
out = inspec("json " + example_profile + " --output " + dst.path)
2016-03-25 00:31:19 +00:00
hm = JSON.load(File.read(dst.path))
_(hm["name"]).must_equal "profile"
_(hm["controls"].length).must_equal 4
_(out.stderr).must_include "----> creating #{dst.path}"
assert_exit_code 0, out
2016-03-25 00:31:19 +00:00
end
describe "json test for pax header archives" do
let(:profile_tgz) { File.join(Dir.mktmpdir, "pax-profile-test.tar.gz") }
it "successfully reads a pax-formatted tar file" do
# TODO: this needs updated to also support windows taring
return if is_windows?
files = Dir.glob("#{example_profile}/**/*").delete_if { |x| !File.file?(x) }
relatives = files.map { |e| Pathname.new(e).relative_path_from(Pathname.new(example_profile)).to_s }
cmd = Mixlib::ShellOut.new("tar --format=pax -czf #{profile_tgz} #{relatives.join(" ")}", cwd: example_profile)
cmd.run_command
cmd.error!
out = inspec("json #{profile_tgz}")
assert_exit_code 0, out
end
end
describe "inspec json with a inheritance profile" do
let(:profile) { File.join(profile_path, "export-json", "empty-wrapper") }
it "can export a profile that uses inheritance" do
out = inspec("json " + profile)
# This will throw an exception if it is garbled
json = JSON.load(out.stdout)
# and here we verify (very passingly!) that is a structure we expect
_(json).must_be_kind_of Hash
json["controls"].each do |control|
_(control["code"].empty?).must_equal false
end
_(out.stderr).must_be_empty
assert_exit_code 0, out
end
end
describe "inspec json does not write logs to STDOUT" do
it "can execute a profile with warn calls and parse STDOUT as valid JSON" do
out = inspec("json " + File.join(profile_path, "warn_logs"))
assert_equal "warn_logs", JSON.load(out.stdout)["name"]
_(out.stderr).must_include "This is a warn call"
assert_exit_code 0, out
end
end
describe "inspec json with a profile containing only_if" do
it "ignores the `only_if`" do
out = inspec("json " + File.join(profile_path, "only-if-os-nope"))
assert_equal "only-if-os-nope", JSON.load(out.stdout)["name"]
_(out.stderr).must_equal ""
assert_exit_code 0, out
end
end
it "can format a profile and validate the json schema" do
out = inspec("json " + example_profile)
data = JSON.parse(out.stdout)
sout = inspec("schema profile-json")
schema = JSONSchemer.schema(sout.stdout)
_(schema.validate(data).to_a).must_equal []
_(out.stderr).must_equal ""
assert_exit_code 0, out
end
it "properly validates all (valid) unit tests against the schema" do
schema = JSONSchemer.schema(JSON.parse(inspec("schema profile-json").stdout))
all_profile_folders.first(1).each do |folder|
begin
out = inspec("json " + folder)
# Ensure it parses properly; discard the result
out = JSON.parse(out.stdout)
failures = schema.validate(out).to_a
_(failures).must_equal []
rescue JSON::ParserError
# We don't actually care about these; cannot validate if parsing fails!
nil
end
end
end
2016-03-25 00:31:19 +00:00
end