Asim Shankar | dc150cd | 2014-08-08 10:59:39 -0700 | [diff] [blame] | 1 | // Package audit provides mechanisms to write method invocations to an audit log. |
Asim Shankar | f9f84bc | 2014-08-05 10:46:40 -0700 | [diff] [blame] | 2 | // |
Asim Shankar | 220a015 | 2014-10-30 21:21:09 -0700 | [diff] [blame] | 3 | // Typical use would be for tracking sensitive operations like private key usage (NewPrincipal), |
Asim Shankar | f9f84bc | 2014-08-05 10:46:40 -0700 | [diff] [blame] | 4 | // or sensitive RPC method invocations. |
Asim Shankar | dc150cd | 2014-08-08 10:59:39 -0700 | [diff] [blame] | 5 | package audit |
Asim Shankar | f9f84bc | 2014-08-05 10:46:40 -0700 | [diff] [blame] | 6 | |
Asim Shankar | 1ac8061 | 2014-08-08 16:04:09 -0700 | [diff] [blame] | 7 | import ( |
| 8 | "fmt" |
| 9 | "strings" |
| 10 | "time" |
| 11 | ) |
Asim Shankar | f9f84bc | 2014-08-05 10:46:40 -0700 | [diff] [blame] | 12 | |
| 13 | // Auditor is the interface for writing auditable events. |
| 14 | type Auditor interface { |
| 15 | Audit(entry Entry) error |
| 16 | } |
| 17 | |
| 18 | // Entry is the information logged on each auditable event. |
| 19 | type Entry struct { |
| 20 | // Method being invoked. |
| 21 | Method string |
| 22 | // Arguments to the method. |
| 23 | // Any sensitive data in the arguments should not be included, |
| 24 | // even if the argument was provided to the real method invocation. |
| 25 | Arguments []interface{} |
| 26 | // Result of the method invocation. |
| 27 | // A common use case is to audit only successful method invocations. |
| 28 | Results []interface{} |
| 29 | |
| 30 | // Timestamp of method invocation. |
| 31 | Timestamp time.Time |
| 32 | } |
Asim Shankar | 1ac8061 | 2014-08-08 16:04:09 -0700 | [diff] [blame] | 33 | |
| 34 | func (e Entry) String() string { |
| 35 | return fmt.Sprintf("%v: %s(%s)%s", e.Timestamp.Format(time.RFC3339), e.Method, join(e.Arguments, "", ""), join(e.Results, " = (", ")")) |
| 36 | } |
| 37 | |
| 38 | func join(elems []interface{}, prefix, suffix string) string { |
| 39 | switch len(elems) { |
| 40 | case 0: |
| 41 | return "" |
| 42 | case 1: |
| 43 | return fmt.Sprintf("%s%v%s", prefix, elems[0], suffix) |
| 44 | } |
| 45 | strs := make([]string, len(elems)) |
| 46 | for i, e := range elems { |
| 47 | strs[i] = fmt.Sprintf("%v", e) |
| 48 | } |
| 49 | return prefix + strings.Join(strs, ", ") + suffix |
| 50 | } |