dockscan 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +9 -0
  3. data/.rspec +2 -0
  4. data/.travis.yml +8 -0
  5. data/CODE_OF_CONDUCT.md +49 -0
  6. data/Gemfile +4 -0
  7. data/LICENSE +340 -0
  8. data/README.md +52 -0
  9. data/Rakefile +6 -0
  10. data/bin/console +14 -0
  11. data/bin/setup +8 -0
  12. data/dockscan.gemspec +27 -0
  13. data/docs/dockscan.png +0 -0
  14. data/exe/dockscan +107 -0
  15. data/lib/dockscan/modules/audit/container-filesystem-diff.rb +41 -0
  16. data/lib/dockscan/modules/audit/container-filesystem-shadow.rb +37 -0
  17. data/lib/dockscan/modules/audit/container-number-process.rb +36 -0
  18. data/lib/dockscan/modules/audit/container-sshd-process.rb +33 -0
  19. data/lib/dockscan/modules/audit/docker-experimental-build.rb +27 -0
  20. data/lib/dockscan/modules/audit/docker-insecure-registries.rb +53 -0
  21. data/lib/dockscan/modules/audit/docker-limits.rb +34 -0
  22. data/lib/dockscan/modules/audit/docker-networking-forwarding.rb +27 -0
  23. data/lib/dockscan/modules/audit/docker-registry-mirror.rb +41 -0
  24. data/lib/dockscan/modules/audit/docker-storage-driver-aufs.rb +28 -0
  25. data/lib/dockscan/modules/audit.rb +32 -0
  26. data/lib/dockscan/modules/discover/get-containers.rb +14 -0
  27. data/lib/dockscan/modules/discover/get-docker-info.rb +17 -0
  28. data/lib/dockscan/modules/discover/get-docker-version.rb +14 -0
  29. data/lib/dockscan/modules/discover/get-images.rb +16 -0
  30. data/lib/dockscan/modules/discover/get-run-containers.rb +14 -0
  31. data/lib/dockscan/modules/discover.rb +16 -0
  32. data/lib/dockscan/modules/genmodule.rb +17 -0
  33. data/lib/dockscan/modules/report/html.rb +182 -0
  34. data/lib/dockscan/modules/report/stdout.rb +36 -0
  35. data/lib/dockscan/modules/report/txt.rb +36 -0
  36. data/lib/dockscan/modules/report.rb +75 -0
  37. data/lib/dockscan/scan/issue.rb +18 -0
  38. data/lib/dockscan/scan/manage.rb +172 -0
  39. data/lib/dockscan/scan/plugin.rb +14 -0
  40. data/lib/dockscan/version.rb +3 -0
  41. data/lib/dockscan.rb +7 -0
  42. metadata +141 -0
