Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

New config framework for the different schedulers #913

Merged
merged 40 commits into from
Mar 13, 2019
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
029db7c
Add the config framework for the different schedulers
na-- Jan 29, 2019
15a4128
Fix minor issues linting and test issues
na-- Jan 30, 2019
4a1c941
Break apart the scheduler config type initializations
na-- Jan 31, 2019
b50ff18
Clean up some commented-out code
na-- Jan 31, 2019
4821c50
Update and test the new scheduler configurations
na-- Feb 25, 2019
99b0994
Merge branch 'master' into scheduler-config-wip
na-- Feb 25, 2019
ad3f6f4
Fix linter issues
na-- Feb 25, 2019
62c910c
Remove the debug output
na-- Feb 26, 2019
8113438
Clean up and add more tests
na-- Feb 26, 2019
ce1c967
Restore the Split() method
na-- Feb 26, 2019
2451ead
Add copyright notices
na-- Feb 28, 2019
507c5c5
Improve the execution config generation from shortcuts
na-- Mar 1, 2019
54eb412
Warn if the user specifies only the new execution option
na-- Mar 1, 2019
9cab261
Improve the scheduler config tests
na-- Mar 1, 2019
977b10e
Refactor some CLI configs and add some basic configuration tests
na-- Mar 1, 2019
d1943ec
Improve the readability of the config test cases
na-- Mar 1, 2019
effbf30
Work arround some strage appveyor environment variable issues
na-- Mar 5, 2019
2120a12
Fix minor issues with the structure of the new config tests
na-- Mar 5, 2019
2c1adf3
Move TestConfigConsolidation() and its helpers to a separate file
na-- Mar 5, 2019
62ff4aa
Fix a typo in a comment
na-- Mar 5, 2019
704dcc3
Fix the duplicate config file path CLI flag
na-- Mar 6, 2019
a7d0b6f
Preserve the trailing spaces in the k6 ASCII banner
na-- Mar 6, 2019
b86d5cd
Automatically create the config file's parent folder
na-- Mar 6, 2019
c8c902a
Add a missing copyright notice
na-- Mar 6, 2019
13e506d
Fix the newline before the banner
na-- Mar 7, 2019
5d8b354
Move the root CLI persistent flags to their own flagset
na-- Mar 7, 2019
588ef66
Fix an env.var/CLI flag conflict and issues with CLI usage messages
na-- Mar 7, 2019
e76972a
Improve the CLI flags test framework in the config consolidation test
na-- Mar 7, 2019
295abd2
Add support for testing the JSON config in the consolidation as well
na-- Mar 7, 2019
9eb4379
Improve the comments on the funcitons that deal with the file config
na-- Mar 7, 2019
2de0923
Add tests and fix a minor bug
na-- Mar 8, 2019
bf84bc0
Merge pull request #935 from loadimpact/config-painful-testing
na-- Mar 8, 2019
93e4eae
Merge branch 'master' into scheduler-config-wip
na-- Mar 8, 2019
227f22c
Merge branch 'master' into scheduler-config-wip
na-- Mar 11, 2019
079282a
Fix or suppress linter errors
na-- Mar 11, 2019
238e224
Use a custom error type for execution conflict errors in the config
na-- Mar 11, 2019
6d19e61
Improve config consolidation, default values and tests
na-- Mar 12, 2019
9404af6
Extend config validation to the cloud and archive subcommands
na-- Mar 12, 2019
2498121
Silence a linter warning... for now!
na-- Mar 12, 2019
d88be34
Override the execution setting when execution shortcuts are used
na-- Mar 13, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 0 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -197,7 +197,6 @@ Configuration mechanisms do have an order of precedence. As presented, options a
As shown above, there are several ways to configure the number of simultaneous virtual users k6 will launch. There are also different ways to specify how long those virtual users will be running. For simple tests you can:
- Set the test duration by the `--duration`/`-d` CLI flag (or the `K6_DURATION` environment variable and the `duration` script/JSON option). For ease of use, `duration` is specified with human readable values like `1h30m10s` - `k6 run --duration 30s script.js`, `k6 cloud -d 15m10s script.js`, `export K6_DURATION=1h`, etc. If set to `0`, k6 wouldn't stop executing the script unless the user manually stops it.
- Set the total number of script iterations with the `--iterations`/`-i` CLI flag (or the `K6_ITERATIONS` environment variable and the `iterations` script/JSON option). k6 will stop executing the script whenever the **total** number of iterations (i.e. the number of iterations across all VUs) reaches the specified number. So if you have `k6 run --iterations 10 --vus 10 script.js`, then each VU would make only a single iteration.
- Set both the test duration and the total number of script iterations. In that case, k6 would stop the script execution whenever either one of the above conditions is reached first.

