-
-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathcmd_ssl_expiry.go
155 lines (123 loc) · 3.14 KB
/
cmd_ssl_expiry.go
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
144
145
146
147
148
149
150
151
152
153
154
155
package main
import (
"crypto/tls"
"flag"
"fmt"
"net"
"regexp"
"strings"
"time"
)
// SSLExpiryCommand is the structure for our options and state.
type SSLExpiryCommand struct {
// Show time in hours (only)
hours bool
// Show time in days (only)
days bool
}
// Arguments adds per-command args to the object.
func (s *SSLExpiryCommand) Arguments(f *flag.FlagSet) {
f.BoolVar(&s.hours, "hours", false, "Report only the number of hours until the certificate expires")
f.BoolVar(&s.days, "days", false, "Report only the number of days until the certificate expires")
}
// Info returns the name of this subcommand.
func (s *SSLExpiryCommand) Info() (string, string) {
return "ssl-expiry",
`Report how long until an SSL certificate expires.
Details:
This sub-command shows the number of hours/days until the SSL
certificate presented upon a remote host expires. The value
displayed is the minimum expiration time of the certificate and
any bundled-chains served with it.
Examples:
Report on an SSL certificate:
$ gobox ssl-expiry https://example.com/
$ gobox ssl-expiry example.com
Report on an SMTP-certificate:
$ gobox ssl-expiry smtp.gmail.com:465
`
}
// Execute runs our sub-command.
func (s *SSLExpiryCommand) Execute(args []string) int {
ret := 0
//
// Ensure we have an argument
//
if len(args) < 1 {
fmt.Printf("You must specify the host(s) to test.\n")
return 1
}
// For each argument
for _, arg := range args {
hours, err := s.SSLExpiration(arg)
if err != nil {
fmt.Printf("\tERROR:%s\n", err.Error())
ret = 1
} else {
// Output for scripting
if s.hours {
fmt.Printf("%s\t%d\thours\n", arg, hours)
}
if s.days {
fmt.Printf("%s\t%d\tdays\n", arg, hours/24)
}
// Output for humans
if !s.hours && !s.days {
fmt.Printf("%s\t%d hours\t%d days\n", arg, hours, hours/24)
}
}
}
return ret
}
// SSLExpiration returns the number of hours remaining for a given
// SSL certificate chain.
func (s *SSLExpiryCommand) SSLExpiration(host string) (int64, error) {
// Expiry time, in hours
var hours int64
hours = -1
//
// If the string matches http[s]://, then strip it off
//
re, err := regexp.Compile(`^https?:\/\/([^\/]+)`)
if err != nil {
return 0, err
}
res := re.FindAllStringSubmatch(host, -1)
for _, v := range res {
host = v[1]
}
//
// If no port is specified default to :443
//
p := strings.Index(host, ":")
if p == -1 {
host += ":443"
}
//
// Connect, with sane timeout
//
conn, err := tls.DialWithDialer(&net.Dialer{Timeout: time.Second * 2}, "tcp", host, nil)
if err != nil {
return 0, err
}
defer conn.Close()
timeNow := time.Now()
for _, chain := range conn.ConnectionState().VerifiedChains {
for _, cert := range chain {
// Get the expiration time, in hours.
expiresIn := int64(cert.NotAfter.Sub(timeNow).Hours())
// If we've not checked anything this is the benchmark
if hours == -1 {
hours = expiresIn
} else {
// Otherwise replace our result if the
// certificate is going to expire more
// recently than the current "winner".
if expiresIn < hours {
hours = expiresIn
}
}
}
}
return hours, nil
}