@@ -0,0 +1,182 @@
1
+ class ReportHTML < Dockscan::Modules::ReportModule
2
+
3
+ def info
4
+ return 'This plugin produces HTML reports'
5
+ end
6
+
7
+ def file_extension
8
+ return ".html"
9
+ end
10
+
11
+ def format
12
+ return "html"
13
+ end
14
+
15
+ def htmlhead
16
+ htmlout = ""
17
+ htmlout << "<HTML><HEAD><TITLE>dockscan Report</TITLE>\n"
18
+ htmlout << '<style>'
19
+ htmlout << '.Vulnerability {
20
+ margin:0px;padding:0px;
21
+ width:100%;
22
+ border:0px solid #000000;
23
+
24
+ -moz-border-radius-bottomleft:0px;
25
+ -webkit-border-bottom-left-radius:0px;
26
+ border-bottom-left-radius:0px;
27
+
28
+ -moz-border-radius-bottomright:0px;
29
+ -webkit-border-bottom-right-radius:0px;
30
+ border-bottom-right-radius:0px;
31
+
32
+ -moz-border-radius-topright:0px;
33
+ -webkit-border-top-right-radius:0px;
34
+ border-top-right-radius:0px;
35
+
36
+ -moz-border-radius-topleft:0px;
37
+ -webkit-border-top-left-radius:0px;
38
+ border-top-left-radius:0px;
39
+ }.Vulnerability table{
40
+ border-collapse: collapse;
41
+ border-spacing: 0;
42
+ width:100%;
43
+ height:100%;
44
+ margin:0px;padding:0px;
45
+ }.Vulnerability tr:last-child td:last-child {
46
+ -moz-border-radius-bottomright:0px;
47
+ -webkit-border-bottom-right-radius:0px;
48
+ border-bottom-right-radius:0px;
49
+ }
50
+ .Vulnerability table tr:first-child td:first-child {
51
+ -moz-border-radius-topleft:0px;
52
+ -webkit-border-top-left-radius:0px;
53
+ border-top-left-radius:0px;
54
+ }
55
+ .Vulnerability table tr:first-child td:last-child {
56
+ -moz-border-radius-topright:0px;
57
+ -webkit-border-top-right-radius:0px;
58
+ border-top-right-radius:0px;
59
+ }.Vulnerability tr:last-child td:first-child{
60
+ -moz-border-radius-bottomleft:0px;
61
+ -webkit-border-bottom-left-radius:0px;
62
+ border-bottom-left-radius:0px;
63
+ }.Vulnerability tr:hover td{
64
+ background-color:#ffffff;
65
+ }
66
+ .Vulnerability td{
67
+ vertical-align:middle;
68
+
69
+ background-color:#ffffff;
70
+
71
+ border:0px solid #000000;
72
+ border-width:0px 1px 1px 0px;
73
+ text-align:left;
74
+ padding:7px;
75
+ font-size:10px;
76
+ font-family:Arial;
77
+ font-weight:normal;
78
+ color:#000000;
79
+ }.Vulnerability tr:last-child td{
80
+ border-width:0px 0px 0px 0px;
81
+ }.Vulnerability tr td:last-child{
82
+ border-width:0px 0px 0px 0px;
83
+ }.Vulnerability tr:last-child td:last-child{
84
+ border-width:0px 0px 0px 0px;
85
+ }
86
+ .Vulnerability tr:first-child td{
87
+ border:0px solid #000000;
88
+ text-align:left;
89
+ border-width:0px 0px 0px 0px;
90
+ font-size:14px;
91
+ font-family:Arial;
92
+ font-weight:bold;
93
+ color:#000000;
94
+ }
95
+ .Vulnerability tr:first-child:hover td{
96
+ }
97
+ .Vulnerability tr:first-child td:first-child{
98
+ border-width:0px 0px 0px 0px;
99
+ }
100
+ .Vulnerability tr:first-child td:last-child{
101
+ border-width:0px 0px 0px 0px;
102
+ }
103
+
104
+ .VulnerabilityBLACK tr:first-child td{
105
+ background-color:#000000;
106
+ color:#FFFFFF;
107
+ }
108
+
109
+ .VulnerabilityRED tr:first-child td{
110
+ background-color:#ff0000;
111
+ color:#ffffff;
112
+ }
113
+
114
+ .VulnerabilityORANGE tr:first-child td{
115
+ background-color:#ff9966;
116
+ color:#000000;
117
+ }
118
+
119
+ .VulnerabilityYELLOW tr:first-child td{
120
+ background-color:#ffff00;
121
+ color:#000000;
122
+ }
123
+
124
+ .VulnerabilityGRAY tr:first-child td{
125
+ background-color:#d0d0d0;
126
+ color:#000000;
127
+ }
128
+ '
129
+ htmlout << '</style>'
130
+ htmlout << "</HEAD>\n"
131
+ htmlout << "<BODY>\n"
132
+ htmlout << '<img width="220" height="169" title="" alt="" src="" />'
133
+ return htmlout
134
+ end
135
+
136
+ def htmlfoot
137
+ htmlout = ""
138
+ htmlout << "</BODY>\n"
139
+ htmlout << "</HTML>\n"
140
+ return htmlout
141
+ end
142
+
143
+ def report(opts)
144
+ output=""
145
+ output << htmlhead
146
+ output << "<h2>dockscan Report<h2>"
147
+ issues = sortvulns
148
+ 7.downto(3) do |sev|
149
+ if issues.key?(sev)
150
+ output << "<h3>" << sev2word(sev) << " vulnerabilities</h3>\n"
151
+ issues[sev].each do |v|
152
+ if sev >= 7 then
153
+ output << '<TABLE CLASS="Vulnerability VulnerabilityBLACK">'
154
+ elsif sev == 6 then
155
+ output << '<TABLE CLASS="Vulnerability VulnerabilityRED">'
156
+ elsif sev == 5
157
+ output << '<TABLE CLASS="Vulnerability VulnerabilityORANGE">'
158
+ elsif sev == 4
159
+ output << '<TABLE CLASS="Vulnerability VulnerabilityYELLOW">'
160
+ elsif sev == 3
161
+ output << '<TABLE CLASS="Vulnerability VulnerabilityGRAY">'
162
+ else
163
+ output << '<TABLE CLASS="Vulnerability">'
164
+ end
165
+ output << "<TR>\n"
166
+ output << "<TD COLSPAN=2>" << v.vuln.title << "</TD>\n"
167
+ output << "<TD>" << sev2word(v.vuln.severity) << "</TD>\n"
168
+ output << "</TR>\n"
169
+ output << "<TR><TD COLSPAN=3>" << v.vuln.description << "</TD></TR>\n"
170
+ output << "<TR><TD COLSPAN=3><PRE>" << v.output << "</PRE></TD></TR>\n"
171
+ output << "<TR><TD COLSPAN=3>" << v.vuln.solution << "</TD></TR>\n"
172
+ output << "<TR><TD>CVSS: " << getkey(v.vuln.risk,"cvss").to_s << "</TD><TD>State: " << v.state << "</TD><TD></TD></TR>\n"
173
+ output << "</TABLE>\n\n"
174
+ end
175
+ output << "<P></P>\n"
176
+ end
177
+ end
178
+ return output
179
+
180
+ end
181
+ end
182
+
@@ -0,0 +1,36 @@
1
+ class ReportStdout < Dockscan::Modules::ReportModule
2
+
3
+ def info
4
+ return 'This plugin produces brief stdout reports'
5
+ end
6
+
7
+ def format
8
+ return "stdout"
9
+ end
10
+
11
+ def file_extension
12
+ return "-stdout.txt"
13
+ end
14
+
15
+ def report(opts)
16
+ output=""
17
+ output << "Dockscan Report\n\n"
18
+
19
+ issues = sortvulns
20
+ 7.downto(3) do |sev|
21
+ if issues.key?(sev)
22
+ output << sev2word(sev) << "\n"
23
+ issues[sev].each do |v|
24
+ # output << k << ": "
25
+ output << v.vuln.title << ": "
26
+ output << v.vuln.solution
27
+ # output << v.output
28
+ output << "\n"
29
+ end
30
+ output << "\n"
31
+ end
32
+ end
33
+ return output
34
+ end
35
+ end
36
+
@@ -0,0 +1,36 @@
1
+ class ReportText < Dockscan::Modules::ReportModule
2
+
3
+ def info
4
+ return 'This plugin produces text reports'
5
+ end
6
+
7
+ def format
8
+ return "txt"
9
+ end
10
+
11
+ def file_extension
12
+ return ".txt"
13
+ end
14
+
15
+ def report(opts)
16
+ output=""
17
+ output << "Dockscan Report\n\n"
18
+
19
+ issues = sortvulns
20
+ 7.downto(3) do |sev|
21
+ if issues.key?(sev)
22
+ output << "-[ " << sev2word(sev) << " ]-\n"
23
+ issues[sev].each do |v|
24
+ output << "=" << v.vuln.title << "=\n"
25
+ output << "Description:\n" << v.vuln.description << "\n"
26
+ output << "Output:\n" << v.output << "\n"
27
+ output << "Solution:\n" << v.vuln.solution << "\n"
28
+ output << "\n"
29
+ end
30
+ output << "\n"
31
+ end
32
+ end
33
+ return output
34
+ end
35
+ end
36
+
@@ -0,0 +1,75 @@
1
+ module Dockscan
2
+ module Modules
3
+
4
+ class ReportModule < GenericModule
5
+ attr_accessor :scandata
6
+
7
+ def info
8
+ raise "#{self.class.name} doesn't implement `handle_command`!"
9
+ end
10
+
11
+ def desc(item)
12
+ if item.vuln.description
13
+ return item.vuln.description
14
+ else
15
+ return false
16
+ end
17
+ end
18
+
19
+ def sortvulns
20
+ severity_sorted=Hash.new
21
+ scandata.each do |classname,scanissue|
22
+ if scanissue.state == "vulnerable" or scanissue.state=="info" then
23
+ severity_sorted[scanissue.vuln.severity] ||= []
24
+ severity_sorted[scanissue.vuln.severity] << scanissue
25
+ end
26
+ end
27
+ return severity_sorted
28
+ end
29
+
30
+ def format
31
+ return "unknown"
32
+ end
33
+
34
+ def file_extension
35
+ return ".unknown"
36
+ end
37
+
38
+ def getkey(hsh,hkey)
39
+ if hsh.has_key?(hkey) then
40
+ return hsh[hkey]
41
+ else
42
+ return ''
43
+ end
44
+ end
45
+
46
+ def sev2word(sev)
47
+ case sev
48
+ when 7
49
+ return "Critical"
50
+ when 6
51
+ return "High"
52
+ when 5
53
+ return "Medium"
54
+ when 4
55
+ return "Low"
56
+ when 3
57
+ return "Info"
58
+ when 2
59
+ return "Verbose"
60
+ when 1
61
+ return "Inspect"
62
+ when 0
63
+ return "Debug"
64
+ else
65
+ return "DebugMiss"
66
+ end
67
+ end
68
+
69
+ def report(opts)
70
+ raise "#{self.class.name} doesn't implement `handle_command`!"
71
+ end
72
+ end # Class
73
+
74
+ end # Module
75
+ end # Dockscan
@@ -0,0 +1,18 @@
1
+ module Dockscan
2
+ module Scan
3
+ class Issue
4
+ attr_accessor :severity, :description, :solution, :title, :tags, :references, :risk, :reflinks
5
+
6
+ def initialize
7
+ severity=0
8
+ description="Forgot to add description"
9
+ solution="Information only."
10
+ title="Forgot to add title"
11
+ tags=Hash.new
12
+ references=Hash.new
13
+ risk=Hash.new
14
+ reflinks=Hash.new
15
+ end
16
+ end # Issue
17
+ end # Scan
18
+ end # Dockscan
@@ -0,0 +1,172 @@
1
+ require 'docker'
2
+
3
+ require 'dockscan/modules/genmodule'
4
+ require 'dockscan/modules/discover'
5
+ require 'dockscan/modules/audit'
6
+ require 'dockscan/modules/report'
7
+
8
+ require 'dockscan/scan/plugin'
9
+
10
+ require 'pp'
11
+
12
+ module Dockscan
13
+ module Scan
14
+ class Manage
15
+
16
+ attr_accessor :log
17
+
18
+ def initialize
19
+ @auditoutput=Hash.new
20
+ @log=Logger.new MultiDelegator.delegate(:write, :close).to(STDERR)
21
+ end
22
+
23
+ def check_connection
24
+ @log.info("Validating version specified: "+Docker.url)
25
+ begin
26
+ Docker.validate_version!
27
+ rescue
28
+ @log.error("Error connecting or validating Docker version")
29
+ return false
30
+ end
31
+ return true
32
+ end
33
+
34
+ def scan (url, opts, logger)
35
+ failed=Array.new
36
+
37
+ if url != nil then
38
+ Docker.url = url
39
+ end
40
+
41
+ @log=logger
42
+
43
+ if not check_connection then
44
+ return
45
+ end
46
+
47
+ moduledirs=Array.new
48
+ if moduledirs.empty? then
49
+ moduledirs.push File.expand_path("../modules", File.dirname(__FILE__))
50
+ end
51
+
52
+ @log.info("Loading discovery modules...")
53
+ moduledirs.each do |moduledir|
54
+ @log.debug("Loading dir: #{moduledir}")
55
+ Dir["#{moduledir}/discover/*.rb"].each do |f|
56
+ @log.info("Loading discovery module: #{f}")
57
+ begin
58
+ require f
59
+ rescue SyntaxError => se
60
+ @log.info("Error loading audit module: #{f}")
61
+ @log.debug("Error executing audit module: #{se.backtrace}")
62
+ failed << f
63
+ end
64
+ end
65
+ end
66
+
67
+ @log.info("Running discovery modules...")
68
+ Dockscan::Modules::DiscoverModule.modules.each do |modclass|
69
+ @log.info("Running discovery module: #{modclass.name}")
70
+ begin
71
+ mod=modclass.new
72
+ mod.scandata=@auditoutput
73
+ @auditoutput[mod.class.name]=mod.run
74
+ rescue Exception => e
75
+ @log.info("Error executing audit module: #{modclass.name}")
76
+ @log.debug("Error executing audit module: #{e.backtrace}")
77
+ failed << modclass.name
78
+ end
79
+ # pp @auditoutput[mod.class.name]
80
+ end
81
+
82
+
83
+ @log.info("Loading audit modules...")
84
+ moduledirs.each do |moduledir|
85
+ @log.debug("Loading dir: #{moduledir}")
86
+ Dir["#{moduledir}/audit/*.rb"].each do |f|
87
+ @log.info("Loading audit module: #{f}")
88
+ begin
89
+ require f
90
+ rescue SyntaxError => se
91
+ @log.info("Error loading audit module: #{f}")
92
+ @log.debug("Error executing audit module: #{se.backtrace}")
93
+ failed << f
94
+ end
95
+ end
96
+ end
97
+
98
+ @log.info("Running audit modules...")
99
+ Dockscan::Modules::AuditModule.modules.each do |modclass|
100
+ @log.info("Running audit module: #{modclass.name}")
101
+ begin
102
+ mod=modclass.new
103
+ mod.scandata=@auditoutput
104
+ @auditoutput[mod.class.name]=mod.check('test')
105
+ rescue Exception => e
106
+ @log.info("Error executing audit module: #{modclass.name}")
107
+ @log.debug("Error executing audit module: #{e.to_s} #{e.backtrace}")
108
+ failed << modclass.name
109
+ end
110
+ # pp @auditoutput[mod.class.name]
111
+ end
112
+
113
+ @log.info("Loading report modules...")
114
+ moduledirs.each do |moduledir|
115
+ Dir["#{moduledir}/report/*.rb"].each do |f|
116
+ @log.info("Loading report #{f}")
117
+ begin
118
+ require f
119
+ rescue SyntaxError => se
120
+ @log.info("Error loading report module: #{f}")
121
+ @log.debug("Error executing report module: #{se.backtrace}")
122
+ failed << f
123
+ end
124
+ end
125
+ end
126
+
127
+ @log.info("Running report modules...")
128
+ Dockscan::Modules::ReportModule.modules.each do |modclass|
129
+ @log.info("Running report module: #{modclass.name}")
130
+ mod=modclass.new
131
+ if opts.key?("report") then
132
+ formats=opts["report"].split(",")
133
+ else
134
+ formats=['stdout']
135
+ end
136
+ formats.each do |fmt|
137
+ if fmt==mod.format then
138
+ begin
139
+ mod.scandata=@auditoutput
140
+ output=mod.report(nil)
141
+ if opts.key?("output") then
142
+ reportfilename = opts["output"]
143
+ reportfilename << mod.file_extension
144
+ File.open(reportfilename, 'w') { |file| file.write(output) }
145
+ else
146
+ puts output
147
+ end
148
+ rescue Exception => e
149
+ @log.info("Error executing report module: #{modclass.name}")
150
+ @log.debug("Error executing report module: #{e.to_s} #{e.backtrace}")
151
+ failed << modclass.name
152
+ end
153
+ @log.debug(output)
154
+ else
155
+ @log.debug("Skipping report module: #{modclass.name}");
156
+ end
157
+ end
158
+ end
159
+
160
+ if failed.count > 0
161
+ failedstr=""
162
+ failed.each do |f|
163
+ failedstr = f + " "
164
+ end
165
+ @log.warn("Following modules failed: #{failedstr}")
166
+ end
167
+ end
168
+
169
+ end # Class
170
+ end # Scan
171
+ end # Dockscan
172
+
@@ -0,0 +1,14 @@
1
+ require 'dockscan/scan/issue'
2
+
3
+ module Dockscan
4
+ module Scan
5
+ class Plugin
6
+ attr_accessor :state, :output, :obj, :vuln, :cname
7
+
8
+ def initialize
9
+ state="untested"
10
+ vuln = Dockscan::Scan::Issue.new
11
+ end
12
+ end # Plugin
13
+ end # Scan
14
+ end # Dockscan
@@ -0,0 +1,3 @@
1
+ module Dockscan
2
+ VERSION = "0.1.0"
3
+ end
data/lib/dockscan.rb ADDED
@@ -0,0 +1,7 @@
1
+ require "dockscan/version"
2
+
3
+ require "dockscan/scan/manage"
4
+
5
+ module Dockscan
6
+ # Your code goes here...
7
+ end