require "functional/helper" describe "inspec shell tests" do include FunctionalHelper parallelize_me! describe "cmd" do def assert_shell_c(code, exit_status, json = false, stderr = "") json_suffix = " --reporter 'json'" if json command = "shell -c '#{code.tr("'", '\\\'')}'#{json_suffix}" out = inspec(command) actual = out.stderr.gsub(/\e\[(\d+)(;\d+)*m/, "") # strip ANSI color codes _(actual).must_equal stderr assert_exit_code exit_status, out out end it "loads a dependency" do res = inspec("shell -c 'example_config' --depends #{example_profile}") _(res.stdout.chop).must_equal "example_config" _(res.stderr).must_equal "" assert_exit_code 0, res end it "confirm file caching is disabled" do out = assert_shell_c("inspec.backend.cache_enabled?(:file)", 0) _(out.stdout.chop).must_equal "false" end it "confirm command caching is disabled" do out = assert_shell_c("inspec.backend.cache_enabled?(:command)", 0) _(out.stdout.chop).must_equal "false" end it "can run ruby expressions (json output)" do x = rand y = rand out = assert_shell_c("#{x} + #{y}", 0, true) j = JSON.load(out.stdout) _(j).must_equal x + y end it "can run ruby expressions" do x = rand y = rand out = assert_shell_c("#{x} + #{y}", 0) _(out.stdout).must_equal "#{x + y}\n" end it "can run arbitrary ruby (json output)" do # You cannot have a pipe in a windows command line return if is_windows? out = assert_shell_c("x = [1,2,3].inject(0) {|a,v| a + v*v}; x+10", 0, true) j = JSON.load(out.stdout) _(j).must_equal 24 # 1^2 + 2^2 + 3^2 + 10 end it "can run arbitrary ruby" do # You cannot have a pipe in a windows command line return if is_windows? out = assert_shell_c("x = [1,2,3].inject(0) {|a,v| a + v*v}; x+10", 0) _(out.stdout).must_equal "24\n" end it "retrieves resources (json output)" do out = assert_shell_c("platform.params", 0, true) j = JSON.load(out.stdout) _(j.keys).must_include "name" _(j.keys).must_include "families" _(j.keys).must_include "arch" _(j.keys).must_include "release" end it "retrieves resources" do out = assert_shell_c("os.params", 0) _(out.stdout).must_include "name" _(out.stdout).must_include "families" _(out.stdout).must_include "arch" _(out.stdout).must_include "release" end it "runs anonymous tests that succeed (json output)" do skip_windows! # Breakage confirmed out = assert_shell_c("describe file(\"#{__FILE__}\") do it { should exist } end", 0, true) j = JSON.load(out.stdout) _(j.keys).must_include "version" _(j.keys).must_include "profiles" _(j.keys).must_include "statistics" end it "runs anonymous tests that succeed" do skip_windows! # Breakage confirmed out = assert_shell_c("describe file(\"#{__FILE__}\") do it { should exist } end", 0) _(out.stdout).must_include "1 successful" _(out.stdout).must_include "0 failures" end it "runs anonymous tests that fail (json output)" do skip_windows! # Breakage confirmed out = assert_shell_c("describe file(\"foo/bar/baz\") do it { should exist } end", 100, true) j = JSON.load(out.stdout) _(j.keys).must_include "version" _(j.keys).must_include "profiles" _(j.keys).must_include "statistics" end it "runs anonymous tests that fail" do skip_windows! # Breakage confirmed out = assert_shell_c("describe file(\"foo/bar/baz\") do it { should exist } end", 100) _(out.stdout).must_include "0 successful" _(out.stdout).must_include "1 failure" end it "runs controls with tests (json output)" do skip_windows! # Breakage confirmed out = assert_shell_c("control \"test\" do describe file(\"#{__FILE__}\") do it { should exist } end end", 0, true) j = JSON.load(out.stdout) _(j.keys).must_include "version" _(j.keys).must_include "profiles" _(j.keys).must_include "statistics" end it "runs controls with tests" do skip_windows! # Breakage confirmed out = assert_shell_c("control \"test\" do describe file(\"#{__FILE__}\") do it { should exist } end end", 0) _(out.stdout).must_include "1 successful" _(out.stdout).must_include "0 failures" end it "runs controls with multiple tests (json output)" do skip_windows! # Breakage confirmed out = assert_shell_c("control \"test\" do describe file(\"#{__FILE__}\") do it { should exist } end; describe file(\"foo/bar/baz\") do it { should exist } end end", 100, true) j = JSON.load(out.stdout) _(j.keys).must_include "version" _(j.keys).must_include "profiles" _(j.keys).must_include "statistics" end it "runs controls with multiple tests" do skip_windows! # Breakage confirmed out = assert_shell_c("control \"test\" do describe file(\"#{__FILE__}\") do it { should exist } end; describe file(\"foo/bar/baz\") do it { should exist } end end", 100) _(out.stdout).must_include "0 successful" _(out.stdout).must_include "1 failure" end end # Pry does not support STDIN from windows currently. Skipping these for now. unless FunctionalHelper.is_windows? describe "shell" do attr_accessor :out def stdout out.stdout.gsub(/\e\[(\d+)(;\d+)*m/, "") # strip ANSI color codes end def stderr out.stderr.gsub(/\e\[(\d+)(;\d+)*m/, "") # strip ANSI color codes end def do_shell(code, exit_status = 0, stderr = "") cmd = "echo '#{code.tr("'", '\\\'')}' | #{exec_inspec} shell" self.out = CMD.run_command(cmd) assert_exit_code exit_status, out out end it "loads a dependency" do cmd = "echo 'example_config' | #{exec_inspec} shell --depends #{example_profile}" res = CMD.run_command(cmd) _(res.stdout).must_include "=> example_config" assert_exit_code 0, res end it "displays the target device information for the user without requiring the help command" do out = do_shell("1+1") _(out.stdout).must_include "You are currently running on:" end it "provides a help command" do out = do_shell("help") _(out.stdout).must_include "Available commands:" _(out.stdout).must_include "You are currently running on:" end it "provides resource help" do out = do_shell("help file") _(out.stdout).must_include "Use the file InSpec audit resource" end it "provides helpful feedback if an invalid resource is provided" do out = do_shell("help not_a_valid_resource") _(out.stdout).must_include "The resource not_a_valid_resource does not exist." end it "provides a list of resources" do out = do_shell("help resources") _(out.stdout).must_include " - command" _(out.stdout).must_include " - file" _(out.stdout).must_include " - sshd_config" end it "provides matchers help" do out = do_shell("help matchers") _(out.stdout).must_include "For more examples, see: https://www.inspec.io/docs/reference/matchers/" end it "provides empty example help" do out = do_shell("help file") _(out.stdout).must_include "Name" _(out.stdout).must_include "Description" _(out.stdout).must_include "Example" _(out.stdout).must_include "Web Reference" end it "exposes all resources" do out = do_shell("os") _(out.stdout).must_match(/\=> .*Operating.* .*System.* .*Detection/) end it "can run ruby expressions" do x = rand y = rand out = do_shell("#{x} + #{y}") _(out.stdout).must_include "#{x + y}" end it "can run arbitrary ruby" do out = do_shell("x = [1,2,3].inject(0) {|a,v| a + v*v}; x+10") _(out.stdout).must_include "24" end it "runs anonymous tests that succeed" do out = do_shell("describe file(\"#{__FILE__}\") do it { should exist } end") _(out.stdout).must_include "1 successful" _(out.stdout).must_include "0 failures" end it "runs anonymous tests that fail" do out = do_shell("describe file(\"foo/bar/baz\") do it { should exist } end") _(out.stdout).must_include "0 successful" _(out.stdout).must_include "1 failure" end it "runs controls with tests" do out = do_shell("control \"test\" do describe file(\"#{__FILE__}\") do it { should exist } end end") _(out.stdout).must_include "1 successful" _(out.stdout).must_include "0 failures" end it "runs controls with multiple tests" do out = do_shell("control \"test\" do describe file(\"#{__FILE__}\") do it { should exist } end; describe file(\"foo/bar/baz\") do it { should exist } end end") _(out.stdout).must_include "0 successful" _(out.stdout).must_include "1 failure" end it "reruns controls when redefined" do out = do_shell("control \"test\" do describe file(\"#{__FILE__}\") do it { should exist } end end\ncontrol \"test\" do describe file(\"foo/bar/baz\") do it { should exist } end end") _(out.stdout).must_include "1 successful" _(out.stdout).must_include "1 failure" end end end end