For more complex cases, you can specify execution stages. They are a combination of `duration,target-VUs` pairs. These pairs instruct k6 to linearly ramp up, ramp down, or stay at the number of VUs specified for the period specified. Execution stages can be set via the `stages` script/JSON option as an array of `{ duration: ..., target: ... }` pairs, or with the `--stage`/`-s` CLI flags and the `K6_STAGE` environment variable via the `duration:target,duration:target...` syntax.

Expand Down
71 changes: 70 additions & 1 deletion cmd/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -27,12 +27,15 @@ import (

"github.com/kelseyhightower/envconfig"
"github.com/loadimpact/k6/lib"
"github.com/loadimpact/k6/lib/scheduler"
"github.com/loadimpact/k6/stats/cloud"
"github.com/loadimpact/k6/stats/datadog"
"github.com/loadimpact/k6/stats/influxdb"
"github.com/loadimpact/k6/stats/kafka"
"github.com/loadimpact/k6/stats/statsd/common"
"github.com/pkg/errors"
"github.com/shibukawa/configdir"
log "github.com/sirupsen/logrus"
"github.com/spf13/afero"
"github.com/spf13/pflag"
null "gopkg.in/guregu/null.v3"
Expand Down Expand Up @@ -177,6 +180,71 @@ func readEnvConfig() (conf Config, err error) {
return conf, nil
}

// This checks for conflicting options and turns any shortcut options (i.e. duration, iterations,
// stages) into the proper scheduler configuration
func buildExecutionConfig(conf Config) (Config, error) {
result := conf
if conf.Duration.Valid {
na-- marked this conversation as resolved.
Show resolved Hide resolved
if conf.Iterations.Valid {
//TODO: make this an error in the next version
log.Warnf("Specifying both duration and iterations is deprecated and won't be supported in the future k6 versions")
}

if conf.Stages != nil {
//TODO: make this an error in the next version
log.Warnf("Specifying both duration and stages is deprecated and won't be supported in the future k6 versions")
}

if conf.Execution != nil {
return result, errors.New("specifying both duration and execution is not supported")
na-- marked this conversation as resolved.
Show resolved Hide resolved
}

ds := scheduler.NewConstantLoopingVUsConfig(lib.DefaultSchedulerName)
ds.VUs = conf.VUs
ds.Duration = conf.Duration
result.Execution = scheduler.ConfigMap{lib.DefaultSchedulerName: ds}
} else if conf.Stages != nil {
if conf.Iterations.Valid {
//TODO: make this an error in the next version
log.Warnf("Specifying both iterations and stages is deprecated and won't be supported in the future k6 versions")
}

if conf.Execution != nil {
return conf, errors.New("specifying both stages and execution is not supported")
}

ds := scheduler.NewVariableLoopingVUsConfig(lib.DefaultSchedulerName)
ds.StartVUs = conf.VUs
for _, s := range conf.Stages {
if s.Duration.Valid {
ds.Stages = append(ds.Stages, scheduler.Stage{Duration: s.Duration, Target: s.Target})
}
}
result.Execution = scheduler.ConfigMap{lib.DefaultSchedulerName: ds}
} else if conf.Iterations.Valid || conf.Execution == nil {
// Either shared iterations were explicitly specified via the shortcut option, or no execution
// parameters were specified in any way, which will run the default 1 iteration in 1 VU
if conf.Iterations.Valid && conf.Execution != nil {
return conf, errors.New("specifying both iterations and execution is not supported")
}

ds := scheduler.NewSharedIterationsConfig(lib.DefaultSchedulerName)
ds.VUs = conf.VUs
if conf.Iterations.Valid { // TODO: fix where the default iterations value is set... sigh...
ds.Iterations = conf.Iterations
}
na-- marked this conversation as resolved.
Show resolved Hide resolved

result.Execution = scheduler.ConfigMap{lib.DefaultSchedulerName: ds}
}

//TODO: validate the config; questions:
// - separately validate the duration, iterations and stages for better error messages?
// - or reuse the execution validation somehow, at the end? or something mixed?
// - here or in getConsolidatedConfig() or somewhere else?
mstoykov marked this conversation as resolved.
Show resolved Hide resolved

return result, nil
}

// Assemble the final consolidated configuration from all of the different sources:
// - start with the CLI-provided options to get shadowed (non-Valid) defaults in there
// - add the global file config options
Expand All @@ -185,6 +253,7 @@ func readEnvConfig() (conf Config, err error) {
// - merge the user-supplied CLI flags back in on top, to give them the greatest priority
// - set some defaults if they weren't previously specified
// TODO: add better validation, more explicit default values and improve consistency between formats
// TODO: accumulate all errors and differentiate between the layers?
func getConsolidatedConfig(fs afero.Fs, cliConf Config, runner lib.Runner) (conf Config, err error) {
cliConf.Collectors.InfluxDB = influxdb.NewConfig().Apply(cliConf.Collectors.InfluxDB)
cliConf.Collectors.Cloud = cloud.NewConfig().Apply(cliConf.Collectors.Cloud)
Expand All @@ -205,5 +274,5 @@ func getConsolidatedConfig(fs afero.Fs, cliConf Config, runner lib.Runner) (conf
}
conf = conf.Apply(envConf).Apply(cliConf)

return conf, nil
return buildExecutionConfig(conf)
}
5 changes: 5 additions & 0 deletions cmd/config_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -134,3 +134,8 @@ func TestConfigApply(t *testing.T) {
assert.Equal(t, []string{"influxdb", "json"}, conf.Out)
})
}

