datamock
is a tool for initializing your API services with dummy (or real) data. It is useful for generating data for rapid prototyping, testing, load testing, ... of your API.
datamock
is implemented as a .NET Core Global Tool, so its installation is quite simple:
dotnet tool install -g Kros.DummyData.Initializer
To update it to the latest version, if it was installed previously, use:
dotnet tool update -g Kros.DummyData.Initializer
Run posting mock data to your API.
datamock run -s D:/dummyData/source
--source, -s: Directory where the initialization data is located.
--verbose, -v: Verbose.
Run generating preview data.
datamock preview -s D:/dummyData -d D:/dummyData/output
--source, -s: Directory where the initialization data is located.
--dest, -d: Directory where the generated data will be saved.
--verbose, -v: Verbose.
--compress, -c: Compress final JSON
written to file.
The purpose of this tool is to generate test data based on templates and send this data to your API. The tool is general, it does everything based on data in a specific directory structure.
The --source
parameter defines the path to the directory where the data that will be sent to your API is located. The settings.json
file must be in this directory. Which contains settings for the whole process.
{
"BaseUrl": "http://localhost:3000",
"Variables": {
"variable1": "value1",
"variable2": "value2"
},
"DefaultHeaders": {
"x-header-1": "header1"
},
"Proxy": {
"Address": "http://192.155.1.1:1111",
"BypassProxyOnLocal": true
},
"Retrying": [
1,
3,
5
],
"RequestTimeOut": 180,
"AuthOptions": {
"AuthServer": "https://identityserver.com",
"ClientId": "yourClient",
"User":{
"Name": "userName",
"Password": "userPassword"
}
},
"MaxConcurrencyCount": 8
}
BaseUrl
: Base URL to your API. (It can be overwritten at the level of each request)
Variables
: Variables (key, value) to be used when processing templates. More below. (optional)
DefaultHeaders
: Default headers to be added to each request.
Proxy
: Proxy settings for outgoing requests. (optional)
Retrying
: The sleep durations to wait for on each retry. (optional)
RequestTimeOut
: Request timeout in seconds. (optional)
AuthOptions
: Settings for the authorization handler. Currently only IdentityServer is supported. (optional)
MaxConcurrencyCount
: Maximum of requests executed in parallel. (optional, Default is 1
- no concurrency.)
The definition of requests and data that will be sent to your API is in separate directories. Directories are processed in alphabetical order, so I recommend using a numeric prefix to define the order.
.
├─ 🗄️ settings.json
├─ 📂001-catalog
│ ├─ 🗄️ request.json
│ └─ 🗄️ catalogs.json
├─ 📂002-users
│ ├─ 🗄️ request.json
│ └─ 🗄️ users.json
├─ 📂003-orders-user1
│ ├─ 🗄️ request.json
│ └─ 🗄️ orders.json
└─ 📂004-orders-user2
├─ 🗄️ request.json
└─ 🗄️ orders.json
Each directory must contain a request.json
file that contains the request definition and at least one JSON
data file that contains the data (or data template).
{
"BaseUrl": "http://localhost:5000",
"Name": "catalog",
"Description": "Catalog",
"Path": "/api/catalog/",
"QueryParams": {},
"Headers": {
"x-header-1": "header1"
},
"Variables": {
"variable1": "value1"
},
"ContinueOnError": true,
"User":{
"Name": "userName",
"Password": "userPassword"
},
"ExtractResponseProperty": "id"
}
BaseUrl
: Base URL to your API. (If not specified, the value from settings.json
is used)
Name
: Name of request. (Will be used when defining the dependency, see below)
Description
: Is used in logging. (optional, if not set, then Name
is used)
Path
: API path.
QueryParams
: Query parameters (key, value). (optional)
Headers
: Headers to be added to the request. If the same header is in DefaultHeaders, it is overwritten. (optional)
Variables
: Variables (key, value) to be used when processing templates. More below. (optional)
ContinueOnError
: Continue with http request error? (optional, default is false
)
User
: User credentials. (optional, if not specified, they will be used from settings.json
)
ExtractResponseProperty
: Specifies which property to extract from the response and use as an output parameter. This can then be used in another request. More in the section on addictions. (optional)
The JSON
file contains a list of []
data that will be sent sequentially to your API using the POST
method Content-Type:application/json
. The data can be real or it can be defined as a template. datamock
uses a great scriban project to preprocess data. This means that you can take advantage of all the language constructs and builtins functions that scriban offers. You can also use functions defined directly in this tool and variables that you have defined in the Variables
section. You will use these variables as follows {{variables.variable1}}
Example:
[
{{ for i in 1..1000 }}
{
"requestId": "{{i}}",
"code": "10-00-{{i}}",
"name": "{{lorem_ipsum 30}}",
"description": "{{lorem_ipsum 60}}",
"price": {{random_int 10 300}},
"type": "{{variables.variable1}}"
}
{{
if i < 1000
","
end
}}
{{ end }}
]
datamock
uses scriban to generate data based on this template.
The requestId
property is used to define the dependency between requests. More see the section on dependencies.
If you want to initialize a complex system, you probably need to resolve the dependencies between individual requests. For example, you create users then you need the id
s of the created users to properly initialize their orders in the next requests.
In the request definition it is possible to define the ExtractResponseProperty
property, which describes the response property to be extract from the response and added to the outputs
variables, which you can use in other requests. The outputs
variables will be available under the key, which is compose as follows requestName_requestId
. You can access it as follows {{outputs.requestName_requestId}}
. In our example, {{outputs.users_1}}
003-orders-user1/request.json
{
"Name": "orders_user1",
"Path": "/api/orders/{{outputs.users_1}}/"
}
All from scriban. And a few directly from this tool
Random int
value. Params: min
max
random_int 1 55
Random double
value. Params: min
max
random_double 1 55
Random person first name.
random_first_name
Random person last name.
random_last_name
Random person email.
random_email
Random person name. Params: nameSeparator
.
random_person_name '-'
Random person.
person = random_person
person.first_name
person.last_name
person.mail
Random lorem ipsum text. Params: maxLength
.
lorem_ipsum 30
Get value from dictionary by key.
Params:
dictionary
: dictionary<string,string>
from which I want the value.key
: the key by which I want the value.
value = get_by_key outputs "key"
Returns a new string that right-aligns the characters in this instance by padding them on the left with a specified Unicode character, for a specified total length.
Params:
totalWidth
: the number of characters in the resulting string, equal to the number of original characters plus any additional padding characters.paddingChar
: A Unicode padding character. (opional, default value is0
)
value | pad_left 2 '0'
For preview
command is allowed function named repating. Sometimes you need to use a same request definition multiple times, for example for multiple users, tenants, ... In this case, you can create a repeat.json
file in the directory where the request definition is located. This file contains a list of iterations defined by name, and it is possible to add custom variables to each iteration to be used in further processing.
[
{
"Name": "company1",
"Variables":{
"userName": "[email protected]",
"userPassword": "password"
}
},
{
"Name": "company2",
"Variables":{
"userName": "[email protected]",
"userPassword": "password"
}
},
{
"Name": "company3",
"Variables":{
"userName": "[email protected]",
"userPassword": "password"
}
}
]
This file is also processed using
scriban
, so it is possible to use its features.
A separate subdirectory is created for each iteration according to the name of the iteration. The definition of the request and its data will be modified in the given subdirectory.
Variables from repeat definition are add to
output
variables by keyname_variableKey
. E.g.:{{outputs.company1_userName}}
.
If you need to reference to the name of the current iteration, it is available in the variable {{variable.index}}
. This is needed, for example, in the requestId
definition.
{
"requestId": "{{variables.index}}_{{i}}"
},
In other requests, where we use repeats
, and in each iteration we want to refer outputs from different iteration of other requests, we must do it indirectly through variables. In the definition of iteration we define the required variable, for example companyId
.
{
"Name": "company1-user1",
"Variables":{
"companyId": "{{ outputs.companies_company1_1 }}",
"userName": "[email protected]",
"userPassword": "password"
}
},
Next, we can use this variable in the request definition.
{
"Name": "invoices",
"Path": "/companies/{{variables.companyId}}/invoices",
"User":{
"Name": "{{variables.userName}}",
"Password": "{{variables.userPassword}}"
}
}