From 16878695872a4b22c22e43dcf60310ddf356f358 Mon Sep 17 00:00:00 2001 From: alpharush <0xalpharush@protonmail.com> Date: Fri, 6 Sep 2024 12:58:23 -0500 Subject: [PATCH] apply mutations to corpus before sequence generation --- fuzzing/fuzzer.go | 20 ++- fuzzing/sequence_generator.go | 241 +++++++++++------------------ fuzzing/sequence_generator_test.go | 14 +- 3 files changed, 108 insertions(+), 167 deletions(-) diff --git a/fuzzing/fuzzer.go b/fuzzing/fuzzer.go index 0b9b1500..b7313bb1 100644 --- a/fuzzing/fuzzer.go +++ b/fuzzing/fuzzer.go @@ -569,17 +569,15 @@ func defaultCallSequenceGeneratorConfigFunc(fuzzer *Fuzzer, valueSet *valuegener // Create a sequence generator config which uses the created value generator. sequenceGenConfig := &CallSequenceGeneratorConfig{ - NewSequenceProbability: 0.3, - RandomUnmodifiedCorpusHeadWeight: 800, - RandomUnmodifiedCorpusTailWeight: 100, - RandomUnmodifiedSpliceAtRandomWeight: 200, - RandomUnmodifiedInterleaveAtRandomWeight: 100, - RandomMutatedCorpusHeadWeight: 80, - RandomMutatedCorpusTailWeight: 10, - RandomMutatedSpliceAtRandomWeight: 20, - RandomMutatedInterleaveAtRandomWeight: 10, - ValueGenerator: mutationalGenerator, - ValueMutator: mutationalGenerator, + NewSequenceProbability: 0.3, + Prepend: 800, + Append: 200, + Splice: 100, + Interleave: 100, + PrependAndMutate: 100, + AppendAndMutate: 100, + ValueGenerator: mutationalGenerator, + ValueMutator: mutationalGenerator, } return sequenceGenConfig, nil } diff --git a/fuzzing/sequence_generator.go b/fuzzing/sequence_generator.go index 9d8b2356..ae81d51d 100644 --- a/fuzzing/sequence_generator.go +++ b/fuzzing/sequence_generator.go @@ -31,10 +31,6 @@ type CallSequenceGenerator struct { // returned when calling PopSequenceElement. fetchIndex int - // prefetchModifyCallFunc describes the method to use to mutate the next indexed call sequence element, prior - // to its fetching by PopSequenceElement. - prefetchModifyCallFunc PrefetchModifyCallFunc - // mutationStrategyChooser is a weighted random selector of functions that prepare the CallSequenceGenerator with // a baseSequence derived from corpus entries. mutationStrategyChooser *randomutils.WeightedRandomChooser[CallSequenceGeneratorMutationStrategy] @@ -47,45 +43,45 @@ type CallSequenceGeneratorConfig struct { // sequence rather than mutating one from the corpus. NewSequenceProbability float32 - // RandomUnmodifiedCorpusHeadWeight defines the weight that the CallSequenceGenerator should use the call sequence + // Prepend defines the weight that the CallSequenceGenerator should use the call sequence // generation strategy of taking the head of a corpus sequence (without mutations) and append newly generated calls // to the end of it. - RandomUnmodifiedCorpusHeadWeight uint64 + Prepend uint64 - // RandomUnmodifiedCorpusTailWeight defines the weight that the CallSequenceGenerator should use the call sequence + // Append defines the weight that the CallSequenceGenerator should use the call sequence // generation strategy of taking the tail of a corpus sequence (without mutations) and prepend newly generated calls // to the start of it. - RandomUnmodifiedCorpusTailWeight uint64 + Append uint64 - // RandomUnmodifiedSpliceAtRandomWeight defines the weight that the CallSequenceGenerator should use the call sequence + // Splice defines the weight that the CallSequenceGenerator should use the call sequence // generation strategy of taking two corpus sequences (without mutations) and splicing them before joining them // together. - RandomUnmodifiedSpliceAtRandomWeight uint64 + Splice uint64 - // RandomUnmodifiedInterleaveAtRandomWeight defines the weight that the CallSequenceGenerator should use the call + // Interleave defines the weight that the CallSequenceGenerator should use the call // sequence generation strategy of taking two corpus sequences (without mutations) and interleaving a random // number of calls from each. - RandomUnmodifiedInterleaveAtRandomWeight uint64 + Interleave uint64 - // RandomMutatedCorpusHeadWeight defines the weight that the CallSequenceGenerator should use the call sequence + // PrependAndMutate defines the weight that the CallSequenceGenerator should use the call sequence // generation strategy of taking the head of a corpus sequence (with mutations) and append newly generated calls // to the end of it. - RandomMutatedCorpusHeadWeight uint64 + PrependAndMutate uint64 - // RandomMutatedCorpusTailWeight defines the weight that the CallSequenceGenerator should use the call sequence + // AppendAndMutate defines the weight that the CallSequenceGenerator should use the call sequence // generation strategy of taking the tao; of a corpus sequence (with mutations) and prepend newly generated calls // to the start of it. - RandomMutatedCorpusTailWeight uint64 + AppendAndMutate uint64 - // RandomMutatedSpliceAtRandomWeight defines the weight that the CallSequenceGenerator should use the call sequence + // SpliceAndMutate defines the weight that the CallSequenceGenerator should use the call sequence // generation strategy of taking two corpus sequences (with mutations) and splicing them before joining them // together. - RandomMutatedSpliceAtRandomWeight uint64 + SpliceAndMutate uint64 - // RandomMutatedInterleaveAtRandomWeight defines the weight that the CallSequenceGenerator should use the call + // InterleaveAndMutate defines the weight that the CallSequenceGenerator should use the call // sequence generation strategy of taking two corpus sequences (with mutations) and interleaving a random // number of calls from each. - RandomMutatedInterleaveAtRandomWeight uint64 + InterleaveAndMutate uint64 // ValueGenerator defines the value provider to use when generating new values for call sequences. This is used both // for ABI call data generation, and generation of additional values such as the "value" field of a @@ -101,18 +97,12 @@ type CallSequenceGeneratorConfig struct { // one occurs. type CallSequenceGeneratorFunc func(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error -// PrefetchModifyCallFunc defines a method used to modify a call sequence element before being fetched from this -// provider for use. -// Returns an error if one occurs. -type PrefetchModifyCallFunc func(sequenceGenerator *CallSequenceGenerator, element *calls.CallSequenceElement) error - // CallSequenceGeneratorMutationStrategy defines a structure for a mutation strategy used by a CallSequenceGenerator. type CallSequenceGeneratorMutationStrategy struct { // CallSequenceGeneratorFunc describes a method used to populate a provided call sequence. CallSequenceGeneratorFunc CallSequenceGeneratorFunc - - // PrefetchModifyCallFunc defines a method used to modify a call sequence element before being fetched. - PrefetchModifyCallFunc PrefetchModifyCallFunc + // Whether to mutate + Mutate bool } // NewCallSequenceGenerator creates a CallSequenceGenerator to generate call sequences for use in fuzzing campaigns. @@ -126,78 +116,41 @@ func NewCallSequenceGenerator(worker *FuzzerWorker, config *CallSequenceGenerato generator.mutationStrategyChooser.AddChoices( randomutils.NewWeightedRandomChoice( CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncCorpusHead, - PrefetchModifyCallFunc: nil, - }, - new(big.Int).SetUint64(config.RandomUnmodifiedCorpusHeadWeight), - ), - randomutils.NewWeightedRandomChoice( - CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncCorpusTail, - PrefetchModifyCallFunc: nil, - }, - new(big.Int).SetUint64(config.RandomUnmodifiedCorpusTailWeight), - ), - randomutils.NewWeightedRandomChoice( - CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncExpansion, - PrefetchModifyCallFunc: nil, - }, - new(big.Int).SetUint64(config.RandomUnmodifiedCorpusTailWeight), - ), - randomutils.NewWeightedRandomChoice( - CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncSpliceAtRandom, - PrefetchModifyCallFunc: nil, + CallSequenceGeneratorFunc: prependFromCorpus, }, - new(big.Int).SetUint64(config.RandomUnmodifiedSpliceAtRandomWeight), + new(big.Int).SetUint64(config.Prepend), ), randomutils.NewWeightedRandomChoice( CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncInterleaveAtRandom, - PrefetchModifyCallFunc: nil, + CallSequenceGeneratorFunc: appendFromCorpus, }, - new(big.Int).SetUint64(config.RandomUnmodifiedInterleaveAtRandomWeight), + new(big.Int).SetUint64(config.Append), ), randomutils.NewWeightedRandomChoice( CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncCorpusHead, - PrefetchModifyCallFunc: prefetchModifyCallFuncMutate, + CallSequenceGeneratorFunc: spliceCorpus, }, - new(big.Int).SetUint64(config.RandomMutatedCorpusHeadWeight), + new(big.Int).SetUint64(config.Splice), ), randomutils.NewWeightedRandomChoice( CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncCorpusTail, - PrefetchModifyCallFunc: prefetchModifyCallFuncMutate, + CallSequenceGeneratorFunc: interleaveCorpus, }, - new(big.Int).SetUint64(config.RandomMutatedCorpusTailWeight), + new(big.Int).SetUint64(config.Interleave), ), randomutils.NewWeightedRandomChoice( CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncSpliceAtRandom, - PrefetchModifyCallFunc: prefetchModifyCallFuncMutate, + CallSequenceGeneratorFunc: prependFromCorpus, + Mutate: true, }, - new(big.Int).SetUint64(config.RandomMutatedSpliceAtRandomWeight), + new(big.Int).SetUint64(config.PrependAndMutate), ), randomutils.NewWeightedRandomChoice( CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqGenFuncInterleaveAtRandom, - PrefetchModifyCallFunc: prefetchModifyCallFuncMutate, + CallSequenceGeneratorFunc: appendFromCorpus, + Mutate: true, }, - new(big.Int).SetUint64(config.RandomMutatedInterleaveAtRandomWeight), - ), - randomutils.NewWeightedRandomChoice(CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqSwapRandomElement, - PrefetchModifyCallFunc: nil, - }, - new(big.Int).SetUint64(config.RandomMutatedInterleaveAtRandomWeight), - ), - randomutils.NewWeightedRandomChoice(CallSequenceGeneratorMutationStrategy{ - CallSequenceGeneratorFunc: callSeqDeleteRandomElement, - PrefetchModifyCallFunc: nil, - }, - new(big.Int).SetUint64(config.RandomMutatedInterleaveAtRandomWeight), + new(big.Int).SetUint64(config.AppendAndMutate), ), ) @@ -212,7 +165,6 @@ func (g *CallSequenceGenerator) InitializeNextSequence() (bool, error) { // Reset the state of our generator. g.baseSequence = make(calls.CallSequence, g.worker.fuzzer.config.Fuzzing.CallSequenceLength) g.fetchIndex = 0 - g.prefetchModifyCallFunc = nil // Check if there are any previously un-executed corpus call sequences. If there are, the fuzzer should execute // those first. @@ -242,11 +194,21 @@ func (g *CallSequenceGenerator) InitializeNextSequence() (bool, error) { // If we have a corpus mutation method, call it to generate our base sequence, then set the pre-fetch modify // call function. if corpusMutationFunc != nil && corpusMutationFunc.CallSequenceGeneratorFunc != nil { - err = corpusMutationFunc.CallSequenceGeneratorFunc(g.worker.randomProvider, g.worker.fuzzer.corpus.RandomMutationTargetSequence, g.baseSequence) + fetchFromCorpusAndMutate := func() (calls.CallSequence, error) { + corpusSequence, err := g.worker.fuzzer.corpus.RandomMutationTargetSequence() + if err != nil { + return corpusSequence, fmt.Errorf("could not generate a corpus mutation derived call sequence due to an error obtaining a mutation target: %v", err) + } + if corpusMutationFunc.Mutate { + err = g.mutateCallSequence(corpusSequence) + } + return corpusSequence, err + } + + err = corpusMutationFunc.CallSequenceGeneratorFunc(g.worker.randomProvider, fetchFromCorpusAndMutate, g.baseSequence) if err != nil { return true, fmt.Errorf("could not generate a corpus mutation derived call sequence due to an error executing a mutation method: %v", err) } - g.prefetchModifyCallFunc = corpusMutationFunc.PrefetchModifyCallFunc } } return true, nil @@ -270,16 +232,6 @@ func (g *CallSequenceGenerator) PopSequenceElement() (*calls.CallSequenceElement if err != nil { return nil, err } - } else { - // We have an element, if our generator set a post-call modify for this function, execute it now to modify - // our call prior to return. This allows mutations to be applied on a per-call time frame, rather than - // per-sequence, making use of the most recent runtime data. - if g.prefetchModifyCallFunc != nil { - err = g.prefetchModifyCallFunc(g, element) - if err != nil { - return nil, err - } - } } // Update the element with the current nonce for the associated chain. @@ -369,10 +321,29 @@ func (g *CallSequenceGenerator) generateNewElement() (*calls.CallSequenceElement return calls.NewCallSequenceElement(selectedMethod.Contract, msg, blockNumberDelay, blockTimestampDelay), nil } -// prefetchModifyCallFuncMutate is a PrefetchModifyCallFunc, called by a CallSequenceGenerator to apply mutations -// to a call sequence element, prior to it being fetched. +// mutateCallSequenceElement applies a mutation to a call sequence element, prior to it being fetched from the corpus. // Returns an error if one occurs. -func prefetchModifyCallFuncMutate(sequenceGenerator *CallSequenceGenerator, element *calls.CallSequenceElement) error { +func (g *CallSequenceGenerator) mutateCallSequence(sequence calls.CallSequence) error { + // Equal probability of mutating a random element, expanding a random element, swapping a random element, or deleting a random element. + if g.worker.randomProvider.Intn(4) == 0 { + return g.mutateRandomElement(sequence) + } + if g.worker.randomProvider.Intn(4) == 1 { + return expandRandomElement(g.worker.randomProvider, sequence) + } + if g.worker.randomProvider.Intn(4) == 2 { + return swapRandomElement(g.worker.randomProvider, sequence) + } + if g.worker.randomProvider.Intn(4) == 3 { + return deleteRandomElement(g.worker.randomProvider, sequence) + } + + return nil +} + +func (g *CallSequenceGenerator) mutateRandomElement(sequence calls.CallSequence) error { + element := sequence[g.worker.randomProvider.Intn(len(sequence))] + // If this element has no ABI value based call data, exit early. if element.Call == nil || element.Call.DataAbiValues == nil { return nil @@ -381,7 +352,7 @@ func prefetchModifyCallFuncMutate(sequenceGenerator *CallSequenceGenerator, elem // Loop for each input value and mutate it abiValuesMsgData := element.Call.DataAbiValues for i := 0; i < len(abiValuesMsgData.InputValues); i++ { - mutatedInput, err := valuegeneration.MutateAbiValue(sequenceGenerator.config.ValueGenerator, sequenceGenerator.config.ValueMutator, &abiValuesMsgData.Method.Inputs[i].Type, abiValuesMsgData.InputValues[i]) + mutatedInput, err := valuegeneration.MutateAbiValue(g.config.ValueGenerator, g.config.ValueMutator, &abiValuesMsgData.Method.Inputs[i].Type, abiValuesMsgData.InputValues[i]) if err != nil { return fmt.Errorf("error when mutating call sequence input argument: %v", err) } @@ -393,7 +364,16 @@ func prefetchModifyCallFuncMutate(sequenceGenerator *CallSequenceGenerator, elem return nil } -func callSeqSwapRandomElement(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { +func expandRandomElement(provider *rand.Rand, sequence calls.CallSequence) error { + // Expand the sequence + expandedSequence := expandRandList(provider, sequence) + + copy(sequence, expandedSequence) + + return nil +} + +func swapRandomElement(provider *rand.Rand, sequence calls.CallSequence) error { // Swap the element swappedSequence := swapRandList(provider, sequence) @@ -402,7 +382,7 @@ func callSeqSwapRandomElement(provider *rand.Rand, sequenceGenerator func() (cal return nil } -func callSeqDeleteRandomElement(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { +func deleteRandomElement(provider *rand.Rand, sequence calls.CallSequence) error { // Delete the element deletedSequence := deleteRandList(provider, sequence) @@ -411,63 +391,46 @@ func callSeqDeleteRandomElement(provider *rand.Rand, sequenceGenerator func() (c return nil } -// callSeqGenFuncCorpusHead is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a sequence +// prependFromCorpus is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a sequence // whose head is based off of an existing corpus call sequence. // Returns an error if one occurs. -func callSeqGenFuncCorpusHead(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { +func prependFromCorpus(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { // Obtain a call sequence from the corpus corpusSequence, err := sequenceGenerator() if err != nil { return fmt.Errorf("could not obtain corpus call sequence for head mutation: %v", err) } - // Prepend the new calls to the end of the corpus sequence - i := provider.Intn(len(corpusSequence)) + 1 - spliced := append(corpusSequence[:i], sequence...) - - copy(sequence, spliced) + // Determine the length of the slice to be copied in the head. + maxLength := utils.Min(len(sequence), len(corpusSequence)) + copy(sequence, corpusSequence[:maxLength]) return nil } -// callSeqGenFuncCorpusTail is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a sequence +// appendFromCorpus is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a sequence // whose tail is based off of an existing corpus call sequence. // Returns an error if one occurs. -func callSeqGenFuncCorpusTail(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { +func appendFromCorpus(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { // Obtain a call sequence from the corpus corpusSequence, err := sequenceGenerator() if err != nil { return fmt.Errorf("could not obtain corpus call sequence for tail mutation: %v", err) } + // Determine a random position to slice the call sequence. maxLength := utils.Min(len(sequence), len(corpusSequence)) - i := provider.Intn(maxLength) + 1 - fmt.Println("i: ", i) - // Append the new calls to the end of the corpus sequence - spliced := append(sequence[i:], corpusSequence...) - - copy(sequence, spliced) - - return nil -} - -// callSeqGenFuncExpansion is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a -// sequence which is expanded up to 30 times by replicating an existing call sequence element at a random position. -func callSeqGenFuncExpansion(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { - - // Expand the sequence - expandedSequence := expandRandList(provider, sequence) - - copy(sequence, expandedSequence) + targetLength := provider.Intn(maxLength) + 1 + copy(sequence[len(sequence)-targetLength:], corpusSequence[len(corpusSequence)-targetLength:]) return nil } -// callSeqGenFuncSpliceAtRandom is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a +// spliceCorpus is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a // sequence which is based off of two corpus call sequence entries, from which a random length head and tail are // respectively sliced and joined together. // Returns an error if one occurs. -func callSeqGenFuncSpliceAtRandom(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { +func spliceCorpus(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { // Obtain two corpus call sequence entries headSequence, err := sequenceGenerator() if err != nil { @@ -486,11 +449,11 @@ func callSeqGenFuncSpliceAtRandom(provider *rand.Rand, sequenceGenerator func() return nil } -// callSeqGenFuncInterleaveAtRandom is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a +// interleaveCorpus is a CallSequenceGeneratorFunc which prepares a CallSequenceGenerator to generate a // sequence which is based off of two corpus call sequence entries, from which a random number of transactions are // taken and interleaved (each element of one sequence will be followed by an element of the other). // Returns an error if one occurs. -func callSeqGenFuncInterleaveAtRandom(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { +func interleaveCorpus(provider *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error { // Obtain two corpus call sequence entries firstSequence, err := sequenceGenerator() if err != nil { @@ -533,7 +496,7 @@ func expandRandList[T any](provider *rand.Rand, xs []*T) []*T { return xs } k := provider.Intn(l) - t := provider.Intn(min(32, l)) + 1 + t := provider.Intn(utils.Min(32, l)) + 1 return expandAt(xs, k, t) } @@ -564,7 +527,7 @@ func swapRandList[T any](provider *rand.Rand, xs []*T) []*T { return xs } i, j := rand.Intn(len(xs)), rand.Intn(len(xs)) - return swapAt(xs, min(i, j), max(i, j)) + return swapAt(xs, utils.Min(i, j), utils.Max(i, j)) } // spliceAtRandom splices two lists at random positions. @@ -589,19 +552,3 @@ func interleaveLL[T any](a, b []*T) []*T { } return append([]*T{a[0], b[0]}, interleaveLL(a[1:], b[1:])...) } - -// min returns the smaller of two integers. -func min(a, b int) int { - if a < b { - return a - } - return b -} - -// max returns the larger of two integers. -func max(a, b int) int { - if a > b { - return a - } - return b -} diff --git a/fuzzing/sequence_generator_test.go b/fuzzing/sequence_generator_test.go index 2466a941..0fe2ad0d 100644 --- a/fuzzing/sequence_generator_test.go +++ b/fuzzing/sequence_generator_test.go @@ -52,13 +52,10 @@ func getMockCallSequenceElementCall(data int) *calls.CallMessage { func TestSplice(t *testing.T) { strategies := map[string]func(rand *rand.Rand, sequenceGenerator func() (calls.CallSequence, error), sequence calls.CallSequence) error{ - // "interleave": callSeqGenFuncInterleaveAtRandom, - // "splice": callSeqGenFuncSpliceAtRandom, - // "expansion": callSeqGenFuncExpansion, - "prepend": callSeqGenFuncCorpusHead, - "apppend": callSeqGenFuncCorpusTail, - // "delete": callSeqDeleteRandomElement, - // "swap": callSeqSwapRandomElement, + "interleave": interleaveCorpus, + "splice": spliceCorpus, + "prepend": prependFromCorpus, + "append": appendFromCorpus, } // Seed the PRNG to make the randomness deterministic @@ -66,10 +63,9 @@ func TestSplice(t *testing.T) { // expectedSize := 8 // Adjust based on the expected interleave size in your scenario // Prepare a source sequence (with the expected size) - for name, strategyFn := range strategies { // Call the function under test - sourceSequence := []calls.CallSequence{getMockCallSequence(1, 1), getMockCallSequence(1, 5)} + sourceSequence := []calls.CallSequence{getMockCallSequence(4, 1), getMockCallSequence(4, 5)} i := 0 sequence := getMockCallSequence(4, 7) sequence = append(sequence, getMockCallSequence(4, 9)...)