Skip to main content

Subscriptions

Subscriptions are used in Red Kite to start jobs based on either a cron expression (cron subscriptions) or a finding (event subscription). They are used to expand Red Kite's automation workflow.

A subscription can belong to a project, in which case, they will only take effect on the mentionned project. If a project is not specified for a subscription, it will take effect on all the projects.

Some subscriptions come built-in Red Kite. These subscriptions are marked as such, but they can still be modified by the users. Built-in subscriptions can even be deleted, but be sure to know what you are doing.

A susbcription, cron or event, will not trigger on a blocked resource.

A subscription is written in yaml format in the front-end. The project for which to apply the subscription can also be chosen using the dropdown menu.

A subscription belongning to a project will be deleted automatically if the project is deleted.

Cron Subscriptions

Cron subscriptions are started based on a cron expression. They are the most simple subscriptions and only require the information necessary to start a job.

Cron subscriptions can be reliably triggered as often as every twenty (20) seconds.

Even though cron subscriptions do not require a project to be set, they require at least one project to exist at the moment it is triggered to properly start at least one job.

Cron Subscription Syntax

A cron subscription contains the following main elements :

ElementDescriptionRequired
nameThe name of the subscription, for future referenceYes
cronExpressionThe cron expression that specifies when to start the jobYes
inputA value representing all the ressources of a typeNo
jobThe job to launch when the cron expression is triggeredYes
cooldownThe minimum amount of time, in seconds, before a subscription can be retriggered for the same project (default=0)No
batchAn object allowing the batching of data to start the job with multiple resourcesNo

N.B. Additonnal details on these elements are given in the following sections

name

  • Can be any string value.
  • Used to distinguish the subscription from others.
  • Example
name: my-first-cron-subscription

cronExpression

  • Must be a valid cron expression.
  • Specifies the schedule for triggering the job.
  • Example (Runs every day at midnight):
cronExpression: "0 0 * * *"

input

  • Optional element defining the resource type to process.
  • Supported values:
    • ALL_DOMAINS
    • ALL_HOSTS
    • ALL_IP_RANGES
    • ALL_TCP_PORTS
    • ALL_WEBSITES
  • Example
input: ALL_WEBSITES

job

  • The job element specifies the task to execute and its parameters.

    Sub-elementDescriptionRequired
    nameThe name of the job (must match a predefined type).Yes
    parametersA list of parameters required by the job.No
  • Example

job:
name: resourceCheck
parameters:
- name: domain
value: example.com

cooldown

  • Specifies the minimum time in seconds before the job can be triggered again for the same project.
  • Default value: 0 (no cooldown).
  • Example (1-hour cooldown):
cooldown: 3600

batch

  • The batch element enables grouping of resources for job execution.

    Sub-elementDescriptionRequired
    enabledBoolean value to enable or disable batching.Yes
    sizeMaximum size of input arrays in each batch. Optional, defaults to no limit.No
  • Example (Batch enabled with a size of 50):

batch:
enabled: true
size: 50

Cron Subscription Simple Example

Here is one of the shortest subscription possible. It triggers every day at noon to launch a DomainNameResolvingJob, using example.com as the domainName parameter value.

Therefore, it will resolve the domain name to an IP address by launching a DomainNameResolvingJob. It will emit a HostnameIpFinding if it properly resolves the domain name.

It will start again the next day at noon.

name: my cron subscription
cronExpression: 0 0 12 * * ?
job:
name: DomainNameResolvingJob
parameters:
- name: domainName
value: example.com

Cron Subscription Cooldown

The cooldown value ensures that a job is only started as often as the cooldown allows it. For instance, a job started at 10:00:00AM for a project, with a cron subscription cooldown period of 3600 seconds (1 hour), could not be started by the same subscription for the same project before 11:00:00AM, even if the cron expression triggers more often.

Another example would be the following subscription. Its cron expression triggers every hour, but its cooldown period is of 86400 seconds, or 24 hours. Therefore, for a project, the DomainNameResolvingJob would only be started once a day. However, it will check every hour if it can start because of the cron expression.

name: my cron subscription
cronExpression: "0 */1 * * *"
input: ALL_DOMAINS
cooldown: 86400
job:
name: DomainNameResolvingJob
parameters:
- name: domainName
value: example.com

Input variable

The input variable specifies an input source. There are multiple input sources described in the following table. The input variable is optionnal. The variables can be injected in the job parameters as well as in the conditions.

If a project is specified for the subscription, only the ressources of the targeted project will be used.