func TestBuildExecutionConfig(t *testing.T) {
//TODO: test the current config building and constructing of the execution plan, and the emitted warnings
//TODO: test the future full overwriting of the duration/iterations/stages/execution options
}
13 changes: 13 additions & 0 deletions cmd/run.go
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,7 @@ const (
teardownTimeoutErrorCode = 101
genericTimeoutErrorCode = 102
genericEngineErrorCode = 103
//invalidOptionsErrorCode = 104
)

var (
Expand Down Expand Up @@ -165,6 +166,18 @@ a commandline interface for interacting with it.`,
conf.Duration = types.NullDuration{}
}

//TODO: move a bunch of the logic above to a config "constructor" and to the Validate() method
if errList := conf.Validate(); len(errList) != 0 {

errMsg := []string{"There were problems with the specified script configuration:"}
for _, err := range errList {
errMsg = append(errMsg, fmt.Sprintf("\t- %s", err.Error()))
}
//TODO: re-enable exiting with validation errors
log.Warn(errors.New(strings.Join(errMsg, "\n")))
//return ExitCode{errors.New(strings.Join(errMsg, "\n")), invalidOptionsErrorCode}
}

// If summary trend stats are defined, update the UI to reflect them
if len(conf.SummaryTrendStats) > 0 {
ui.UpdateTrendColumns(conf.SummaryTrendStats)
Expand Down
2 changes: 1 addition & 1 deletion converter/har/converter.go
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,7 @@ func Convert(h HAR, options lib.Options, minSleep, maxSleep uint, enableChecks b
}

fprint(w, "\nexport let options = {\n")
options.ForEachValid("json", func(key string, val interface{}) {
options.ForEachSpecified("json", func(key string, val interface{}) {
if valJSON, err := json.MarshalIndent(val, " ", " "); err != nil {
convertErr = err
} else {
Expand Down
2 changes: 1 addition & 1 deletion js/bundle.go
Original file line number Diff line number Diff line change
Expand Up @@ -224,7 +224,7 @@ func (b *Bundle) Instantiate() (bi *BundleInstance, instErr error) {
} else {
jsOptionsObj = jsOptions.ToObject(rt)
}
b.Options.ForEachValid("json", func(key string, val interface{}) {
b.Options.ForEachSpecified("json", func(key string, val interface{}) {
if err := jsOptionsObj.Set(key, val); err != nil {
instErr = err
}
Expand Down
47 changes: 44 additions & 3 deletions lib/options.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,12 +29,18 @@ import (
"reflect"
"strings"

"github.com/loadimpact/k6/lib/scheduler"
"github.com/loadimpact/k6/lib/types"
"github.com/loadimpact/k6/stats"
"github.com/pkg/errors"
"gopkg.in/guregu/null.v3"
)

// DefaultSchedulerName is used as the default key/ID of the scheduler config entries
// that were created due to the use of the shortcut execution control options (i.e. duration+vus,
// iterations+vus, or stages)
const DefaultSchedulerName = "default"

// DefaultSystemTagList includes all of the system tags emitted with metrics by default.
// Other tags that are not enabled by default include: iter, vu, ocsp_status, ip
var DefaultSystemTagList = []string{
Expand Down Expand Up @@ -204,12 +210,16 @@ type Options struct {

// Initial values for VUs, max VUs, duration cap, iteration cap, and stages.
// See the Runner or Executor interfaces for more information.
VUs null.Int `json:"vus" envconfig:"vus"`
VUs null.Int `json:"vus" envconfig:"vus"`

//TODO: deprecate this? or reuse it in the manual control "scheduler"?
VUsMax null.Int `json:"vusMax" envconfig:"vus_max"`
Duration types.NullDuration `json:"duration" envconfig:"duration"`
Iterations null.Int `json:"iterations" envconfig:"iterations"`
Stages []Stage `json:"stages" envconfig:"stages"`

Execution scheduler.ConfigMap `json:"execution,omitempty" envconfig:"execution"`

// Timeouts for the setup() and teardown() functions
SetupTimeout types.NullDuration `json:"setupTimeout" envconfig:"setup_timeout"`
TeardownTimeout types.NullDuration `json:"teardownTimeout" envconfig:"teardown_timeout"`
Expand Down Expand Up @@ -309,6 +319,23 @@ func (o Options) Apply(opts Options) Options {
if opts.VUsMax.Valid {
o.VUsMax = opts.VUsMax
}

// Specifying duration, iterations, stages, or execution in a "higher" config tier
// will overwrite all of the the previous execution settings (if any) from any
// "lower" config tiers
// Still, if more than one of those options is simultaneously specified in the same
// config tier, they will be preserved, so the validation after we've consolidated
// all of the options can return an error.
//TODO: uncomment this after we start using the new schedulers
/*
if opts.Duration.Valid || opts.Iterations.Valid || opts.Stages != nil || o.Execution != nil {
o.Duration = types.NewNullDuration(0, false)
o.Iterations = null.NewInt(0, false)
o.Stages = nil
o.Execution = nil
}
*/

if opts.Duration.Valid {
o.Duration = opts.Duration
}
Expand All @@ -323,6 +350,13 @@ func (o Options) Apply(opts Options) Options {
}
}
}
// o.Execution can also be populated by the duration/iterations/stages config shortcuts, but
// that happens after the configuration from the different sources is consolidated. It can't
// happen here, because something like `K6_ITERATIONS=10 k6 run --vus 5 script.js` wont't
// work correctly at this level.
if opts.Execution != nil {
o.Execution = opts.Execution
}
if opts.SetupTimeout.Valid {
o.SetupTimeout = opts.SetupTimeout
}
Expand Down Expand Up @@ -411,10 +445,17 @@ func (o Options) Apply(opts Options) Options {
return o
}

// ForEachValid enumerates all struct fields and calls the supplied function with each
// Validate checks if all of the specified options make sense
func (o Options) Validate() []error {
//TODO: validate all of the other options... that we should have already been validating...
//TODO: maybe integrate an external validation lib: https://github.com/avelino/awesome-go#validation
return o.Execution.Validate()
}

// ForEachSpecified enumerates all struct fields and calls the supplied function with each
// element that is valid. It panics for any unfamiliar or unexpected fields, so make sure
// new fields in Options are accounted for.
func (o Options) ForEachValid(structTag string, callback func(key string, value interface{})) {
func (o Options) ForEachSpecified(structTag string, callback func(key string, value interface{})) {
structType := reflect.TypeOf(o)
structVal := reflect.ValueOf(o)
for i := 0; i < structType.NumField(); i++ {
Expand Down
12 changes: 12 additions & 0 deletions lib/options_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ import (
"time"

"github.com/kelseyhightower/envconfig"
"github.com/loadimpact/k6/lib/scheduler"
"github.com/loadimpact/k6/lib/types"
"github.com/loadimpact/k6/stats"
"github.com/stretchr/testify/assert"
Expand Down Expand Up @@ -87,6 +88,17 @@ func TestOptions(t *testing.T) {
assert.Equal(t, oneStage, opts.Apply(Options{Stages: oneStage}).Stages)
assert.Equal(t, oneStage, Options{}.Apply(opts).Apply(Options{Stages: oneStage}).Apply(Options{Stages: oneStage}).Stages)
})
t.Run("Execution", func(t *testing.T) {
sched := scheduler.NewConstantLoopingVUsConfig("test")
sched.VUs = null.IntFrom(123)
sched.Duration = types.NullDurationFrom(3 * time.Minute)
opts := Options{}.Apply(Options{Execution: scheduler.ConfigMap{"test": sched}})
cs, ok := opts.Execution["test"].(scheduler.ConstantLoopingVUsConfig)
assert.True(t, ok)
assert.Equal(t, int64(123), cs.VUs.Int64)
assert.Equal(t, "3m0s", cs.Duration.String())
})
//TODO: test that any execution option overwrites any other lower-level options
t.Run("RPS", func(t *testing.T) {
opts := Options{}.Apply(Options{RPS: null.IntFrom(12345)})
assert.True(t, opts.RPS.Valid)
Expand Down
83 changes: 83 additions & 0 deletions lib/scheduler/base_config.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
package scheduler

import (
"fmt"
"time"

"github.com/loadimpact/k6/lib/types"
null "gopkg.in/guregu/null.v3"
)

const minPercentage = 0.01

// The maximum time k6 will wait after an iteration is supposed to be done
const maxIterationTimeout = 300 * time.Second

// BaseConfig contains the common config fields for all schedulers
type BaseConfig struct {
Name string `json:"-"` // set via the JS object key
Type string `json:"type"`
StartTime types.NullDuration `json:"startTime"`
Interruptible null.Bool `json:"interruptible"`
IterationTimeout types.NullDuration `json:"iterationTimeout"`
Env map[string]string `json:"env"`
Exec null.String `json:"exec"` // function name, externally validated
Percentage float64 `json:"-"` // 100, unless Split() was called

//TODO: future extensions like tags, distribution, others?
}

// NewBaseConfig returns a default base config with the default values
func NewBaseConfig(name, configType string, interruptible bool) BaseConfig {
return BaseConfig{
Name: name,
Type: configType,
Interruptible: null.NewBool(interruptible, false),
IterationTimeout: types.NewNullDuration(30*time.Second, false),
Percentage: 100,
}
}

// Validate checks some basic things like present name, type, and a positive start time
func (bc BaseConfig) Validate() (errors []error) {
// Some just-in-case checks, since those things are likely checked in other places or
// even assigned by us:
if bc.Name == "" {
errors = append(errors, fmt.Errorf("scheduler name shouldn't be empty"))
}
if bc.Type == "" {
errors = append(errors, fmt.Errorf("missing or empty type field"))
}
if bc.Percentage < minPercentage || bc.Percentage > 100 {
errors = append(errors, fmt.Errorf(
"percentage should be between %f and 100, but is %f", minPercentage, bc.Percentage,
))
}
if bc.Exec.Valid && bc.Exec.String == "" {
errors = append(errors, fmt.Errorf("exec value cannot be empty"))
}
// The actually reasonable checks:
if bc.StartTime.Valid && bc.StartTime.Duration < 0 {
errors = append(errors, fmt.Errorf("scheduler start time should be positive"))
}
iterTimeout := time.Duration(bc.IterationTimeout.Duration)
if iterTimeout < 0 || iterTimeout > maxIterationTimeout {
errors = append(errors, fmt.Errorf(
"the iteration timeout should be between 0 and %s, but is %s", maxIterationTimeout, iterTimeout,
))
}
return errors
}

// GetBaseConfig just returns itself
func (bc BaseConfig) GetBaseConfig() BaseConfig {
return bc
}

// CopyWithPercentage is a helper function that just sets the percentage to
// the specified amount.
func (bc BaseConfig) CopyWithPercentage(percentage float64) *BaseConfig {
c := bc
c.Percentage = percentage
return &c
}
Loading