|
| 1 | +<div align="center" style="margin-bottom:20px"> |
| 2 | + <!-- <img src=".assets/banner.png" alt="go-logspy" /> --> |
| 3 | + <div align="center"> |
| 4 | + <a href="https://github.com/blugnu/go-logspy/actions/workflows/qa.yml"><img alt="build-status" src="https://github.com/blugnu/go-logspy/actions/workflows/qa.yml/badge.svg?branch=master&style=flat-square"/></a> |
| 5 | + <a href="https://goreportcard.com/report/github.com/blugnu/go-logspy" ><img alt="go report" src="https://goreportcard.com/badge/github.com/blugnu/go-logspy"/></a> |
| 6 | + <a><img alt="go version >= 1.14" src="https://img.shields.io/github/go-mod/go-version/blugnu/go-logspy?style=flat-square"/></a> |
| 7 | + <a href="https://github.com/blugnu/go-logspy/blob/master/LICENSE"><img alt="MIT License" src="https://img.shields.io/github/license/blugnu/go-logspy?color=%234275f5&style=flat-square"/></a> |
| 8 | + <a href="https://coveralls.io/github/blugnu/go-logspy?branch=master"><img alt="coverage" src="https://img.shields.io/coveralls/github/blugnu/go-logspy?style=flat-square"/></a> |
| 9 | + <a href="https://pkg.go.dev/github.com/blugnu/go-logspy"><img alt="docs" src="https://pkg.go.dev/badge/github.com/blugnu/go-logspy"/></a> |
| 10 | + </div> |
| 11 | +</div> |
| 12 | + |
| 13 | +<br> |
| 14 | + |
| 15 | +# go-logspy |
| 16 | + |
| 17 | +A truly trivial package for assisting in incorporating log output verification in `go` unit tests. |
| 18 | + |
| 19 | +LogSpy works with any log package that provides a mechanism for redirecting log output to an `io.Writer`, such as the built-in `log` package or `logrus`. |
| 20 | + |
| 21 | +## How LogSpy Works |
| 22 | +Log output is redirected into a `bytes.Buffer` (the "sink"). |
| 23 | + |
| 24 | +After code under test has been executed, the contents of the log are tested using either normal string testing techniques, or any of the various helper methods provided (intended to simplify common log tests). |
| 25 | + |
| 26 | +That's it. |
| 27 | + |
| 28 | +<br> |
| 29 | +<hr> |
| 30 | +<br> |
| 31 | + |
| 32 | +## How to Use LogSpy |
| 33 | +LogSpy is as trivial to use as it is to understand: |
| 34 | + |
| 35 | +1. Configure and sink the log |
| 36 | +2. `Reset()` LogSpy before each test |
| 37 | +3. Execute code to be tested |
| 38 | +4. Test log content with helpers (or your preferred string testing techniques) |
| 39 | + |
| 40 | +<br> |
| 41 | + |
| 42 | +### **1. Configure and Sink the Log** |
| 43 | +In your tests, redirect log output to the `logspy.Sink()`. |
| 44 | + |
| 45 | +When using a package such as `logrus`, this is typically combined with configuring your log package for test runs in the same way that it is configured for normal execution of the code under test. |
| 46 | + |
| 47 | +If using `go test`, a good place for this could be a `TestMain`: |
| 48 | + |
| 49 | +```golang |
| 50 | +func TestMain(m *testing.M) { |
| 51 | + // Confgure the log formatter |
| 52 | + // (TIP: Use a common func to ensure identical formatting |
| 53 | + // in both test and application logging) |
| 54 | + logrus.SetFormatter(&logrus.JSONFormatter{}) |
| 55 | + |
| 56 | + // Redirect log output to the logspy sink |
| 57 | + logrus.SetOutput(logspy.Sink()) |
| 58 | + |
| 59 | + // Run the tests and set the exit value to the result |
| 60 | + os.Exit(m.Run()) |
| 61 | +} |
| 62 | +``` |
| 63 | + |
| 64 | +**NOTE:** *Since `go test` runs tests at the package level, it is necessary to configure and sink the log in each package; a common func called from `TestMain()` in each package might be one way to achieve this.* |
| 65 | + |
| 66 | +<br> |
| 67 | + |
| 68 | +### **2. `Reset()` Logs Before Each Test (ARRANGE)** |
| 69 | +With log formatting and sink in place, you then need to ensure you `Reset()` LogSpy at the beginning of each test: |
| 70 | + |
| 71 | +```golang |
| 72 | +func TestThatSomeFunctionEmitsExpectedLogs(t *testing.T) { |
| 73 | + // ARRANGE |
| 74 | + logspy.Reset() |
| 75 | + |
| 76 | + // ACT |
| 77 | + SomeFunction() |
| 78 | + |
| 79 | + // ASSERT |
| 80 | + ... |
| 81 | +} |
| 82 | +``` |
| 83 | + |
| 84 | +This is important as it ensures that log output captured in one test does not "pollute" the log in others. |
| 85 | + |
| 86 | +<br> |
| 87 | + |
| 88 | +### **3. Execute Code Under Test (ACT)** |
| 89 | +This doesn't need any explanation, right? |
| 90 | + |
| 91 | +<br> |
| 92 | + |
| 93 | +### **4. Test Log Content (ASSERT)** |
| 94 | +The raw log output captured by LogSpy is available using the `logspy.String()` function. Using this, the log content can be tested using whatever techniques you prefer when testing string values. |
| 95 | + |
| 96 | +However, LogSpy also provides a number of helper functions to simplify common tests of log content. |
| 97 | + |
| 98 | +For example, to test that an expected number of log entries have been emitted, the `NumEntries()` function can be used, which returns the number of non-empty log entries (since the most recent `Reset()`). |
| 99 | + |
| 100 | +An example showing this in use: |
| 101 | + |
| 102 | +```golang |
| 103 | +func TestThatNumEntriesReturnsTheNumberOfLogEntries(t *testing.T) { |
| 104 | + // ARRANGE |
| 105 | + logspy.Reset() |
| 106 | + |
| 107 | + // ACT |
| 108 | + log.Println("output 1") |
| 109 | + log.Println("output 2") |
| 110 | + log.Println("output 3") |
| 111 | + |
| 112 | + // ASSERT |
| 113 | + wanted := 3 |
| 114 | + got := logspy.NumEntries() |
| 115 | + if wanted != got { |
| 116 | + t.Errorf("Wanted %d log entries, got %d", wanted, got) |
| 117 | + } |
| 118 | +} |
| 119 | +``` |
0 commit comments