Project

General

Profile

Download (38.6 KB) Statistics
| Branch: | Tag: | Revision:
require 'apipie-bindings'
require 'hammer_cli'
require 'json'
require 'open-uri'
require 'csv'
require 'hammer_cli_csv/csv'

# rubocop:disable ClassLength
module HammerCLICsv
class BaseCommand < HammerCLI::Apipie::Command
include ::HammerCLICsv::Utils::Config

option %w(-v --verbose), :flag, 'be verbose'
option %w(--threads), 'THREAD_COUNT', 'Number of threads to hammer with',
:default => 1, :hidden => true
option %w(--export), :flag, 'Export current data instead of importing'
option %w(--search), 'SEARCH', 'Only export search results'
option %w(--file), 'FILE_NAME', 'CSV file (default to /dev/stdout with --export, otherwise required)'
option %w(--prefix), 'PREFIX', 'Prefix for all name columns',
:hidden => true
option %w(--organization), 'ORGANIZATION', _('Only process organization matching this name')
option %w(--continue-on-error), :flag, _('Continue processing even if individual resource error')

option %w(--csv-file), 'FILE_NAME', 'Option --csv-file is deprecated. Use --file',
:deprecated => "Use --file", :hidden => true,
:attribute_name => :option_file
option %w(--csv-export), :flag, 'Option --csv-export is deprecated. Use --export',
:deprecated => "Use --export", :hidden => true,
:attribute_name => :option_export

NAME = 'Name'
COUNT = 'Count'

def self.supported?
false
end

def supported?
self.class.supported?
end

def help
print_message _('**** This command is unsupported and is provided as tech preview. ****') unless supported?
super + self.class.help_columns
end

def self.help_columns
''
end

def execute
@api = api_connection
@server_status = check_server_status(@server)

if option_export?
if option_file
CSV.open(option_file, 'wb', {:force_quotes => false}) do |csv|
export csv
end
else
CSV do |csv|
export csv
end
end
else
import
end
HammerCLI::EX_OK
end

def authenticate_request(request)
if HammerCLI.context[:api_connection]
HammerCLIForeman.foreman_api_connection.authenticator.authenticate(request, {})
else
request.basic_auth(
HammerCLIForeman.credentials.username,
HammerCLIForeman.credentials.password
)
end
request
end

def check_server_status(server)
url = "#{server}/api/status"
uri = URI(url)
nethttp = Net::HTTP.new(uri.host, uri.port)
nethttp.use_ssl = uri.scheme == 'https'
nethttp.verify_mode = OpenSSL::SSL::VERIFY_NONE
server_status = nethttp.start do |http|
request = Net::HTTP::Get.new uri
authenticate_request(request)
response = http.request(request)
JSON.parse(response.body)
end

url = "#{server}/api/v2/plugins"
uri = URI(url)
nethttp = Net::HTTP.new(uri.host, uri.port)
nethttp.use_ssl = uri.scheme == 'https'
nethttp.verify_mode = OpenSSL::SSL::VERIFY_NONE
server_status['plugins'] = nethttp.start do |http|
request = Net::HTTP::Get.new uri
authenticate_request(request)
response = http.request(request)
JSON.parse(response.body)['results']
end

server_status
end

def namify(name_format, number = 0)
return '' unless name_format
if name_format.index('%')
name = name_format % number
else
name = name_format
end
name = "#{option_prefix}#{name}" if option_prefix
name
end

def labelize(name)
name.gsub(/[^a-z0-9\-_]/i, '_')
end

def thread_import(return_headers = false, filename=nil, name_column=nil)
filename ||= option_file || '/dev/stdin'
csv = []
CSV.new(open(filename), {
:skip_blanks => true,
:headers => :first_row,
:return_headers => return_headers
}).each do |line|
csv << line
end
lines_per_thread = csv.length / option_threads.to_i + 1
splits = []

