suite

package
v4.4.0+incompatible Latest Latest
Warning

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

Go to latest
Published: Aug 22, 2017 License: BSD-3-Clause Imports: 29 Imported by: 2

Documentation

Overview

Package suite allows to read tests and collections of tests (suites) from disk and execute them in a controlled way or run throughput load test from these test/suites.

Variable Handling

There are three "scopes" for variables when executing a suite read from disk (a RawSuite):

  • Global Scope: Variables set from "the outside", typically via the -D command line flag to cmd/ht. Executing a suite or tests does not modify variables in this scope. This global set of varibales is input to the Execute method of RawSuite.

  • Suite Scope: This is the context of the suite, it can be changed dynamically via variable extraction from executed tests. The initial suite scope is:

  • a copy of the Global Scope, with

  • added COUNTER and RANDOM variables and

  • merged defaults from the suite's Variables section. Variable expansion happens inside the values of the Variables section.

  • Test Scope: The tests scope is the context of a test, actually the set of variables used to make a ht.Test from a RawTest. It is constructed as follows:

  • To a copy of the actual suite scope,

  • the automatic variables COUNTER and RANDOM are added (with new values),

  • merged with the call Variables

  • merged with the tests Variables section. Variable expansion happens inside the values of the call and test Variables section.

Here "merging" means that the variables in the Variables section are added to the scope if not already present. I.e. the variables from outer scope dominate variables from inner scopes.

It might be helpful to think from the inside out: Given a test on disk

{
    Request: { URL: "{{A}}{{B}}{{C}}" }
    Variables: {
        A: "http://"
        B: "example.org"
        C: "/path"
        D: "?query"
    }
}

which provides defaults for A, B, C and D which are used unless set in some outer scope. If this test is used in a suite

