etch

package module
v0.6.7 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Jan 28, 2018 License: BSD-3-Clause Imports: 7 Imported by: 0

README

Etch

Go Report Card

Package etch provides a simple facility to write graphical regression tests. The Assert function handles the common case. Give it the test variable, the images you have and want, and it will fail your case if they differ.

Synopsis

have = image.NewRGBA(r)
want = image.NewRGBA(r)

// fail the test if the images differ, and write the delta to a png
etch.Assert(t, have, want, "Test.png")

Visualize

Optionally, provide a filename to store the graphical difference as an uncompressed PNG if the test fails.

paint

The Extra data in the image (have but don't want) is represented in Red. The Missing data (want, but dont have) is represented in Blue. These can be changed by modifying Extra and Missing package variables

Example

I observed a bug in A where the text on the last line wasn't cleaned up unless that last line ended in a newline character. This means if the frame displays ^a\nb\nc$ and b\n is deleted, the user would see ^a\nc\nc$. Nasty.

We can programatically check for any defect as long as we know how to reproduce it.

Step 1: Find Reproduction and Expected Result

Find the reproduction. In this case I also found steps that generate the expected result. You can also use a cached expected result from a previously known good configuration.

  • Insert the multi-line text containing no trailing newline (good: insert a trailing newline)

paint

  • Select any line but the last

paint

  • Delete the selection

paint

Above you can see the result of the middle line's deletion for both sessions. The window that did not have the trailing newline did not clean up the last line after copying it up toward the top of the frame

Step 2: Create Images

Create two images

	have = image.NewRGBA(r)
	want = image.NewRGBA(r)

Now for the test case specific stuff. Your steps will replace mine below depending on what you're actually doing to the images. In my case the frame draws on them directly, so we really don't care about its inner workings too much, just that there's a bug and we're going to test for its existence using these two images: have and want.

	// Create two frames
	h = New(r, font.NewBasic(fsize), have, A)
	w = New(r, font.NewBasic(fsize), want, A)
	
	// Insert some text with and without trailing newlines
	w.Insert([]byte("1234\ncccc\ndddd\n"), 0)
	h.Insert([]byte("1234\ncccc\ndddd"), 0)
	
	// Delete the second line
	h.Delete(5, 10)
	w.Delete(5, 10)

By this point, want will be an image with the defect-free state and have will be an image with the defective state

	etch.Assert(t, have, want, "TestDeleteLastLineNoNL.png")

Step 4: Go Test

We run go test

--- FAIL: TestDeleteLastLineNoNL (0.03s)
	etch.go:64: delta: TestDeleteLastLineNoNL.png
FAIL
exit status 1
FAIL	github.com/as/frame	0.125s

We can look at the image to see what went wrong: TestDeleteLastLineNoNL.png

paint

Although it looks obvious, remember that this test would fail if any of the pixels differ. It's not easy to compare images visually, and you shouldn't avoid automating tests for it. Automating the tests helps prevent regressions from going undetected and speeds up the edit/compile/test cycle.

Step 5: Apply the Fix

	f.Draw(f.b, image.Rect(pt0.X, pt0.Y, pt0.X+(f.r.Max.X-pt1.X), q0), f.b, pt1, f.op)
	f.Draw(f.b, image.Rect(f.r.Min.X, q0, f.r.Max.X, q0+(q2-q1)), f.b, image.Pt(f.r.Min.X, q1), f.op)
	// f.Paint(image.Pt(pt2.X, pt2.Y-(pt1.Y-pt0.Y)), pt2, f.Color.Back)

The bug is the commented line above. Once the comment is removed, the test passes. Because go test can be run automatically on file changes, this eliminates the manual step of checking the image. The test passes once have and want are the same, and when they're not, just open the delta in an image viewer to see what went wrong.

Documentation

Overview

Package etch provides a simple facility to write graphical regression tests. The Assertf function provides the common case functionality. Provide it the test variable, along the image you have and want, and it will fail your case if want != have.

Optionally, provide a filename to store the graphical difference as an uncompressed PNG if the test fails.

The Extra data in the image (have but don't want) is represented in Red The Missing data (want, but dont have) is represented in Blue These can be changed by modifying Extra and Missing package variables

To simplify the package, the alpha channel is ignored. A color triplet is equal to another if it's R,G,B values are identical.

The foreground variable, fg, is what to paint on the delta image if two pixels match The background variable, BG, is the common background color between two images

If two pixels at the same (x,y) coordinate don't match, the ambiguity is resolved by comparing the image you have's color value at that coordinate to the background color. If the color matches, the pixel you have is missing. Otherwise, it's extra.

This may seem confusing, so a worked example is made available in the README

Index

Constants

This section is empty.

Variables

View Source
var (

	// Colors from as/frame
	Red   = image.NewUniform(color.RGBA{255, 0, 0, 255})
	Blue  = image.NewUniform(color.RGBA{0, 0, 255, 255})
	Black = image.NewUniform(color.RGBA{0, 0, 0, 255})
	White = image.NewUniform(color.RGBA{255, 255, 255, 255})
	Gray  = image.NewUniform(color.RGBA{33, 33, 33, 255})
	Peach = image.NewUniform(color.RGBA{255, 248, 232, 255})

	// Defaults used by this package
	// BG should be the similar background color between two images
	BG      = Peach
	Extra   = Red
	Missing = Blue
)

Functions

func Assert

func Assert(t *testing.T, have, want image.Image, filename string)

Assert compares two test images and fails the provided test if the images differ at any pixel(x,y). It saves the delta as a png to the given filename (if set) and provides the path to that image in an error string upon failure.

func AssertFile

func AssertFile(t *testing.T, have image.Image, wantfile string, filename string)

AssertFile is like assert, but reads the wanted result from the named file

func Assertf

func Assertf(t *testing.T, have, want image.Image, filename string, fm string, i ...interface{})

Assertf is like assert, except it logs a custom message with a format string and interface parameter list (like fmt.Printf)

func Delta

func Delta(a, b image.Image) (delta *image.RGBA, ok bool)

Delta computes a difference between image a and b by comparing each pixel to the fg and BG colors. If a pixel in a and b are equal, the delta pixel is fg. Otherwise the pixel is either red or blue depending if its extra or missing respectively.

func EqualRGB

func EqualRGB(c0, c1 color.Color) bool

EqualRGB returns true if and only if the two colors share the same RGB triplets

func ReadFile

func ReadFile(t *testing.T, file string) (img image.Image)

ReadFile reads in the named file and returns it as an image.Image. The supported format is an uncompressed PNG.

func Report

func Report(have, want, delta image.Image) image.Image

Report generates a visual summary of the actual (have) and expected (want) results, alongside the delta image. See Delta for details on the delta image format.

func WriteFile

func WriteFile(t *testing.T, file string, img image.Image)

WriteFile writes the input img to the names file and fails the test.

Types

This section is empty.

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL