mirror of
https://github.com/inspec/inspec
synced 2024-11-27 15:10:44 +00:00
577688a3a0
Many of the resources are named as a top-level class with a fairly generic class name, such as "OS". This causes an issue specifically with kitchen-google which depends on a gem which depends on the "os" gem which itself defines an OS class with a different superclass. This prevents users from using TK, Google Compute, and Inspec without this fix. Some mocked commands had their digest changed as well due to the new indentation, specifically in the User and RegistryKey classes. I strongly recommend viewing this diff with `git diff --ignore-space-change` to see the *real* changes. :)
144 lines
3.8 KiB
Ruby
144 lines
3.8 KiB
Ruby
# encoding: utf-8
|
|
# author: Christoph Hartmann
|
|
# author: Dominik Richter
|
|
|
|
require 'utils/parser'
|
|
|
|
module Inspec::Resources
|
|
class XinetdConf < Inspec.resource(1) # rubocop:disable Metrics/ClassLength
|
|
name 'xinetd_conf'
|
|
desc 'Xinetd services configuration.'
|
|
example "
|
|
describe xinetd_conf.services('chargen') do
|
|
its('socket_types') { should include 'dgram' }
|
|
end
|
|
|
|
describe xinetd_conf.services('chargen').socket_types('dgram') do
|
|
it { should be_disabled }
|
|
end
|
|
"
|
|
|
|
include XinetdParser
|
|
|
|
def initialize(conf_path = '/etc/xinetd.conf', opts = {})
|
|
@conf_path = conf_path
|
|
@params = opts[:params] unless opts[:params].nil?
|
|
@filters = opts[:filters] || ''
|
|
@contents = {}
|
|
end
|
|
|
|
def to_s
|
|
"Xinetd config #{@conf_path}"
|
|
end
|
|
|
|
def services(condition = nil)
|
|
condition.nil? ? params['services'].keys : filter(service: condition)
|
|
end
|
|
|
|
def ids(condition = nil)
|
|
condition.nil? ? services_field('id') : filter(id: condition)
|
|
end
|
|
|
|
def socket_types(condition = nil)
|
|
condition.nil? ? services_field('socket_type') : filter(socket_type: condition)
|
|
end
|
|
|
|
def types(condition = nil)
|
|
condition.nil? ? services_field('type') : filter(type: condition)
|
|
end
|
|
|
|
def wait(condition = nil)
|
|
condition.nil? ? services_field('wait') : filter(wait: condition)
|
|
end
|
|
|
|
def disabled?
|
|
filter(disable: 'no').services.empty?
|
|
end
|
|
|
|
def enabled?
|
|
filter(disable: 'yes').services.empty?
|
|
end
|
|
|
|
def params
|
|
return @params if defined?(@params)
|
|
return @params = {} if read_content.nil?
|
|
flat_params = parse_xinetd(read_content)
|
|
@params = { 'services' => {} }
|
|
flat_params.each do |k, v|
|
|
name = k[/^service (.+)$/, 1]
|
|
if name.nil?
|
|
@params[k] = v
|
|
else
|
|
@params['services'][name] = v
|
|
end
|
|
end
|
|
@params
|
|
end
|
|
|
|
def filter(conditions = {})
|
|
res = params.dup
|
|
filters = ''
|
|
conditions.each do |k, v|
|
|
v = v.to_s if v.is_a? Integer
|
|
filters += " #{k} = #{v.inspect}"
|
|
res['services'] = filter_by(res['services'], k.to_s, v)
|
|
end
|
|
XinetdConf.new(@conf_path, params: res, filters: filters)
|
|
end
|
|
|
|
private
|
|
|
|
# Retrieve the provided field from all configured services.
|
|
#
|
|
# @param [String] field name, e.g. `socket_type`
|
|
# @return [Array[String]] all values of this field across services
|
|
def services_field(field)
|
|
params['services'].values.compact.flatten
|
|
.map { |x| x.params[field] }.flatten.compact
|
|
end
|
|
|
|
def match_condition(sth, condition)
|
|
case sth
|
|
# this does Regex-matching as well as string comparison
|
|
when condition
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
# Filter services by a criteria. This allows for search queries for
|
|
# certain values.
|
|
#
|
|
# @param [Hash] service collection
|
|
# @param [String] search key you want to query
|
|
# @param [Any] search value that the key should match
|
|
# @return [Hash] filtered service collection
|
|
def filter_by(services, k, v)
|
|
if k == 'service'
|
|
return Hash[services.find_all { |name, _| match_condition(v, name) }]
|
|
end
|
|
Hash[services.map { |name, service_arr|
|
|
found = service_arr.find_all { |service|
|
|
match_condition(service.params[k], v)
|
|
}
|
|
found.empty? ? nil : [name, found]
|
|
}.compact]
|
|
end
|
|
|
|
def read_content(path = @conf_path)
|
|
return @contents[path] if @contents.key?(path)
|
|
file = inspec.file(path)
|
|
if !file.file?
|
|
return skip_resource "Can't find file \"#{path}\""
|
|
end
|
|
|
|
@contents[path] = file.content
|
|
if @contents[path].empty? && file.size > 0
|
|
return skip_resource "Can't read file \"#{path}\""
|
|
end
|
|
|
|
@contents[path]
|
|
end
|
|
end
|
|
end
|