Merge pull request #970 from chef/ksubrama/package

Speed up windows package lookup
This commit is contained in:
Christoph Hartmann 2016-08-24 14:57:30 +02:00 committed by GitHub
commit 8825b71412
6 changed files with 58 additions and 18 deletions

View file

@ -184,14 +184,27 @@ module Inspec::Resources
end end
end end
# Determines the installed packages on Windows # Determines the installed packages on Windows using the Windows package registry entries.
# Currently we use 'Get-WmiObject -Class Win32_Product' as a detection method
# TODO: evaluate if alternative methods as proposed by Microsoft are still valid:
# @see: http://blogs.technet.com/b/heyscriptingguy/archive/2013/11/15/use-powershell-to-find-installed-software.aspx # @see: http://blogs.technet.com/b/heyscriptingguy/archive/2013/11/15/use-powershell-to-find-installed-software.aspx
class WindowsPkg < PkgManagement class WindowsPkg < PkgManagement
def info(package_name) def info(package_name)
search_paths = [
'HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\*',
'HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\*',
]
# add 64 bit search paths
if inspec.os.arch == 'x86_64'
search_paths << 'HKLM:\SOFTWARE\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall\*'
search_paths << 'HKCU:\SOFTWARE\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall\*'
end
# Find the package # Find the package
cmd = inspec.command("Get-WmiObject -Class Win32_Product | Where-Object {$_.Name -eq '#{package_name}'} | Select-Object -Property Name,Version,Vendor,PackageCode,Caption,Description | ConvertTo-Json") cmd = inspec.command <<-EOF.gsub(/^\s*/, '')
Get-ItemProperty (@("#{search_paths.join('", "')}") | Where-Object { Test-Path $_ }) |
Where-Object { $_.DisplayName -like "#{package_name}*" -or $_.PSChildName -like "#{package_name}" } |
Select-Object -Property DisplayName,DisplayVersion | ConvertTo-Json
EOF
begin begin
package = JSON.parse(cmd.stdout) package = JSON.parse(cmd.stdout)
@ -199,10 +212,13 @@ module Inspec::Resources
return nil return nil
end end
# What if we match multiple packages? just pick the first one for now.
package = package[0] if package.is_a?(Array)
{ {
name: package['Name'], name: package['DisplayName'],
installed: true, installed: true,
version: package['Version'], version: package['DisplayVersion'],
type: 'windows', type: 'windows',
} }
end end

View file

@ -173,7 +173,7 @@ class MockLoader
# ports on freebsd # ports on freebsd
'sockstat -46l' => cmd.call('sockstat'), 'sockstat -46l' => cmd.call('sockstat'),
# packages on windows # packages on windows
"Get-WmiObject -Class Win32_Product | Where-Object {$_.Name -eq 'Microsoft Visual C++ 2008 Redistributable - x64 9.0.30729.6161'} | Select-Object -Property Name,Version,Vendor,PackageCode,Caption,Description | ConvertTo-Json" => cmd.call('win32_product'), 'f18912b2e36924b367a110c31da6b835a1c217cd10014c7312b7435bf79a601c' => cmd.call('get-item-property-package'),
# service status upstart on ubuntu # service status upstart on ubuntu
'initctl status ssh' => cmd.call('initctl-status-ssh'), 'initctl status ssh' => cmd.call('initctl-status-ssh'),
# service config for upstart on ubuntu # service config for upstart on ubuntu

View file

@ -28,3 +28,31 @@ end
describe package('nginx') do describe package('nginx') do
it { should_not be_installed } it { should_not be_installed }
end end
# the following test will iterate over all packages retrieved via WMI and verifies that the
# optimized package implementation is returning the same results
if os.windows?
# compare optimized version with wmi results
packages = powershell("Get-WmiObject -Class Win32_Product | % { $_.Name }").stdout.strip.split("\n")
packages.each { |pkg|
package_name = pkg.strip
# get wmi package information
cmd = powershell("Get-WmiObject -Class Win32_Product | Where-Object {$_.Name -eq '#{package_name}'} | Select-Object -Property Name,Version,Vendor,PackageCode,Caption,Description | ConvertTo-Json")
wmi_package = JSON.parse(cmd.stdout)
wmi_info = {
name: wmi_package['Name'],
installed: true,
version: wmi_package['Version'],
type: 'windows',
}
# get registry package information
info = package(package_name).info
# compare results
describe wmi_info do
it { should eq info}
end
}
end

View file

@ -0,0 +1,4 @@
{
"DisplayName": "Chef Client v12.12.15",
"DisplayVersion": "12.12.15.1"
}

View file

@ -1,8 +0,0 @@
{
"Name": "Microsoft Visual C++ 2008 Redistributable - x64 9.0.30729.6161",
"Version": "9.0.30729.6161",
"Vendor": "Microsoft Corporation",
"PackageCode": "{9C7D912C-6EDE-47A4-962E-7A83663440BA}",
"Caption": "Microsoft Visual C++ 2008 Redistributable - x64 9.0.30729.6161",
"Description": "Microsoft Visual C++ 2008 Redistributable - x64 9.0.30729.6161"
}

View file

@ -53,10 +53,10 @@ describe 'Inspec::Resources::Package' do
# windows # windows
it 'verify windows package parsing' do it 'verify windows package parsing' do
resource = MockLoader.new(:windows).load_resource('package', 'Microsoft Visual C++ 2008 Redistributable - x64 9.0.30729.6161') resource = MockLoader.new(:windows).load_resource('package', 'Chef Client v12.12.15')
pkg = { name: 'Microsoft Visual C++ 2008 Redistributable - x64 9.0.30729.6161', installed: true, version: '9.0.30729.6161', type: 'windows' } pkg = { name: 'Chef Client v12.12.15', installed: true, version: '12.12.15.1', type: 'windows' }
_(resource.installed?).must_equal true _(resource.installed?).must_equal true
_(resource.version).must_equal '9.0.30729.6161' _(resource.version).must_equal '12.12.15.1'
_(resource.info).must_equal pkg _(resource.info).must_equal pkg
end end