Input sourceVariablesBatching variables
ALL_DOMAINS${domainName}${domainBatch}
ALL_HOSTS${ip}${ipBatch}
ALL_TCP_PORTS${ip}, ${port}, ${protocol}${ipBatch}, ${portBatch}, ${protocolBatch}
ALL_IP_RANGES${ip}, ${mask}${ipBatch}, ${maskBatch}
ALL_WEBSITES${domainName}, ${ip}, ${port}, ${protocol}, ${ssl}, ${path}${domainBatch}, ${ipBatch}, ${portBatch}, ${protocolBatch}, ${sslBatch}, ${pathBatch}

When you specify the ALL_DOMAINS input, you have access to the ${domainName} injectable variable. Red Kite will apply the subscription for all the domains, and the domain's value will be injected where specified.

name: Refreshing all domain names
input: ALL_DOMAINS
cronExpression: "0 0 * * *"
job:
name: DomainNameResolvingJob
parameters:
- name: domainName
value: ${domainName}

When you specify the ALL_HOSTS input, you have access to the ${ip} injectable variable. Red Kite will apply the subscription for all the hosts, and the hosts's ip value will be injected where specified.

name: Rescanning all hosts
input: ALL_HOSTS
cronExpression: "0 0 * * *"
job:
name: TcpPortScanningJob
parameters:
- name: targetIp
value: ${ip}
- name: threads
value: 1000
- name: socketTimeoutSeconds
value: 0.7
- name: portMin
value: 1
- name: portMax
value: 65535
- name: ports
value: []

When you specify the ALL_TCP_PORTS input, you have access to the ${ip}, ${port} and ${protocol} injectable variables. Red Kite will apply the subscription for all the ports, and the ip, port and protocol values can be injected where specified. Protocol is either tcp or udp.

Only TCP ports are sent with this input, so the protocol variable will always equal 'tcp'

name: All tcp ports with condition
input: ALL_TCP_PORTS
cronExpression: "0 0 * * *"
job:
name: HttpServerCheckJob
parameters:
- name: targetIp
value: ${ip}
- name: ports
value: ["${port}"]
conditions:
- lhs: "${protocol}"
operator: "equals"
rhs: "tcp"

When you specify the ALL_IP_RANGES input, you have access to the ${ip} and ${mask} injectable variables. Red Kite will apply the subscription for all the projects' ip ranges.

name: Scanning IP ranges
cronExpression: 0 0 */7 * *
input: ALL_IP_RANGES
job:
name: TcpIpRangeScanningJob
parameters:
- name: targetIp
value: ${ip}
- name: targetMask
value: ${mask}
- name: rate
value: 100000
- name: portMin
value: 1
- name: portMax
value: 1000
- name: ports
value: []

When you specify the ALL_WEBSITES input, you have access to the ${domainName}, ${ip}, ${port}, ${path} and ${ssl} injectable variables. Red Kite will apply the subscription for all the websites, and the websites's different values will be injected where specified.

name: Recrawling websites
cronExpression: 0 0 */7 * *
input: ALL_WEBSITES
job:
name: WebsiteCrawlingJob
parameters:
- name: domainName
value: ${domainName}
- name: targetIp
value: ${ip}
- name: port
value: ${port}
- name: path
value: ${path}
- name: ssl
value: ${ssl}
- name: maxDepth
value: 3
- name: crawlDurationSeconds
value: 1800
- name: fetcherConcurrency
value: 10
- name: inputParallelism
value: 10
- name: extraOptions
value: -jc -kf all -duc -j -or -ob -silent
Batching job inputs

The cron subscription batching syntax can be used to launch a job with mulitple values as input. Instead of giving a standard string as input like the standard way of making a cron subscription, the batching syntax would give you an array of strings up to the specified size.

For instance, the following subscription will start a DomainNameResolvingJob by giving an array of up to 100 domain names in the domainNames parameter using the domainBatch input variable.

All the possible input variables are listed in the table here.

name: Refreshing all domain names
input: ALL_DOMAINS
batch:
enabled: true
size: 100
cronExpression: "0 0 */7 * *"
job:
name: DomainNameResolvingJob
parameters:
- name: domainNames
value: ${domainBatch}

Some ressources, like ports, provide multiple values as input to a cron subscription. When the batching syntax is used, all the values will be given in seperate arrays, and all the values at an index i belong to the same resource. Therefore, all the arrays will have the same length.

For instance, if you have three ports in the database with the following characteristics:

IndexPortIPProtocol
0221.1.1.1tcp
1802.2.2.2tcp
24433.3.3.3tcp

And a cron subscription like the following:

name: Example cron subscription
input: ALL_TCP_PORTS
batch:
enabled: true # Input batching is enabled
size: 100 # with a max size of 100 items per array per job
cronExpression: "0 0 */7 * *"
job:
name: PortExampleJob
parameters:
- name: targetIps
value: ${ipBatch}
- name: ports
value: ${portBatch}
- name: protocols
value: ${protocolBatch}