{
    Main: [
        {File: "test.ht" }
        {File: "test.ht", Variables: { D: "?other" }
    ]
    Variables: {
        A: "https://"
    }
}

Then A will be "https" unless set in some outer scope and D will be the test-default "?query" once and "?other" in the second invocation. If such a suite is called with

A == "ftp://"
B == "localhost"
D == "?none"

in the global scope (e.g. by running cmd/ht with -D B=localhost) then these values will dominate any default from the various Variables sections.

Package suite allows to read tests and collections of tests (suites) from disk and execute them in a controlled way or run throughput load test from these test/suites.

Throughput Load-Testing

Load tests in the form of a throuhghput-test can be made through the Throughput function. This function will try to create a certain throughput load, i.e. a certain average number of request per second also known as query per second (QPS). The intervals between request follow an exponential distribution which mimics the load generated from real-world, uncorrelated users.

The requests are generated from different Scenarios which contribute a certain percentage of requests to the set of all requests. The scenarios are basically just suites of tests: One suite might simulate the bahaviour of a bot while an other scenario can simulate the behaviour of a "normal" user and a third scenario performs actions a user with special interests.

The Tests of each suite/scenario are executed, including the checks. Note that some checks can produce additional requests which are not accounted in the throughput rate of the load test (but do hit the target server). Checks can be turned off on per scenario basis.

If all tests (and thus request) in a suite/scenario have been executed, the suite is repeated. To reach the desired request througput rate each scenario is run in multiple parallel threads. New threads are started as needed. The number of threads may be limited on a per scenario basis.

Index

Constants

This section is empty.

Variables

View Source
var (
	SuiteTmpl      *template.Template
	ShortSuiteTmpl *template.Template
	HtmlSuiteTmpl  *htmltemplate.Template
)

Templates used to generate default and short text output and HTML page.

View Source
var (
	// ErrAbortExecution indicates that suite iteration should stop.
	ErrAbortExecution = errors.New("Abort Execution")
)
View Source
var IDSep = "\u2055" // "\u203b" "\u2237"

IDSep is the separator string used in constructing IDs for the individual test executed. The test Name are (miss)used to report details:

<ScenarioNo>/<ThreadNo>/<Repetition>/<TestNo> IDSep <ScenarioName> IDSep <TestName>

Functions

func DataPrint

func DataPrint(data []TestData, out io.Writer)

DataPrint prints data to out in human readable tabular format.

func DataToCSV

func DataToCSV(data []TestData, out io.Writer) error

DataToCSV prints data as a CSV table to out after sorting data by Started.

func ErrorList

func ErrorList(err error) htmltemplate.HTML

ErrorList renders err in HTML. It creates an <ul> if err is of type ht.ErrorList and has more than one entry.

func HTMLReport

func HTMLReport(dir string, s *Suite) error

HTMLReport generates a report of the outcome of s to directory dir.

func LoadMock

func LoadMock(filename string, fs FileSystem, variables map[string]string, auto bool) (*mock.Mock, error)

LoadMock loads filename from fs and replaces the variables. If auto is true the automatic variables COUNTER and RANDOM are set too.

func Throughput

func Throughput(scenarios []Scenario, opts ThroughputOptions, csvout io.Writer) ([]TestData, *Suite, error)

Throughput runs a throughput load test with request taken from the given scenarios. During the ramp the request rate is linearely increased until it reaches the desired rate of requests/second (QPS). This rate is kept for the rest of the loadtest i.e. for duration-ramp.

Setup and Teardown tests in the scenarios are executed once for each scenario before and after starting the loadtest. Note that loadtesting differs from suite execution here: Cookies set during Setup are _not_ propagated to the Main tests (but Setup and Teardown share a cookie jar).

The actual load is generated from the Main tests. A scenario can be executed multiple times in parallel threads. After full execution of all Main tests each thread starts over and re-executes the Main tests again by generating a new Suite.

Setup tests may populate the set of variables used for the Main (and of course Teardown) test. Additionally the two variables THREAD and REPETITION are set for each round of Main tests. The distinction between thread and repetition looks strange given that each repetition is isolated and independent from the thread it occors, but it is not: Repetitions are unbound and an endurance test running for 8 hours might repeate the Main test severla thousend times but maybe just using 15 threads which allows to prepare the system under test e.g. with 15 numbered user-accounts. It will aggregate all executed Test with a Status higher (bader) or equal to CollectFrom.

It returns a summary of all request, the aggregated tests and an error indicating if the throughput test reached the targeted rate and the desired distribution between scenarios.

Types

type ByStarted

type ByStarted []TestData

ByStarted sorts []TestData by the Started field.

func (ByStarted) Len

func (s ByStarted) Len() int

func (ByStarted) Less

func (s ByStarted) Less(i, j int) bool

func (ByStarted) Swap

func (s ByStarted) Swap(i, j int)

type Executor

type Executor func(test *ht.Test) error

A Executor is responsible for executing the given test during the Iterate'ion of a Suite. It should return nil if execution should continue and ErrAbortExecution to stop further iteration.

type File

type File struct {
	Data string
	Name string
}

File is a textual representation of a hjson data read from disk.

func LoadFile

func LoadFile(filename string) (*File, error)

LoadFile reads the given file and returns it as a File.

func (*File) Basename

func (f *File) Basename() string

Basename of f.

func (*File) Dirname

func (f *File) Dirname() string

Dirname of f.

type FileSystem

type FileSystem map[string]*File

FileSystem acts like an in-memory filesystem. A empty FileSystem accesses the real OS file system.

func NewFileSystem

func NewFileSystem(txt string) (FileSystem, error)

NewFileSystem parses txt which must be of the form

# <filename1>
<filecontent1>

# <filename2>
<filecontent2>

...

into a new FileSystem.

func (FileSystem) Load

func (fs FileSystem) Load(name string) (*File, error)

Load the file with the given name from fs.

type LoopIteration

type LoopIteration struct {
	Data      interface{}
	I         int // 0-based loop index
	N         int // 1-based loop index
	Odd, Even bool
}

LoopIteration helps ranging over Data in a template.

type Mixin

type Mixin struct {
	*File
}

Mixin of a test.

type RawElement

type RawElement struct {
	File      string
	Variables map[string]string
	Mocks     []string

	Test map[string]interface{}
}

RawElement represents one test in a RawSuite.

type RawLoadTest

type RawLoadTest struct {
	*File
	Name        string
	Description string
	Scenarios   []RawScenario
	Variables   map[string]string
}

RawLoadTest as read from disk.

func LoadRawLoadtest

func LoadRawLoadtest(filename string, fs FileSystem) (*RawLoadTest, error)

LoadRawLoadtest from the given filename.

func (*RawLoadTest) ToScenario

func (raw *RawLoadTest) ToScenario(globals map[string]string) []Scenario

ToScenario produces a list of scenarios from raw.

type RawMock

type RawMock struct {
	*File

	// Variables are the defaults of the variables.
	Variables map[string]string
}

RawMock is the raw form of a mock read in from disk.

func LoadRawMock

func LoadRawMock(filename string, fs FileSystem) (*RawMock, error)

LoadRawMock tries to reads filename from fs as a RawMock.

func (*RawMock) ToMock

func (rm *RawMock) ToMock(variables scope.Variables, auto bool) (*mock.Mock, error)

ToMock produces a real Mock from rm. The auto flag triggers generation of COUNTER and RANDOM variables.

type RawScenario

type RawScenario struct {
	Name       string            // Name of this Scenario
	File       string            // File is the RawSuite to use as scenario
	Percentage int               // Percentage this scenario contributes to the load test.
	MaxThreads int               // MaxThreads to use for this scenario. 0 means unlimited.
	Variables  map[string]string // Variables used.
	OmitChecks bool              // OmitChecks in the tests.
	// contains filtered or unexported fields
}

RawScenario represents a scenario in a load test.

type RawSuite

type RawSuite struct {
	*File
	Name, Description     string
	Setup, Main, Teardown []RawElement
	KeepCookies           bool
	OmitChecks            bool
	Variables             map[string]string
	Verbosity             int
	// contains filtered or unexported fields
}

RawSuite represents a suite as represented on disk as a HJSON file.

func LoadRawSuite

func LoadRawSuite(filename string, fs FileSystem) (*RawSuite, error)

LoadRawSuite with the given filename from fs.

func (*RawSuite) AddRawTests

func (rs *RawSuite) AddRawTests(ts ...*RawTest)

AddRawTests adds ts to the tests in rs.

func (*RawSuite) Execute

func (rs *RawSuite) Execute(global map[string]string, jar *cookiejar.Jar, logger *log.Logger) *Suite

Execute the raw suite rs and capture the outcome in a Suite.

Tests are executed linearely, first the Setup, then the Main and finally the Teardown test. Any Failure or error during setup will skip further setup and main test (but allteardown will be executed). The following table shows two runs with possible outcomes.

  Test Type     Run1     Run2
------------------------------------------
  Setup-1       Pass     Pass
  Setup-2       Fail     Pass
  Setup-3       Skip     Pass
  Main-1        Skip     Fail
  Main-2        Skip     Pass
  Teardown-1    Pass     Pass
  Teardown-2    Fail     Error
  Teardown-3    Pass     Pass

func (*RawSuite) RawTests

func (rs *RawSuite) RawTests() []*RawTest

RawTests return all tests in rs.

func (*RawSuite) Validate

func (rs *RawSuite) Validate(global map[string]string) error

Validate rs to make sure it can be decoded into welformed ht.Tests.

type RawTest

type RawTest struct {
	*File
	Mixins    []*Mixin          // Mixins of this test.
	Variables map[string]string // Variables are the defaults of the variables.
	// contains filtered or unexported fields
}

RawTest is a raw for of a test as read from disk with its mixins and its variables.

func LoadRawTest

func LoadRawTest(filename string, fs FileSystem) (*RawTest, error)

LoadRawTest reads filename and produces a new RawTest.

func (*RawTest) Disable

func (rt *RawTest) Disable()

Disable rt.

func (*RawTest) Enable

func (rt *RawTest) Enable()

Enable rt.

func (*RawTest) IsEnabled

func (rt *RawTest) IsEnabled() bool

IsEnabled reports if rt is enabled.

func (*RawTest) String

func (rt *RawTest) String() string

func (*RawTest) ToTest

func (rt *RawTest) ToTest(variables scope.Variables) (*ht.Test, error)

ToTest produces a ht.Test from a raw test rt.

type Scenario

type Scenario struct {
	// Name of this scenario. May differ from the suite.
	Name string

	// RawSuite is the suite to execute repeatedly to generate load.
	*RawSuite

	// Percentage of requests in the throughput test taken from this
	// scenario.
	Percentage int

	// MaxThreads limits the number of threads used to generate load from
	// this scenario. The value 0 indicates unlimited number of threads.
	MaxThreads int

	// OmitChecks turns off checks in this scenario, i.e. only the request
	// is made but no checks are performed on the response.
	OmitChecks bool
	// contains filtered or unexported fields
}

Scenario describes a single scenario to be run as part of a throughput test.

type StatusRing

type StatusRing struct {
	// contains filtered or unexported fields
}

A StatusRing is a ring buffer for ht.Status value counts.

func NewStatusRing

func NewStatusRing(n int, limit float64) *StatusRing

NewStatusRing returns a StatusRing of size n and the given error limit.

func (*StatusRing) Status

func (sr *StatusRing) Status(out []int) bool

Status fills out with current counts and returns whether the error limit is execed. After execution out[0] contains the number of ht.NotRun values stored, out[1] the number of ht.Skipped values and so on.

func (*StatusRing) Store

func (sr *StatusRing) Store(v ht.Status)

Store v in the ring buffer sr.

type Suite

type Suite struct {
	Name        string // Name of the Suite.
	Description string // Description of what's going on here.
	KeepCookies bool   // KeepCookies in a cookie jar common to all Tests.

	Status   ht.Status     // Status is the overall status of the whole suite.
	Error    error         // Error encountered during execution of the suite.
	Started  time.Time     // Start of the execution.
	Duration time.Duration // Duration of the execution.

	Tests []*ht.Test // The Tests to execute

	Variables      scope.Variables // The initial variable assignment
	FinalVariables scope.Variables // The final set of variables.
	Jar            *cookiejar.Jar  // The cookie jar used

	Verbosity int
	Log       interface {
		Printf(format string, a ...interface{})
	}
	// contains filtered or unexported fields
}

A Suite is a collection of Tests which can be executed sequentily with the result captured.

func NewFromRaw

func NewFromRaw(rs *RawSuite, global map[string]string, jar *cookiejar.Jar, logger *log.Logger) *Suite

NewFromRaw sets up a new Suite from rs, read to be Iterated.

func (*Suite) Iterate

func (suite *Suite) Iterate(executor Executor)

Iterate the suite through the given executor.

func (*Suite) JUnit4XML

func (s *Suite) JUnit4XML() (string, error)

JUnit4XML generates a JUnit 4 compatible XML result with each Test reported as an individual testcase. The following mapping is used from ht to JUnit

Suite  -->  testsuite
Test   -->  testcase
Check  -->  assertion (reported only as number)

NotRun checks are reported as Skipped and Bogus checks are counted as Errored tests.

Teardown tests are _not_ included in the JUnit XML output: A failed or errored teardwon test is not a suite failure/error but this cannot be modeled in the XML output (such test setup/teardwon code is not part of the testsuite in JUnit). Tools like Teamcity would report a broken build if any failed/errored teardown test was included in the JUnit XML output. We do not omit the Setup Tests as these are supposed to pass.

func (*Suite) PrintReport

func (s *Suite) PrintReport(w io.Writer) error

PrintReport outputs a textual report of s to w.

func (*Suite) PrintShortReport

func (s *Suite) PrintShortReport(w io.Writer) error

PrintShortReport outputs a short textual report of s to w.

func (*Suite) Stats

func (suite *Suite) Stats() (notRun int, skipped int, passed int, failed int, errored int, bogus int)

Stats counts the test results of s.

type TestData

type TestData struct {
	Started      time.Time
	Status       ht.Status
	ReqDuration  time.Duration
	TestDuration time.Duration
	ID           TestIdentifier
	Error        error
	Wait         time.Duration
	Overage      time.Duration
}

TestData collects main information about a single Test executed during a throughput test.

func ReadLiveCSV

func ReadLiveCSV(r io.Reader) ([]TestData, error)

ReadLiveCSV unmarshals the csv generated during a throughput test back to TestData.

type TestIdentifier

type TestIdentifier struct {
	Scenario     int
	Thread       int
	Repetition   int
	Test         int
	ScenarioName string
	TestName     string
}

TestIdentifier identifies a single Test, i.e. a single request in a throughput test. All number fields are 1-based.

func TestIDFromString

func TestIDFromString(s string) (TestIdentifier, error)

func (TestIdentifier) NumericalID

func (ti TestIdentifier) NumericalID() string

func (TestIdentifier) String

func (ti TestIdentifier) String() string

type ThroughputOptions

type ThroughputOptions struct {
	// Rate is the target rate of request per second (QPS).
	Rate float64

	// Duration of the whole throughput test.
	Duration time.Duration

	// Ramp duration during which the actual rate (QPS) is linearily
	// increased to the target rate.
	Ramp time.Duration

	// MaxErrorRate is the maximal tolerable error rate over the last
	// 50 request. If the error rate exceeds this limit the throughput
	// test finishes early. Values <= 0 disable aborting on errors.
	MaxErrorRate float64

	// CollectFrom limit collection of tests to those test with a
	// status equal or bader.
	CollectFrom ht.Status
}

ThroughputOptions collects options which control execution of a throughput load test.

Jump to

Keyboard shortcuts

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