forked from mitre/nginx-baseline
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlinux_updates.rb
143 lines (125 loc) · 3.83 KB
/
linux_updates.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
# encoding: utf-8
# copyright: 2016, Christoph Hartmann
# copyright: 2016, Dominik Richter
# license: MPLv2
#
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
require 'json'
class LinuxUpdateManager < Inspec.resource(1)
name 'linux_update'
desc 'Use the linux_update InSpec audit resource to test for available or installed updates'
def initialize
if inspec.os.redhat?
@update_mgmt = RHELUpdateFetcher.new(inspec)
elsif inspec.os.debian?
@update_mgmt = UbuntuUpdateFetcher.new(inspec)
end
return skip_resource 'The `linux_update` resource is not supported on your OS.' if @update_mgmt.nil?
end
def updates
return [] if @update_mgmt.nil?
u = @update_mgmt.updates
return [] if u.nil? || u.empty?
u['available']
end
def uptodate?
return nil if @update_mgmt.nil?
u = @update_mgmt.updates
return false if u.nil? || !u['available'].empty?
true
end
def packages
return [] if @update_mgmt.nil?
p = @update_mgmt.packages
return [] if p.nil? || u.empty?
p['installed']
end
def to_s
'Linux Update'
end
end
class UpdateFetcher
def initialize(inspec)
@inspec = inspec
end
def packages
[]
end
def updates
[]
end
def parse_json(script)
cmd = @inspec.bash(script)
begin
JSON.parse(cmd.stdout)
rescue JSON::ParserError => _e
return []
end
end
end
class UbuntuUpdateFetcher < UpdateFetcher
def packages
ubuntu_packages = ubuntu_base + <<-EOH
echo -n '{"installed":['
dpkg-query -W -f='${Status}\\t${Package}\\t${Version}\\t${Architecture}\\n' |\\
grep '^install ok installed\\s' |\\
awk '{ printf "{\\"name\\":\\""$4"\\",\\"version\\":\\""$5"\\",\\"arch\\":\\""$6"\\"}," }' | rev | cut -c 2- | rev | tr -d '\\n'
echo -n ']}'
EOH
parse_json(ubuntu_packages)
end
def updates
ubuntu_updates = ubuntu_base + <<-EOH
echo -n '{"available":['
DEBIAN_FRONTEND=noninteractive apt-get upgrade --dry-run | grep Inst | tr -d '[]()' |\\
awk '{ printf "{\\"name\\":\\""$2"\\",\\"version\\":\\""$4"\\",\\"repo\\":\\""$5"\\",\\"arch\\":\\""$6"\\"}," }' | rev | cut -c 2- | rev | tr -d '\\n'
echo -n ']}'
EOH
parse_json(ubuntu_updates)
end
private
def ubuntu_base
base = <<-EOH
#!/bin/sh
DEBIAN_FRONTEND=noninteractive apt-get update >/dev/null 2>&1
readlock() { cat /proc/locks | awk '{print $5}' | grep -v ^0 | xargs -I {1} find /proc/{1}/fd -maxdepth 1 -exec readlink {} \\; | grep '^/var/lib/dpkg/lock$'; }
while test -n "$(readlock)"; do sleep 1; done
echo " "
EOH
base
end
end
class RHELUpdateFetcher < UpdateFetcher
def packages
rhel_packages = <<-EOH
sleep 2 && echo " "
echo -n '{"installed":['
rpm -qa --queryformat '"name":"%{NAME}","version":"%{VERSION}-%{RELEASE}","arch":"%{ARCH}"\\n' |\\
awk '{ printf "{"$1"}," }' | rev | cut -c 2- | rev | tr -d '\\n'
echo -n ']}'
EOH
parse_json(rhel_packages)
end
def updates
rhel_updates = <<-EOH
#!/bin/sh
python -c 'import sys; sys.path.insert(0, "/usr/share/yum-cli"); import cli; list = cli.YumBaseCli().returnPkgLists(["updates"]);res = ["{\\"name\\":\\""+x.name+"\\", \\"version\\":\\""+x.version+"-"+x.release+"\\",\\"arch\\":\\""+x.arch+"\\",\\"repository\\":\\""+x.repo.id+"\\"}" for x in list.updates]; print "{\\"available\\":["+",".join(res)+"]}"'
EOH
# puts rhel_updates
cmd = @inspec.bash(rhel_updates)
unless cmd.exit_status == 0
# essentially we want https://github.com/chef/inspec/issues/1205
# STDERR.puts 'Could not determine patch status.'
return nil
end
first = cmd.stdout.index('{')
res = cmd.stdout.slice(first, cmd.stdout.size - first)
begin
JSON.parse(res)
rescue JSON::ParserError => _e
return []
end
end
end