Releases: google/go-cmp
v0.5.0
The most notable changes in this release are improvements to the reporter output (i.e., the implementation that produces output for Diff
). The changes seek to improve the signal-to-noise ratio such that differences of interest to the user are emphasized, while information that is less relevant are de-emphasized.
The reporter largely consists of a set of heuristics to determine what would be the best way to format the difference in the common case. If the output seems sub-par, please file an issue. The reports are highly valuable in improving its output (e.g., #195, #194, #202, #185).
Reporter changes:
- (#201) Do not use custom format for nil slice. Previously, the reporter had a bug (#157) where it would fail to print the difference between empty slices that were nil versus non-nil, which is now fixed by this change.
- (#212) Use custom triple-quote syntax for diffing string literals. This adds another way that strings are displayed. Previously, the reporter used a
strings.Join({...}, "\n")
syntax to show diffs between lines within a string. While this representation is unambiguous, the need to escape every line made the output visually distracting. Now, we add a new syntax where multiple lines are represented by a literal syntax using"""
as delimiters. When possible, this syntax is preferred if it can unambiguously represent the difference. - (#208) Batch reporter output for simple lists of text elements. Previously, when formatting a slice of primitives, the reporter would print each element on a new line, causing the output to be long. Now, multiple elements are batched together into a single line to keep the output more dense.
- (#210) Allow batched diffing of slices with a custom comparer. As a performance optimization, a user may choose to pass
cmp.Comparer(bytes.Equal)
so that large byte slices are compared using an efficient implementation. Previously, this would prevent the reporter from being able to use specialized logic to show the per-element difference between these two slices if they were different, but now it is able to. - (#213, #215) Limit verbosity of reporter output. For slices, maps, and structs, the reporter now imposes a limit on the number of elements, entries, and structs that it will print. The limit is chosen based on heuristics such as the depth of the tree and also whether the node represents an equal value or not, where inequal values are given a larger verbosity budget.
- (#216) Disambiguate reporter output. Previously, there were certain edge cases where the reporter output failed to show a difference between two values that the comparer determined to be different. Now, the reporter works harder to ensure that the output is guaranteed to be different if a semantic difference is there. For example, it may try increasing the verbosity limit, printing pointer addresses, avoid calling the
String
method, using fully qualified type names, or some combination of the above. - (#217) Improve reporting of values with cycles. Previously, the reporter would simply truncate the result if it ever detected a cycle. While this prevents a stack overflow trying to print a graph, it failed to properly show the topology of the graph. Now, the reporter additionally provides reference markers so that the user can visually identify where a pointer refers to in the output.
- (1776240) Forcibly export fields for use by the reporter. This allows the reporter to use the
String
orError
method (if available) to format values within unexported fields. This occurs regardless of whether anycmp.Exporter
orcmp.AllowUnexported
options are used or not. - (#210) Use raw string literal syntax only for valid UTF-8. This fixes a minor bug in the reporter to ensure that the output is always valid UTF-8.
Feature changes:
- (#203) Permit use of IgnoreFields with unexported fields. The
cmpopts.IgnoreFields
option now accepts unexported field names. Unlike exported fields, unexported field must be explicitly specified as they do not respect forwarding due to struct embedding.
Comparer changes:
- (#204) Optimize Diff for frequent equality. The
cmp
package is intended for use primarily within tests where the expected outcome is generally equality. Previously, even if the result is equal,cmp.Diff
would construct an expensive diff tree only to discard the result since the values are equal. Now, optimize for the common case by first checking whether the values are equal first, and only construct a diff if they are not. - (#214) Introduce deliberate instability to difference output. The internal algorithm used for diffing elements of a slice is sub-optimal. It does not produce an optimal edit script (i.e., one with the fewest possible reported differences), but is guaranteed to run in O(n). To keep the future open for algorithmic improvements, introduce some degree of deliberate instability so that users do not accidentally rely on it's output being stable.
- (#206) Avoid leaking implementation details of the exporter. In order for the current implementation to forcibly access unexported fields using
unsafe
, it requires that the parent struct be addressable. In Go, a struct field is only addressable if and only if the parent struct is addressable. To avoid leaking this internal implementation detail, we shallow copy the result to remove properties of addressability.
v0.4.1
v0.4.0
Added features:
- (#85) The comparison logic now has cycle detection and can now handle graphs.
- (#176) The
cmp.Exporter
option provides finer-grained control over exactly what types to permit unexported field access on. - (#158)
cmpopts.EquateApproxTime
provides the ability to compare timestamps with looser precision. - (#178)
cmpopts.EquateErrors
provides the ability to compare error values using the newerrors.Is
semantics from Go 1.13.
Bug fixes:
- (#169) Use of
unsafe.Pointer
has been fixed to comply with stricter pointer rules for Go 1.14.
v0.3.1
Minor reporter fixes:
- (#142) Fixed reporter to invoke the
String
method when formatting map keys. - (#148) Fixed reporter to properly elide a struct field on the right side if it is a zero value.
- (#147) Fixed reporter to avoid diffing by lines if any line exceeds a certain maximum length.
- (#152) Fixed reporter to properly treat -0.0 as not being the zero value.
v0.3.0
Added features:
- (#108) The
_
field is always ignored when comparing a struct. - (#109) The naming rules for
cmp.Transformer
is relaxed and clearly specified. - (#82)
cmpopts.AcyclicTransformer
makes it easier to write a transformer that avoids infinite recursion when the output type contains the input type. - (#126)
cmpopts.IgnoreSliceElements
andcmpopts.IgnoreMapEntries
provide the ability to ignore specific slice elements or map entries. - (#123)
cmp.Reporter
provides users the ability to add custom difference reporters. - (#119)
cmp.PathStep.Values
provide the currentreflect.Value
for thex
andy
arguments at that given path step.
Bug fixes:
- (#87) Fixed proper comparing of slices where one is a sub-slice of the other.
Reporter changes:
- (#124) The
cmp.Diff
output has been completely re-written to provide a unified diff of the two Go objects as a literal in pseudo-Go syntax (example). - (#131) The default reporter has built-in heuristics to detect whether a string or a byte-slice looks like binary data or human readable text, and can provide more humanly readable differences in certain use-cases. For example, a string may be detecting as containing multiple lines of text, for which it will print the difference based on the lines.
v0.2.0
Added Features:
-
Add implicit filter to transformers where a
Transformer
can only apply if that specific transformer does not already exist within the tail of the currentPath
since the last non-transform step. This filter makes it easier to create same-type transformations (e.g.,func(T) T
) without needing to manually add your own filter. (#29) -
Add
Path.Index
method, which provides an easier way to index aPath
without needing to manually perform bounds checking. (#56) -
Add
Transform.Option
method, which returns the originalTransformer
option. This enables the creation of non-reentrant transformers. (#59) -
Add handling for
purego
build tag, which prevents use ofunsafe
, which is necessary to implementAllowUnexported
. However,unsafe
is not available in some build environments such as GopherJS and Google AppEngine Standard. A future release will remove special casing for these two environments. (#68)
Reporting changes:
Diff
only batches multiple differences together for slices of primitives. (#45)Diff
prints uses as
prefix to indicate that theString
method was used. (#46)Diff
prints strings using the raw literal syntax when helpful. (#46)Diff
reports the type of a primitive when helpful. (#65)Path.GoString
elides printing type assertions on anonymous types. (#21)
Bug fixes:
v0.1.0
Package cmp
is intended to be a more powerful and safer alternative to
reflect.DeepEqual
for comparing whether two values are semantically equal.
The primary features of cmp
are:
-
When the default behavior of equality does not suit the needs of the test,
custom equality functions can override the equality operation.
For example, an equality function may report floats as equal so long as they
are within some tolerance of each other. -
Types that have an
Equal
method may use that method to determine equality.
This allows package authors to determine the equality operation for the types
that they define. -
If no custom equality functions are used and no
Equal
method is defined,
equality is determined by recursively comparing the primitive kinds on both
values, much likereflect.DeepEqual
. Unlikereflect.DeepEqual
, unexported
fields are not compared by default; they result in panics unless suppressed
by using anIgnore
option (seecmpopts.IgnoreUnexported
) or explictly
compared using theAllowUnexported
option.
func Equal(x, y interface{}, opts ...Option) bool { ... }
func Diff(x, y interface{}, opts ...Option) string { ... }
// Section: 1. Configuration options for Equal and Diff
type Option interface{ ... }
func AllowUnexported(types ...interface{}) Option { ... }
type Options []Option
// Section: 1.1. Fundamental options to customize comparison of values
func Ignore() Option { ... }
func Comparer(f func(T, T) bool) Option { ... }
func Transformer(name string, f func(T) R) Option { ... }
// Section: 1.2. Filter options to control the scope of fundamental options
func FilterPath(f func(Path) bool, opt Option) Option { ... }
func FilterValues(f func(T, T) bool, opt Option) Option { ... }
// Section: 2. Path to a node in the value tree
type Path []PathStep
type PathStep interface{ ... }
// Section: 2.1. Individual steps that comprise a Path
type StructField interface{ ... }
type SliceIndex interface{ ... }
type MapIndex interface{ ... }
type Indirect interface{ ... }
type TypeAssertion interface{ ... }
type Transform interface{ ... }
Package cmpopts
provides helper functions for creating cmp.Option
values
to configure comparisons for common use-cases.
Package API:
// Section: Comparers
func EquateApprox(fraction, margin float64) cmp.Option { ... }
func EquateEmpty() cmp.Option { ... }
func EquateNaNs() cmp.Option { ... }
// Section: Ignorers
func IgnoreFields(typ interface{}, names ...string) cmp.Option { ... }
func IgnoreInterfaces(ifaces interface{}) cmp.Option { ... }
func IgnoreTypes(typs ...interface{}) cmp.Option { ... }
func IgnoreUnexported(typs ...interface{}) cmp.Option { ... }
// Section: Transformers
func SortMaps(less func(T, T) bool) cmp.Option { ... }
func SortSlices(less func(T, T) bool) cmp.Option { ... }