185 lines
4.2 KiB
Go
185 lines
4.2 KiB
Go
// Package lib contains functions useful for most programs.
|
|
package lib
|
|
|
|
import (
|
|
"crypto/x509"
|
|
"encoding/pem"
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
)
|
|
|
|
var progname = filepath.Base(os.Args[0])
|
|
|
|
// ProgName returns what lib thinks the program name is, namely the
|
|
// basename of of argv0.
|
|
//
|
|
// It is similar to the Linux __progname function.
|
|
func ProgName() string {
|
|
return progname
|
|
}
|
|
|
|
// Warnx displays a formatted error message to standard error, à la
|
|
// warnx(3).
|
|
func Warnx(format string, a ...interface{}) (int, error) {
|
|
format = fmt.Sprintf("[%s] %s", progname, format)
|
|
format += "\n"
|
|
return fmt.Fprintf(os.Stderr, format, a...)
|
|
}
|
|
|
|
// Warn displays a formatted error message to standard output,
|
|
// appending the error string, à la warn(3).
|
|
func Warn(err error, format string, a ...interface{}) (int, error) {
|
|
format = fmt.Sprintf("[%s] %s", progname, format)
|
|
format += ": %v\n"
|
|
a = append(a, err)
|
|
return fmt.Fprintf(os.Stderr, format, a...)
|
|
}
|
|
|
|
// Errx displays a formatted error message to standard error and exits
|
|
// with the status code from `exit`, à la errx(3).
|
|
func Errx(exit int, format string, a ...interface{}) {
|
|
format = fmt.Sprintf("[%s] %s", progname, format)
|
|
format += "\n"
|
|
fmt.Fprintf(os.Stderr, format, a...)
|
|
os.Exit(exit)
|
|
}
|
|
|
|
// Err displays a formatting error message to standard error,
|
|
// appending the error string, and exits with the status code from
|
|
// `exit`, à la err(3).
|
|
func Err(exit int, err error, format string, a ...interface{}) {
|
|
format = fmt.Sprintf("[%s] %s", progname, format)
|
|
format += ": %v\n"
|
|
a = append(a, err)
|
|
fmt.Fprintf(os.Stderr, format, a...)
|
|
os.Exit(exit)
|
|
}
|
|
|
|
// Itoa provides cheap integer to fixed-width decimal ASCII. Give a
|
|
// negative width to avoid zero-padding. Adapted from the 'itoa'
|
|
// function in the log/log.go file in the standard library.
|
|
func Itoa(i int, wid int) string {
|
|
// Assemble decimal in reverse order.
|
|
var b [20]byte
|
|
bp := len(b) - 1
|
|
for i >= 10 || wid > 1 {
|
|
wid--
|
|
q := i / 10
|
|
b[bp] = byte('0' + i - q*10)
|
|
bp--
|
|
i = q
|
|
}
|
|
// i < 10
|
|
b[bp] = byte('0' + i)
|
|
return string(b[bp:])
|
|
}
|
|
|
|
var (
|
|
dayDuration = 24 * time.Hour
|
|
yearDuration = (365 * dayDuration) + (6 * time.Hour)
|
|
)
|
|
|
|
// Duration returns a prettier string for time.Durations.
|
|
func Duration(d time.Duration) string {
|
|
var s string
|
|
if d >= yearDuration {
|
|
years := d / yearDuration
|
|
s += fmt.Sprintf("%dy", years)
|
|
d -= years * yearDuration
|
|
}
|
|
|
|
if d >= dayDuration {
|
|
days := d / dayDuration
|
|
s += fmt.Sprintf("%dd", days)
|
|
}
|
|
|
|
if s != "" {
|
|
return s
|
|
}
|
|
|
|
d %= 1 * time.Second
|
|
hours := d / time.Hour
|
|
d -= hours * time.Hour
|
|
s += fmt.Sprintf("%dh%s", hours, d)
|
|
return s
|
|
}
|
|
|
|
// ReadCertificate reads a DER or PEM-encoded certificate from the
|
|
// byte slice.
|
|
func ReadCertificate(in []byte) (cert *x509.Certificate, rest []byte, err error) {
|
|
if len(in) == 0 {
|
|
err = errors.New("lib: empty certificate")
|
|
return
|
|
}
|
|
|
|
if in[0] == '-' {
|
|
p, remaining := pem.Decode(in)
|
|
if p == nil {
|
|
err = errors.New("lib: invalid PEM file")
|
|
return
|
|
}
|
|
|
|
rest = remaining
|
|
if p.Type != "CERTIFICATE" {
|
|
err = fmt.Errorf("lib: expected a CERTIFICATE PEM file, but have %s", p.Type)
|
|
return
|
|
}
|
|
|
|
in = p.Bytes
|
|
}
|
|
|
|
cert, err = x509.ParseCertificate(in)
|
|
return
|
|
}
|
|
|
|
// ReadCertificates tries to read all the certificates in a
|
|
// PEM-encoded collection.
|
|
func ReadCertificates(in []byte) (certs []*x509.Certificate, err error) {
|
|
var cert *x509.Certificate
|
|
for {
|
|
cert, in, err = ReadCertificate(in)
|
|
if err != nil {
|
|
break
|
|
}
|
|
|
|
if cert == nil {
|
|
break
|
|
}
|
|
|
|
certs = append(certs, cert)
|
|
if len(in) == 0 {
|
|
break
|
|
}
|
|
}
|
|
|
|
return certs, err
|
|
}
|
|
|
|
// LoadCertificate tries to read a single certificate from disk. If
|
|
// the file contains multiple certificates (e.g. a chain), only the
|
|
// first certificate is returned.
|
|
func LoadCertificate(path string) (*x509.Certificate, error) {
|
|
in, err := ioutil.ReadFile(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cert, _, err := ReadCertificate(in)
|
|
return cert, err
|
|
}
|
|
|
|
// LoadCertificates tries to read all the certificates in a file,
|
|
// returning them in the order that it found them in the file.
|
|
func LoadCertificates(path string) ([]*x509.Certificate, error) {
|
|
in, err := ioutil.ReadFile(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return ReadCertificates(in)
|
|
}
|