mockingbird - a service for emulating REST services and queue-interface services
Leviysoft mockingbird is an independently maintained fork of Tinkoff/mockingbird and is not related to Tinkoff in any kind.
mockingbird supports the following scenarios:
- Execution of a specific case with a specific set of events and HTTP/GRPC responses
- Constant emulation of a happy-path to ensure autonomy of the stage environment(s)
Types of configurations:
- countdown - standalone configurations for testing a specific scenario. They have the highest priority when resolving ambiguities. Each mock is triggered n times (the number is set during creation). Automatically deleted at midnight.
- ephemeral - configurations that are automatically deleted at midnight. If a method/message is called/arrives simultaneously, for which both countdown and ephemeral mocks are suitable - countdown will be triggered.
- persistent - configuration intended for continuous operation. Has the lowest priority
To organize mocks in the UI and minimize the number of conflict situations, so-called services are implemented in mockingbird. Each mock (both HTTP and scenario) always belongs to one of the services. Services are created in advance and stored in the database. A service has a suffix (which also serves as the unique service id) and a human-readable name.
To achieve flexibility while maintaining the relative simplicity of configurations, a JSON templating feature is implemented in the service. To start, here's a simple example:
Template:
{
"description": "${description}",
"topic": "${extras.topic}",
"comment": "${extras.comments.[0].text}",
"meta": {
"field1": "${extras.fields.[0]}"
}
}
Values for substitution:
{
"description": "Some description",
"extras": {
"fields": ["f1", "f2"],
"topic": "Main topic",
"comments": [
{"text": "First nah!"}, {"text": "Okay"}
]
}
}
Result:
{
"description": "Some description",
"topic": "Main topic",
"comment": "First nah!",
"meta": {
"field1": "f1"
}
}
Currently, the following syntax is supported:
${a.[0].b}
- value substitution (JSON)${/a/b/c}
- value substitution (XPath)
WARNING! DO NOT USE NAMESPACES IN XPATH EXPRESSIONS
Template:
<root>
<tag1>${/r/t1}</tag1>
<tag2 a2="${/r/t2/@a2}">${/r/t2}</tag2>
</root>
Values for substitution:
<r>
<t1>test</t1>
<t2 a2="attr2">42</t2>
</r>
Result:
<root>
<tag1>test</tag1>
<tag2 a2="attr2">42</tag2>
</root>
To support complex scenarios, the service supports saving arbitrary states. A state is a document with an arbitrary schema, technically a state is a document in MongoDB. Writing new states can occur:
- when writing to state (the persist section) with an empty (or missing) predicate (the state section)
State is cumulatively appended. Overwriting fields is allowed.
Fields used for searching (used in predicates) must start with "_".
a sparse index will be automatically created for such fields
Prefixes:
seed
- values from the seed block (randomized at the start of the application)state
- the current statereq
- the request body (modes json, jlens, xpath)message
- the message body (in scenarios)query
- query parameters (in stubs)pathParts
- values extracted from the URL (in stubs) seeData Extraction from URL
extracted
- extracted valuesheaders
- HTTP headers
{
"a": "Just a string", //The field "a" is assigned a constant (can be any JSON value)
"b": "${req.fieldB}", //The field "b" is assigned the value from the fieldB of the request
"c": "${state.c}", //The field "c" is assigned the value from the "c" field of the current state
"d": "${req.fieldA}: ${state.a}" //The field d will contain a string consisting of req.fieldA and state.a
}
Predicates for state search are listed in the state
block. An empty object ({}
) in the state field is not allowed.
For state search, request data (without prefix), query parameters (prefix __query
), values extracted from the URL (prefix __segments
), and HTTP headers (prefix __headers
) can be used
Example:
{
"_a": "${fieldB}", //field from the request body
"_b": "${__query.arg1}", //query parameter
"_c": "${__segments.id}", //URL segment, see `Data Extraction from URL`
"_d": "${__headers.Accept}" //HTTP header
}
Sometimes there is a need to generate a random value and save and/or return it as a result of the mock's operation. To support such scenarios, a seed field is provided, allowing to set variables that will be generated at the mock's initialization. This avoids the need to recreate mocks with hardcoded ids
JavaScript evaluation is supported in seeds. The following functions are defined for backwards compatibility with "pseudofunctions":
%{randomString(n)}
- substitution of a random string of length n%{randomString("ABCDEF1234567890", m, n)}
- substitution of a random string consisting ofABCDEF1234567890
characters in the range [m, n)%{randomNumericString(n)}
- substitution of a random string consisting only of digits, of length n%{randomInt(n)}
- substitution of a random Int in the range [0, n)%{randomInt(m,n)}
- substitution of a random Int in the range [m, n)%{randomLong(n)}
- substitution of a random Long in the range [0, n)%{randomLong(m,n)}
- substitution of a random Long in the range [m, n)%{UUID}
- substitution of a random UUID%{now(yyyy-MM-dd'T'HH:mm:ss)}
- the current time in the specified format%{today(yyyy-MM-dd)}
- the current date in the specified format
Complex formatted strings can be defined: %{randomInt(10)}: %{randomLong(10)} | %{randomString(12)}
, all pseudo-functions from the list above are supported
Found stubs - candidates remaining after validation of URL, headers, and request body Found scenarios - candidates remaining after validation of the message body
Found Stubs (Scenarios) | State Required | States Found | Result |
---|---|---|---|
№1 | No | - | №1 is triggered |
№1 | Yes | 0 | Error |
№1 | Yes | 1 | №1 is triggered |
№1 №2 |
No No |
- | Error |
№1 №2 |
No Yes |
- 0 |
№1 is triggered |
№1 №2 |
No Yes |
- 1 |
№2 is triggered |
№1 №2 |
No Yes |
- 2 (and more) |
Error |
№1 №2 |
Yes Yes |
0 0 |
Error |
№1 №2 |
Yes Yes |
0 1 |
№2 is triggered |
№1 №2 |
Yes Yes |
0 2 (and more) |
Error |
№1 №2 |
Yes Yes |
1 1 (and more) |
Error |
№1 №2 №3 |
Yes Yes Yes |
0 1 0 |
№2 is triggered |
№1 №2 №3 |
Yes Yes Yes |
0 1 1 |
Error |
№1 №2 №3 |
Yes Yes Yes |
0 2 0 |
Error |
Workflow:
- Search for a mock by URL/HTTP-verb/headers
- Body validation
- Search for state by predicate
- Substitution of values in the response template
- State modification
- Sending the response
HTTP headers are validated for exact match values, extra headers are not considered an error
Request body validation in HTTP stubs can work in the following modes:
- no_body - the request must be without a body
- any_body - the request body must be non-empty, while it is not parsed or checked
- raw - the request body is not parsed and is checked for full correspondence with the content of request.body
- json - the request body must be a valid JSON and is checked for correspondence with the content of request.body
- xml - the request body must be a valid XML and is checked for correspondence with the content of request.body
- jlens - the request body must be a valid JSON and is validated according to conditions described in request.body
- xpath - the request body must be a valid XML and is validated according to conditions described in request.body
- web_form - the request body must be in x-www-form-urlencoded format and is validated according to conditions described in request.body
- multipart - the request body must be in multipart/form-data format. Validation rules for parts are configured individually (see the section below)
ATTENTION! multipart requests must be made to a separate method - /api/mockingbird/execmp
For responses, the following modes are supported:
- raw
- json
- xml
- binary
- proxy
- json-proxy
- xml-proxy
- no_body
The no_body
mode in the response is needed if the stub returns a 204 or 304 code. These codes are distinguished from others by the fact that they cannot have any body in the response, this behavior is described in RFC 7231 and RFC 7232. The no_body
mode can also be used with other HTTP codes, but it is mandatory for the specified ones.
Request and response modes are completely independent of each other (you can configure a response in XML to a JSON request if desired, except for json-proxy and xml-proxy modes).
In the delay field, you can pass a correct FiniteDuration no longer than 30 seconds
Sometimes, a URL contains an identifier not as a parameter but as a direct part of the path. In such cases, it becomes impossible to describe a persistent stub due to the inability to have a full path match. This is where the pathPattern
field comes in handy, into which a regex can be passed, and the path will be checked for a match against this regex. It should be noted that although the matching is done in MongoDB in an efficient manner, this feature should not be abused, and the pathPattern
should not be used if matching by full equality is possible.
Example:
{
"name": "Sample stub",
"scope": "persistent",
"pathPattern": "/pattern/(?<id>\d+)",
"method": "GET",
"request": {
"headers": {},
"mode": "no_body",
"body": {}
},
"response": {
"code": 200,
"mode": "json",
"headers": {"Content-Type": "application/json"},
"body": {"id": "${pathParts.id}"}
}
}
Anything that needs to be extracted from the path should be done with a named group, and there can be as many groups as needed. Later on, these can be referred to through pathParts.<group_name>
.
In some cases, it's necessary to insert into the response data that cannot be extracted by simple means. For these purposes, extractors have been added.
Extracts values from JSON located within CDATA.
Configuration:
{
"type": "jcdata",
"prefix": "/root/inner/tag", // Path to the tag with CDATA
"path": "path.to" // Path to the desired value
}
Sometimes you have to deal with requests in which XML is nested inside CDATA. In such cases, you can inline the CDATA content using the inlineCData
parameter (supported in xpath
and xml
).
{
"name": "Sample stub",
"method": "POST",
"path": "/pos-loans/api/cl/get_partner_lead_info",
"state": {
// Predicates
},
"request": {
"headers": {"Content-Type": "application/json"},
"mode": "json",
"body": {
"trace_id": "42",
"account_number": "228"
}
},
"persist": {
// State modifications
},
"response": {
"code": 200,
"mode": "json",
"body": {
"code": 0,
"credit_amount": 802400,
"credit_term": 120,
"interest_rate": 13.9,
"partnum": "CL3.15"
},
"headers": {"Content-Type": "application/json"},
"delay": "1 second"
}
}
{
"name": "Sample stub",
"method": "POST",
"path": "/pos-loans/api/evil/soap/service",
"state": {
// Predicates
},
"request": {
"headers": {"Content-Type": "application/xml"},
"mode": "raw",
"body": "<xml><request type=\"rqt\"></request></xml>"
},
"persist": {
// State modifications
},
"response": {
"code": 200,
"mode": "raw",
"body": "<xml><response type=\"rqt\"></response></xml>",
"headers": {"Content-Type": "application/xml"},
"delay": "1 second"
}
}
{
"name": "Sample stub",
"method": "POST",
"path": "/pos-loans/api/cl/get_partner_lead_info",
"state": {
// Predicates
},
"request": {
"headers": {"Content-Type": "application/json"},
"mode": "jlens",
"body": {
"meta.id": {"==": 42}
}
},
"persist": {
// State modifications
},
"response": {
"code": 200,
"mode": "json",
"body": {
"code": 0,
"credit_amount": 802400,
"credit_term": 120,
"interest_rate": 13.9,
"partnum": "CL3.15"
},
"headers": {"Content-Type": "application/json"},
"delay": "1 second"
}
}
WARNING! DO NOT USE NAMESPACES IN XPATH EXPRESSIONS
{
"name": "Sample stub",
"method": "POST",
"path": "/pos-loans/api/cl/get_partner_lead_info",
"state": {
// Predicates
},
"request": {
"headers": {"Content-Type": "application/xml"},
"mode": "xpath",
"body": {
"/payload/response/id": {"==": 42}
},
"extractors": {"name": {...}, ...} //optional
},
"persist": {
// State modifications
},
"response": {
"code": 200,
"mode": "raw",
"body": "<xml><response type=\"rst\"></response></xml>",
"headers": {"Content-Type": "application/xml"},
"delay": "1 second"
}
}
WARNING! multipart requests must be performed on a separate method - /api/mockingbird/execmp
Part validation modes:
any
- value is not validatedraw
- exact matchjson
- exact match, value parsed as Jsonxml
- exact match, value parsed as XMLurlencoded
- similar toweb_form
mode for validating the entire bodyjlens
- Json condition checkxpath
- XML condition check
{
"name": "Sample stub",
"method": "POST",
"path": "/test/multipart",
"state": {
// Predicates
},
"request": {
"headers": {},
"mode": "multipart",
"body": {
"part1": {
"mode": "json", //validation mode
"headers": {}, //part headers
"value": {} //value specification for the validator
},
"part2": {
...
}
},
"bypassUnknownParts": true //flag allowing to ignore all parts not present in the validator's specification
//by default, the flag is enabled, can be passed only to disable (false)
},
"persist": {
// State modifications
},
"response": {
"code": 200,
"mode": "json",
"body": {
"code": 0,
"credit_amount": 802400,
"credit_term": 120,
"interest_rate": 13.9,
"partnum": "CL3.15"
},
"headers": {"Content-Type": "application/json"},
"delay": "1 second"
}
}
{
"name": "Simple proxy",
"method": "POST",
"path": "/pos-loans/api/cl/get_partner_lead_info",
"state": {
// Predicates
},
"request": {
// Request specification
},
"response": {
"mode": "proxy",
"uri": "http://some.host/api/cl/get_partner_lead_info"
}
}
{
"name": "Simple proxy",
"method": "POST",
"path": "/pos-loans/api/cl/get_partner_lead_info",
"state": {
// Predicates
},
"request": {
// Request specification, mode json or jlens
},
"response": {
"mode": "json-proxy",
"uri": "http://some.host/api/cl/get_partner_lead_info",
"patch": {
"field.innerField": "${req.someRequestField}"
}
}
}
{
"name": "Simple proxy",
"method": "POST",
"path": "/pos-loans/api/cl/get_partner_lead_info",
"state": {
// Predicates
},
"request": {
// Request specification, mode xml or xpath
},
"response": {
"mode": "xml-proxy",
"uri": "http://some.host/api/cl/get_partner_lead_info",
"patch": {
"/env/someTag": "${/some/requestTag}"
}
}
}
In jlens and xpath modes, the following is supported:
{
"a": {"==": "some value"}, //exact match
"b": {"!=": "some value"}, //not equal
"c": {">": 42} | {">=": 42} | {"<": 42} | {"<=": 42}, //comparisons, for numbers only, can be combined
"d": {"~=": "\\d+"}, //regexp match
"e": {"size": 10}, //length, for arrays and strings
"f": {"exists": true} //existence check
}
Keys in such objects are either a path in json ("a.b.[0].c") or xpath ("/a/b/c").
Note: Currently, comparison functions may not work correctly with xpath pointing to XML attributes.
The problem can be bypassed by checking for existence/non-existence:
/tag/otherTag/[@attr='2']": {"exists": true}
In jlens mode, the following operations are additionally supported:
{
"g": {"[_]": ["1", 2, true]}, //the field must contain one of the listed values
"h": {"![_]": ["1", 2, true]}, //the field must NOT contain any of the listed values
"i": {"&[_]": ["1", 2, true]} //the field must be an array containing all listed values (order does not matter)
}
In xpath mode, the following operations are additionally supported:
"/some/tag": {"cdata": {"==": "test"}}, //validation for exact match of CDATA, argument must be a STRING
"/some/tag": {"cdata": {"~=": "\d+"}}, //CDATA regex validation, argument must be a STRING
"/some/tag": {"jcdata": {"a": {"==": 42}}}, //validating CDATA content as JSON, all available predicates are supported
"/other/tag": {"xcdata": {"/b": {"==": 42}}} //validating CDATA content as XML, all available predicates are supported
In web_form mode, ONLY the following operations are supported:
==
, !=
, ~=
, size
, [_]
, ![_]
, &[_]
How it works under the hood: When creating a mock, the proto files nested in the request are parsed and transformed into a json representation of the protobuf schema. The database stores the json representation, not the original proto file. The first triggering of the mock may take a little longer than subsequent ones because a protobuf message decoder is generated from the json representation on the first trigger. After decoding, the data is transformed into json, which is checked by json predicates specified in the requestPredicates field. If the conditions are met, then the json from response.data (in fill mode) is serialized into protobuf and returned as a response.
Workflow:
- Search for mocks by method name
- Body validation
- Search for state by predicate
- Substituting values in the response template
- State modification
- Response delivery
Workflow:
- Search for the mock by source.
- Search for state by predicate.
- Validate incoming message.
- Substitute values into the response template.
- Modify state.
- Send response.
- Execute callbacks (see the "callbacks configuration" section).
Supported modes for input:
- raw
- json
- xml
- jlens
- xpath
Supported modes for output:
- raw
- json
- xml
{
"name": "Spring has come",
"service": "test",
"source": "rmq_example_autobroker_decision", //source from the config
"input": {
"mode": .. //as for HTTP stubs
"payload": .. //as body for HTTP stubs
},
"state": {
// Predicates
},
"persist": { //Optional
// State modifications
},
"destination": "rmq_example_q1", // destination from the config, optional
"output": { //Optional
"mode": "raw",
"payload": "..",
"delay": "1 second"
},
"callback": { .. }
}
To mimic the behavior of the real world, sometimes it is necessary to call an HTTP service (for example, to fetch GBO when a message arrives) or to send additional messages to queues. For this purpose, callbacks can be used. The result of the service call can be parsed and saved in the state if necessary. Callbacks use the state of the caller.
Supported modes for request:
- no_body
- raw
- json
- xml
Supported modes for response:
- json
- xml
Please note! The initial state is passed along the entire chain of callbacks, and it is not modified by the persist block (!!!)
{
"type": "http",
"request": {
"url": "http://some.host/api/v2/peka",
"method": "POST",
"headers": {"Content-Type": "application/json"},
"mode": "json",
"body": {
"trace_id": "42",
"account_number": "228"
}
},
"responseMode": "json" | "xml", //Mandatory only if the persist block is present
"persist": { //Optional
// State modifications
},
"delay": "1 second", //Delay BEFORE executing the callback, optional
"callback": { .. } //Optional
}
Supported modes for output:
- raw
- json
- xml
{
"type": "message",
"destination": "rmq_example_q1", // destination from the config
"output": {
"mode": "raw",
"payload": ".."
},
"callback": { .. } //Optional
}