option_threads.to_i.times do |current_thread|
start_index = ((current_thread) * lines_per_thread).to_i
finish_index = ((current_thread + 1) * lines_per_thread).to_i
finish_index = csv.length if finish_index > csv.length
if start_index <= finish_index
lines = csv[start_index...finish_index].clone
splits << Thread.new do
lines.each do |line|
next if !line[name_column || NAME].nil? && line[name_column || NAME][0] == '#'
begin
logger.debug(line)
yield line
rescue RuntimeError => e
message = "#{e}\n#{line}"
option_continue_on_error? ? $stderr.puts(_("Error: %{message}") % {:message => message}) : raise(message)
end
end
end
end
end

splits.each do |thread|
thread.join
end
end

def hammer_context
{
:interactive => false,
:username => 'admin', # TODO: this needs to come from config/settings
:password => 'changeme' # TODO: this needs to come from config/settings
}
end

def hammer(context = nil)
HammerCLI::MainCommand.new('', context || hammer_context)
end

def foreman_organization(options = {})
@organizations ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @organizations[options[:name]]
if !options[:id]
organization = @api.resource(:organizations).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Organization '%{name}' not found") % {:name => options[:name]} if !organization || organization.empty?
options[:id] = organization[0]['id']
@organizations[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @organizations.key(options[:id])
if !options[:name]
organization = @api.resource(:organizations).call(:show, {'id' => options[:id]})
raise _("Organization 'id=%{id}' not found") % {:id => options[:id]} if !organization || organization.empty?
options[:name] = organization['name']
@organizations[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_location(options = {})
@locations ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @locations[options[:name]]
if !options[:id]
location = @api.resource(:locations).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Location '%{name}' not found") % {:name => options[:name]} if !location || location.empty?
options[:id] = location[0]['id']
@locations[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @locations.key(options[:id])
if !options[:name]
location = @api.resource(:locations).call(:show, {'id' => options[:id]})
raise _("Location 'id=%{id}' not found") % {:id => options[:id]} if !location || location.empty?
options[:name] = location['name']
@locations[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_role(options = {})
@roles ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @roles[options[:name]]
if !options[:id]
role = @api.resource(:roles).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Role '%{name}' not found") % {:name => options[:name]} if !role || role.empty?
options[:id] = role[0]['id']
@roles[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @roles.key(options[:id])
if !options[:name]
role = @api.resource(:roles).call(:show, {'id' => options[:id]})
raise _("Role 'id=%{id}' not found") % {:id => options[:id]} if !role || role.empty?
options[:name] = role['name']
@roles[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_permission(options = {})
@permissions ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @permissions[options[:name]]
if !options[:id]
permission = @api.resource(:permissions).call(:index, {
:per_page => 999999,
'name' => options[:name]
})['results']
raise _("Permission '%{name}' not found") % {:name => options[:name]} if !permission || permission.empty?
options[:id] = permission[0]['id']
@permissions[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @permissions.key(options[:id])
if !options[:name]
permission = @api.resource(:permissions).call(:show, {'id' => options[:id]})
raise _("Permission 'id=%{id}' not found") % {:id => options[:id]} if !permission || permission.empty?
options[:name] = permission['name']
@permissions[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_filter(role, resource, search)
search = nil if search && search.empty?
filters = @api.resource(:filters).call(:index, {
:per_page => 999999,
'search' => "role=\"#{role}\""
})['results']
filters.each do |filter|
resource_type = (filter['resource_type'] || '').split(':')[-1] # To remove "Katello::" when present
return filter['id'] if resource_type == resource && filter['search'] == search
end

nil
end

def foreman_environment(options = {})
@environments ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @environments[options[:name]]
if !options[:id]
environment = @api.resource(:environments).call(:index, {
:per_page => 999999,
'search' => "name=\"#{ options[:name] }\""
})['results']
raise _("Puppet environment '%{name}' not found") % {:name => options[:name]} if !environment || environment.empty?
options[:id] = environment[0]['id']
@environments[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @environments.key(options[:id])
if !options[:name]
environment = @api.resource(:environments).call(:show, {'id' => options[:id]})
raise _("Puppet environment 'id=%{id}' not found") % {:id => options[:id]} if !environment || environment.empty?
options[:name] = environment['name']
@environments[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_template_kind(options = {})
@template_kinds ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @template_kinds[options[:name]]
if !options[:id]
template_kind = @api.resource(:template_kinds).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Template kind '%{name}' not found") % {:name => options[:name]} if !template_kind || template_kind.empty?
options[:id] = template_kind[0]['id']
@template_kinds[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @template_kinds.key(options[:id])
if !options[:name]
template_kind = @api.resource(:template_kinds).call(:show, {'id' => options[:id]})
raise _("Template kind 'id=%{id}' not found") % {:id => options[:id]} if !template_kind || template_kind.empty?
options[:name] = template_kind['name']
@template_kinds[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_operatingsystem(options = {})
@operatingsystems ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @operatingsystems[options[:name]]
if !options[:id]
(osname, major, minor) = split_os_name(options[:name])
search = "name=\"#{osname}\" and major=\"#{major}\" and minor=\"#{minor}\""
operatingsystems = @api.resource(:operatingsystems).call(:index, {
:per_page => 999999,
'search' => search
})['results']
operatingsystem = operatingsystems[0]
raise _("Operating system '%{name}' not found") % {:name => options[:name]} if !operatingsystem || operatingsystem.empty?
options[:id] = operatingsystem['id']
@operatingsystems[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @operatingsystems.key(options[:id])
if !options[:name]
operatingsystem = @api.resource(:operatingsystems).call(:show, {'id' => options[:id]})
raise _("Operating system 'id=%{id}' not found") % {:id => options[:id]} if !operatingsystem || operatingsystem.empty?
options[:name] = build_os_name(operatingsystem['name'],
operatingsystem['major'],
operatingsystem['minor'])
@operatingsystems[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_architecture(options = {})
@architectures ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @architectures[options[:name]]
if !options[:id]
architecture = @api.resource(:architectures).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Architecture '%{name}' not found") % {:name => options[:name]} if !architecture || architecture.empty?
options[:id] = architecture[0]['id']
@architectures[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @architectures.key(options[:id])
if !options[:name]
architecture = @api.resource(:architectures).call(:show, {'id' => options[:id]})
raise _("Architecture 'id=%{id}' not found") % {:id => options[:id]} if !architecture || architecture.empty?
options[:name] = architecture['name']
@architectures[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_domain(options = {})
@domains ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @domains[options[:name]]
if !options[:id]
domain = @api.resource(:domains).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Domain '%{name}' not found") % {:name => options[:name]} if !domain || domain.empty?
options[:id] = domain[0]['id']
@domains[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @domains.key(options[:id])
if !options[:name]
domain = @api.resource(:domains).call(:show, {'id' => options[:id]})
raise _("Domain 'id=%{id}' not found") % {:id => options[:id]} if !domain || domain.empty?
options[:name] = domain['name']
@domains[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_partitiontable(options = {})
@ptables ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @ptables[options[:name]]
if !options[:id]
ptable = @api.resource(:ptables).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Partition table '%{name}' not found") % {:name => options[:name]} if !ptable || ptable.empty?
options[:id] = ptable[0]['id']
@ptables[options[:name]] = options[:id]
end
result = options[:id]
elsif options[:id]
return nil if options[:id].nil?
options[:name] = @ptables.key(options[:id])
if !options[:name]
ptable = @api.resource(:ptables).call(:show, {'id' => options[:id]})
options[:name] = ptable['name']
@ptables[options[:name]] = options[:id]
end
result = options[:name]
elsif !options[:name] && !options[:id]
result = ''
end

result
end

def foreman_medium(options = {})
@media ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @media[options[:name]]
if !options[:id]
ptable = @api.resource(:media).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Partition table '%{name}' not found") % {:name => options[:name]} if !ptable || ptable.empty?
options[:id] = ptable[0]['id']
@media[options[:name]] = options[:id]
end
result = options[:id]
elsif options[:id]
return nil if options[:id].nil?
options[:name] = @media.key(options[:id])
if !options[:name]
ptable = @api.resource(:media).call(:show, {'id' => options[:id]})
options[:name] = ptable['name']
@media[options[:name]] = options[:id]
end
result = options[:name]
elsif !options[:name] && !options[:id]
result = ''
end

result
end

def foreman_host(options = {})
@query_hosts ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @query_hosts[options[:name]]
if !options[:id]
host = @api.resource(:hosts).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Host '%{name}' not found") % {:name => options[:name]} if !host || host.empty?
options[:id] = host[0]['id']
@query_hosts[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @query_hosts.key(options[:id])
if !options[:name]
host = @api.resource(:hosts).call(:show, {'id' => options[:id]})
raise _("Host 'id=%{id}' not found") % {:id => options[:id]} if !host || host.empty?
options[:name] = host['name']
@query_hosts[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_hostgroup(options = {})
@query_hostgroups ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @query_hostgroups[options[:name]]
if !options[:id]
hostgroup = @api.resource(:hostgroups).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Host Group '%{name}' not found") % {:name => options[:name]} if !hostgroup || hostgroup.empty?
options[:id] = hostgroup[0]['id']
@query_hostgroups[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @query_hostgroups.key(options[:id])
if !options[:name]
hostgroup = @api.resource(:hostgroups).call(:show, {'id' => options[:id]})
raise _("Host Group 'id=%{id}' not found") % {:id => options[:id]} if !hostgroup || hostgroup.empty?
options[:name] = hostgroup['name']
@query_hostgroups[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_provisioning_template(options = {})
@query_config_templates ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @query_config_templates[options[:name]]
if !options[:id]
config_template = @api.resource(:config_templates).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Provisioning template '%{name}' not found") % {:name => options[:name]} if !config_template || config_template.empty?
options[:id] = config_template[0]['id']
@query_config_templates[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @query_config_templates.key(options[:id])
if !options[:name]
config_template = @api.resource(:config_templates).call(:show, {'id' => options[:id]})
raise _("Provisioning template 'id=%{id}' not found") % {:id => options[:id]} if !config_template || config_template.empty?
options[:name] = config_template['name']
@query_config_templates[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_smart_proxy(options = {})
@query_smart_proxies ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @query_smart_proxies[options[:name]]
if !options[:id]
smart_proxy = @api.resource(:smart_proxies).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Smart Proxy '%{name}' not found") % {:name => options[:name]} if !smart_proxy || smart_proxy.empty?
options[:id] = smart_proxy[0]['id']
@query_smart_proxies[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @query_smart_proxies.key(options[:id])
if !options[:name]
smart_proxy = @api.resource(:smart_proxies).call(:show, {'id' => options[:id]})
raise _("Smart Proxy 'id=%{id}' not found") % {:id => options[:id]} if !smart_proxy || smart_proxy.empty?
options[:name] = smart_proxy['name']
@query_smart_proxies[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def lifecycle_environment(organization, options = {})
@lifecycle_environments ||= {}
@lifecycle_environments[organization] ||= {
}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @lifecycle_environments[organization][options[:name]]
if !options[:id]
@api.resource(:lifecycle_environments).call(:index, {
:per_page => 999999,
'organization_id' => foreman_organization(:name => organization)
})['results'].each do |environment|
@lifecycle_environments[organization][environment['name']] = environment['id']
end
options[:id] = @lifecycle_environments[organization][options[:name]]
raise _("Lifecycle environment '%{name}' not found") % {:name => options[:name]} if !options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @lifecycle_environments.key(options[:id])
if !options[:name]
environment = @api.resource(:lifecycle_environments).call(:show, {'id' => options[:id]})
raise _("Lifecycle environment '%{name}' not found") % {:name => options[:name]} if !environment || environment.empty?
options[:name] = environment['name']
@lifecycle_environments[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def katello_contentview(organization, options = {})
@contentviews ||= {}
@contentviews[organization] ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @contentviews[organization][options[:name]]
if !options[:id]
@api.resource(:content_views).call(:index, {
:per_page => 999999,
'organization_id' => foreman_organization(:name => organization)
})['results'].each do |contentview|
@contentviews[organization][contentview['name']] = contentview['id']
end
options[:id] = @contentviews[organization][options[:name]]
raise _("Content view '%{name}' not found") % {:name => options[:name]} if !options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @contentviews.key(options[:id])
if !options[:name]
contentview = @api.resource(:content_views).call(:show, {'id' => options[:id]})
raise _("Puppet contentview 'id=%{id}' not found") % {:id => options[:id]} if !contentview || contentview.empty?
options[:name] = contentview['name']
@contentviews[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def katello_contentviewversion(organization, name, version='latest')
@contentviewversions ||= {}
@contentviewversions[organization] ||= {}
versionname = "#{version}|#{name}"

return nil if name.nil? || name.empty?
id = @contentviewversions[organization][versionname]
if !id
contentview_id = katello_contentview(organization, :name => name)
contentviewversions = @api.resource(:content_view_versions).call(:index, {
:per_page => 999999,
'content_view_id' => contentview_id
})['results'].sort { |a, b| a['created_at'] <=> b['created_at'] }
if version == 'latest'
@contentviewversions[organization][versionname] = contentviewversions[-1]['id']
else
contentviewversions.each do |contentviewversion|
if contentviewversion['version'] == version.to_f
@contentviewversions[organization][versionname] = contentviewversion['id']
end
end
end
id = @contentviewversions[organization][versionname]
raise _("Content view version '%{name}' with version '%{version}' not found") % {:name => name, :version => version} if !id
end

id
end

def katello_repository(organization, options = {})
@repositories ||= {}
@repositories[organization] ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @repositories[organization][options[:name]]
if !options[:id]
@api.resource(:repositories).call(:index, {
:per_page => 999999,
'organization_id' => foreman_organization(:name => organization)
})['results'].each do |repository|
@repositories[organization][repository['name']] = repository['id']
end
options[:id] = @repositories[organization][options[:name]]
raise _("Repository '%{name}' not found") % {:name => options[:name]} if !options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @repositories.key(options[:id])
if !options[:name]
repository = @api.resource(:repositories).call(:show, {'id' => options[:id]})
raise _("Puppet repository 'id=%{id}' not found") % {:id => options[:id]} if !repository || repository.empty?
options[:name] = repository['name']
@repositoriesr[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def katello_hostcollection(organization, options = {})
@hostcollections ||= {}
@hostcollections[organization] ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @hostcollections[organization][options[:name]]
if !options[:id]
@api.resource(:host_collections).call(:index,
{
:per_page => 999999,
'organization_id' => foreman_organization(:name => organization),
'search' => search_string('host-collections',options[:name])
})['results'].each do |hostcollection|
@hostcollections[organization][hostcollection['name']] = hostcollection['id'] if hostcollection
end
options[:id] = @hostcollections[organization][options[:name]]
raise _("Host collection '%{name}' not found") % {:name => options[:name]} if !options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @hostcollections.key(options[:id])
if !options[:name]
hostcollection = @api.resource(:host_collections).call(:show, {'id' => options[:id]})
raise _("Host collection 'id=%{id}' not found") % {:id => options[:id]} if !hostcollection || hostcollection.empty?
options[:name] = hostcollection['name']
@hostcollections[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def katello_product(organization, options = {})
@products ||= {}
@products[organization] ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @products[organization][options[:name]]
if !options[:id]
@api.resource(:products).call(:index,
{
:per_page => 999999,
'organization_id' => foreman_organization(:name => organization),
'search' => search_string('host-collections',options[:name])
})['results'].each do |product|
@products[organization][product['name']] = product['id'] if product
end
options[:id] = @products[organization][options[:name]]
raise _("Host collection '%{name}' not found") % {:name => options[:name]} if !options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @products.key(options[:id])
if !options[:name]
product = @api.resource(:host_collections).call(:show, {'id' => options[:id]})
raise _("Host collection 'id=%{id}' not found") % {:id => options[:id]} if !product || product.empty?
options[:name] = product['name']
@products[options[:name]] = options[:id]
end
result = options[:name]
end

result
end

def foreman_container(options = {})
@containers ||= {}

if options[:name]
return nil if options[:name].nil? || options[:name].empty?
options[:id] = @containers[options[:name]]
if !options[:id]
container = @api.resource(:containers).call(:index, {
:per_page => 999999,
'search' => "name=\"#{options[:name]}\""
})['results']
raise _("Container '%{name}' not found") % {:name => options[:name]} if !container || container.empty?
options[:id] = container[0]['id']
@containers[options[:name]] = options[:id]
end
result = options[:id]
else
return nil if options[:id].nil?
options[:name] = @containers.key(options[:id])
if !options[:name]
container = @api.resource(:containers).call(:show, {'id' => options[:id]})
raise _("Container 'id=%{id}' not found") % {:id => options[:id]} if !container || container.empty?
options[:name] = container['name']
@containers[options[:name]] = options[:id]
end
result = options[:name]
end

result
end


def build_os_name(name, major, minor)
name += " #{major}" if major && major != ''
name += ".#{minor}" if minor && minor != ''
name
end

# "Red Hat 6.4" => "Red Hat", "6", "4"
# "Red Hat 6" => "Red Hat", "6", ''
def split_os_name(name)
tokens = name.split(' ')
is_number = Float(tokens[-1]) rescue false
if is_number
(major, minor) = tokens[-1].split('.').flatten
name = tokens[0...-1].join(' ')
else
name = tokens.join(' ')
end
[name, major || '', minor || '']
end

def export_column(object, name, field=nil)
return '' unless object[name]
values = CSV.generate do |column|
column << object[name].collect do |fields|
field.nil? ? yield(fields) : fields[field]
end
end
values.delete!("\n")
end

def collect_column(column)
return [] if column.nil? || column.empty?
CSV.parse_line(column, {:skip_blanks => true}).collect do |value|
yield value
end
end

def pluralize(name)
case name
when /smart_proxy/
'smart_proxies'
else
"#{name}s"
end
end

def associate_organizations(id, organizations, name)
return if organizations.nil?

associations ||= {}
CSV.parse_line(organizations).each do |organization|
organization_id = foreman_organization(:name => organization)
if associations[organization].nil?
associations[organization] = @api.resource(:organizations).call(:show, {'id' => organization_id})[pluralize(name)].collect do |reference_object|
reference_object['id']
end
end
associations[organization] += [id] if !associations[organization].include? id
@api.resource(:organizations).call(:update, {
'id' => organization_id,
'organization' => {
"#{name}_ids" => associations[organization]
}
})
end if organizations && !organizations.empty?
end

def associate_locations(id, locations, name)
return if locations.nil?

associations ||= {}
CSV.parse_line(locations).each do |location|
location_id = foreman_location(:name => location)
if associations[location].nil?
associations[location] = @api.resource(:locations).call(:show, {'id' => location_id})[pluralize(name)].collect do |reference_object|
reference_object['id']
end
end
associations[location] += [id] if !associations[location].include? id

@api.resource(:locations).call(:update, {
'id' => location_id,
'location' => {
"#{name}_ids" => associations[location]
}
})
end if locations && !locations.empty?
end

def apipie_check_param(resource, action, name)
method = @api.resource(pluralize(resource).to_sym).apidoc[:methods].detect do |api_method|
api_method[:name] == action.to_s
end
return false unless method

found = method[:params].detect do |param|
param[:full_name] == name
end
if !found
nested = method[:params].detect do |param|
param[:name] == resource.to_s
end
if nested
found = nested[:params].detect do |param|
param[:full_name] == name
end
end
end
found
end

def count(value)
return 1 if value.nil? || value.empty?
value.to_i
end

private

def search_string(resource, name)
operator = case resource
when "gpg-key", "sync-plan", "lifecycle-environment", "host-collections"
@server_status['version'] && @server_status['version'].match(/\A1\.6/) ? ':' : '='
else
':'
end
"name#{operator}\"#{name}\""
end
end
end
(3-3/39)