You would end up with the following parameters after the injection of values by the backend:

parameters:
- name: targetIps
value: ["1.1.1.1", "2.2.2.2", "3.3.3.3"]
- name: ports
value: [22, 80, 443]
- name: protocols
value: ["tcp", "tcp", "tcp"]

Which would be usable in the following way, in python:

import os
import json

ips = json.loads(os.environ.get("targetIps")) # ['1.1.1.1', '2.2.2.2', '3.3.3.3']
ports = json.loads(os.environ.get("ports")) # [22, 80, 443]
protocols = json.loads(os.environ.get("protocols")) # ['tcp', 'tcp', 'tcp']

The batch size of 100, with only three ports in the database, resulted in the launch of only one job. However, if we had 10 ports in the database, and a batch size of 3, you would end up starting 4 jobs, following the logic

ceiling(10 / 3) => 4

Event Subscriptions

When Red Kite finds some information, either through the output of a Job or through user input, a Finding is emitted. This Finding contains the output information given by the Job or the user. From this information, new Jobs are started that will output more Findings. This is roughly how Red Kite's automation workflow works.

An event subscription allows for the customization of Red Kite's automation workflow by starting any Job on any Finding. These jobs can be started on specified conditions. Also, the output of the finding can be used as a job input, as well as a condition parameter.

Event subscriptions also have a cooldown in seconds. This value ensures that a subscription is not triggered too often for the same ressource. It prevents infinite loops and rescanning the same host right away, for instance.

Event Subscription Syntax

An event subscription can contain these main elements :

ElementDescriptionMandatory
nameThe name of the subscription, for future referenceYes
findingsThe findings to react to. A list of the finding names.Yes
cooldownThe minimum amount of time, in seconds, before a subscription can be retriggered for the same ressourceYes
jobThe job to launch when the finding and conditions are metYes
conditionsThe preconditions to launching the jobNo

N.B. Additonnal details on these elements are given in the following sections

  • The name element can be anything and is only used to distinguish the Subscription from the others.
  • The findings elements must be existing Finding's types. See the Findings documentation for the list.
  • The cooldown is a number in seconds for which to wait before relaunching the job for the same ressource.
  • The job element contains multiple values:
    • name : mandatory, must be an existing Job's type. See the Jobs section for the list of valid values.
    • parameters : optionnal, but almost always needed. It describes the input values of the job by the parameter name and its value in a list.
  • The conditions element is the only non-mandatory main element. If it is not provided, the Job will always be started when the Finding is found. Multiple conditions can be provided in a list. Conditions contain mulitple elements:
    • lhs : The left-hand side operand.
    • operator : The operator to compare the two operands.
    • rhs : The right-hand side operand.

Event Subscription Dynamic Input

You can add dynamic input to an event subscription either by referencing a finding's fields, or by injecting a secret.

You can reference a Finding's output variable by name in a Job parameter's value or in a condition's operand using the following syntax: ${parameterName}. The variable name is case insensitive.

In a finding, you can find dynamic fields in the fields array. The text based dynamic fields' values can be injected in the same way as a regular field, with the ${parameterName} syntax. Simply reference the key part of a dynamic field as the variable name, and its data will be injected.

You can also inject a secret as a parameter value with the ${secrets.secretName} syntax. You can learn more about secrets here.

Event Subscription Simple Example

Here is one of the shortest event subscription possible. It reacts to a HostnameFinding to launch a DomainNameResolvingJob, using the output of the Finding as the domainName parameter value.

Therefore, it will resolve the domain name found in the HostnameFinding to an IP address by launching a DomainNameResolvingJob. It will emit a HostnameIpFinding if it properly resolves the domain name.

The cooldown here represents 23 hours. Therefore, the same hostname can only be resolved as often as every 23 hours through this subscription.

This subscription is already built in Red Kite

name: Domain name resolution
findings:
- HostnameFinding
cooldown: 82800
job:
name: DomainNameResolvingJob
parameters:
- name: domainName
value: ${domainName}

Event Subscription Complex Example

Here is an example of how to start a TcpPortScanningJob when a HostnameIpFinding is emitted. This job will however only start if all the conditions are met. The Finding's found ip must contain the case insensitive string "13.37", and 5 must be greater than or equal to 3.

If the conditions are met when a HostnameIpFinding is emitted, then the job will start with the specified parameters. Note here that the targetIp parameter will contain the value of the Finding's output variable ip.

The Job, in this case, will scan the ${ip}'s 1000 first ports ([1-1000]) as well as the ports 1234, 3389 and 8080. It will do so in 10 parallel threads. Every thread's socket will have a 1 second timeout if it does not respond. We can therefore expect the Job to finish in a maximum of around 100 seconds.

Something similar in behavior, except from the conditions, is already built in Red Kite.

name: My complex subscription
findings:
- HostnameIpFinding
cooldown: 82800
job:
name: TcpPortScanningJob
parameters:
- name: targetIp
value: ${ip}
- name: threads
value: 10
- name: socketTimeoutSeconds
value: 1
- name: portMin
value: 1
- name: portMax
value: 1000
- name: ports
value: [1234, 3389, 8080]
conditions:
- lhs: ${ip}
operator: contains_i
rhs: "13.37"
- lhs: 5
operator: gte
rhs: 3

Mutliple Findings In Subscriptions

An event subscription can be triggered by multiple finding types, as long as the values referenced in the subscription are available in the two finding types.

For instance, a subscription can be triggered from both an HostnameIpFinding as well as an IpFinding and use the overlapping fields.

The HostnameIpFinding:

{
"type": "HostnameIpFinding",
"key": "HostnameIpFinding",
"domainName": "red-kite.io",
"ip": "0.0.0.0"
}

The IpFinding:

{
"type": "IpFinding",
"key": "IpFinding",
"ip": "0.0.0.0"
}

Since both the HostnameIpFinding and the IpFinding have an ip field, it can be used in the subscription.

name: My complex subscription
findings:
- HostnameIpFinding
- IpFinding
cooldown: 82800
job:
name: TcpPortScanningJob
parameters:
- name: targetIp
value: ${ip}
- name: threads
value: 10
- name: socketTimeoutSeconds
value: 1
- name: portMin
value: 1
- name: portMax
value: 1000
- name: ports
value: [1234, 3389, 8080]
conditions:
- lhs: ${ip}
operator: contains_i
rhs: "13.37"
- lhs: 5
operator: gte
rhs: 3

Findings

A finding event is propagated by Red Kite whenever an information comes into play. Every finding type contains information that is specific to it.

It is possible to reference a finding outputted by a job as an input of a new job, as well as a condition operand. All references to a finding's output variable are case insensitive.

To learn more about findings, click here.

Conditions

By default, an event subscription's Job will always start when the specified Finding is found. However, it is not always the desired behavior. Sometimes, the Job should only start if the Finding's output respect some established conditions.

A condition is usually made of three elements: a left-hand side parameter (lhs), an operator, and a right-hand side parameter (rhs). The lhs parameter is compared to the rhs parameter using the operator. A Finding's output variable can be used as a Condition's lhs or rhs parameter when referenced using the syntax ${paramName}.

The primitive type of the parameters must match together and must match with the operator for the condition to return true. The allowed types of an operand are string, number, boolean, or an array of these types. When two arrays are provided, every element of the lhs array will be compared with every element of the rhs array. If all the condtions return true, then the specified job will be started.

OperatorAccepted Operand TypesDescription
gtenumberlhs >= rhs
gtnumberlhs > rhs
ltenumberlhs <= rhs
ltnumberlhs < rhs
equalsstring, number, booleanlhs == rhs
equals_istringCase insensitive, lhs == rhs
containsstringValidates if the lhs string contains the rhs string.
contains_istringValidates if the lhs string contains the rhs string. Case insensitive.
startsWithstringValidates if the lhs string starts with the rhs string.
startsWith_istringValidates if the lhs string starts with the rhs string. Case insensitive.
endsWithstringValidates if the lhs string ends with the rhs string.
endsWith_istringValidates if the lhs string ends with the rhs string. Case insensitive.
not_string, number, booleanPrefix not_ to another operator to have the result negated. For instance, not_equals would be true when two operands are not considered equal.
or_string, number, booleanUse with arrays to change the boolean operator between array elements from and to or.

Arrays of the corresponding type can be used on any operator

By default, conditions are linked with an and operator. It means that all conditions have to be met to launch the job. However, you can use the and and or operators in the condition yaml to change the operator applied on the different conditions. In the following example, even though the condition 4 >= 5 will be false, since it is in an or operator, the whole operator will return true. Since the or operator will return true and 1 <= 2 will be true as well, the job will start.

conditions:
- or:
- and:
- lhs: asdf
operator: endsWith
rhs: df
- lhs: qwerty
operator: startsWith
rhs: qwe
- lhs: 4
operator: gte
rhs: 5
- lhs: 1
operator: lte
rhs: 2

The previous condition would be roughly equivalent to the following python if statement:

if (("qwerty".startswith("qwe") and "asdf".endswith("df")) or 4 >= 5) and 1 <= 2:
print("Job will start")