Skip to content

Topology definition#

Containerlab builds labs based on the topology information that users pass to it. This topology information is expressed as a code contained in the topology definition file which structure is the prime focus of this document.

Topology definition components#

The topology definition file is a configuration file expressed in YAML and has a name pattern of *.clab.yml1. In this document, we take a pre-packaged Nokia SR Linux and Arista cEOS lab and explain the topology definition structure using its definition file srlceos01.clab.yml which is pasted below:

name: srlceos01

topology:
  nodes:
    srl:
      kind: nokia_srlinux
      image: ghcr.io/nokia/srlinux
    ceos:
      kind: arista_ceos
      image: ceos:4.32.0F

  links:
    - endpoints: ["srl:e1-1", "ceos:eth1"]

Note

Containerlab provides a JSON schema file for the topology file. The schema is used to live-validate user's input if a code editor supports this feature.

This topology results in the two nodes being started up and interconnected with each other using a single point-to-point interface:

Let's touch on the key components of the topology definition file used in this example.

Name#

The topology must have a name associated with it. The name is used to distinct one topology from another, to allow multiple topologies to be deployed on the same host without clashes.

name: srlceos01

Its user's responsibility to give labs unique names if they plan to run multiple labs.

The name is a free-formed string, though it is better not to use dashes (-) as they are used to separate lab names from node names.

When containerlab starts the containers, their names will be generated using the following pattern: clab-${lab-name}-${node-name}. The lab name here is used to make the container's names unique between two different labs, even if the nodes are named the same.

Prefix#

It is possible to change the prefix that containerlab adds to node names. The prefix parameter is in charge of that. It follows the below-mentioned logic:

  1. When prefix is not present in the topology file, the default prefix logic applies. Containers will be named as clab-<lab-name>-<node-name>.
  2. When prefix is set to some value, for example, myprefix, this string is used instead of clab, and the resulting container name will be: myprefix-<lab-name>-<node-name>.
  3. When prefix is set to a magic value __lab-name the resulting container name will not have the clab prefix, but will keep the lab name: <lab-name>-<node-name>.
  4. When set to an empty string, the node names will not be prefixed at all. If your node is named mynode, you will get the mynode container in your system.

Warning

In the case of an empty prefix, you have to keep in mind that nodes need to be named uniquely across all labs.

Examples:

name: mylab
prefix: myprefix
nodes:
  n1:
  # <some config>

With a prefix set to myprefix the container name for node n1 will be myprefix-mylab-n1.

name: mylab
prefix: ""
nodes:
  n1:
  # <some config>

When a prefix is set to an empty string, the container name will match the node name - n1.

Note

Even when you change the prefix, the lab directory is still uniformly named using the clab-<lab-name> pattern.

Topology#

The topology object inside the topology definition is the core element of the file. Under the topology element you will find all the main building blocks of a topology such as nodes, kinds, defaults and links.

Nodes#

As with every other topology the nodes are in the center of things. With nodes we define which lab elements we want to run, in what configuration and flavor.

Let's zoom into the two nodes we have defined in our topology:

topology:
  nodes:
    srl:                    # this is a name of the 1st node
      kind: nokia_srlinux
      type: ixr-d2l
      image: ghcr.io/nokia/srlinux
    ceos:                   # this is a name of the 2nd node
      kind: arista_ceos
      image: ceos:4.32.0F

We defined individual nodes under the topology.nodes container. The name of the node is the key under which it is defined. Following the example, our two nodes are named srl and ceos respectively.

Each node can have multiple configuration properties which make containerlab quite a flexible tool. The srl node in our example is defined with the a few node-specific properties:

srl:
  kind: nokia_srlinux
  type: ixr-d2l
  image: ghcr.io/nokia/srlinux

Refer to the node configuration document to meet all other options a node can have.

Although it is absolutely fine to define a node without any links (like in this lab), we usually interconnect the nodes to make topologies. One of containerlab purposes is to make the interconnection of the nodes simple.

Links are defined under the topology.links section of the topology file. Containerlab understands two formats of link definition - brief and extended.
A brief form of a link definition compresses link parameters in a single string and provide a quick way to define a link at the cost of link features available.
A more expressive extended form exposes all link features, but requires more typing if done manually. The extended format is perfect for machine-generated link topologies.

Interface naming#

Containerlab supports two kinds of interface naming: Linux interfaces2 and interface aliases.

The "raw" Linux interface names are the names of the interfaces as they are expected to be seen inside the container (but not necessarily how they look like in the configuration file). Have a look at this topology that features SR Linux and cEOS nodes interconnected with a single link using Linux interface names:

using Linux interface names
# nodes configuration omitted for clarity
topology:
  nodes:
    srl:
    ceos:

  links:
    - endpoints: ["srl:e1-2", "ceos:eth2"] # (1)!
  1. In this example, the srl node has an interface named e1-2 and the ceos node has an interface named eth2. These are the Linux interface names which can be seen if you enter the container' shell and issue ip link.
Aliases#

The downside of using Linux interface names is that they often do not match the interface naming convention used by the Network OS. This is where Interface Aliases feature (added in Containerlab v0.56.0) comes in handy. Imagine we want to create a lab with four different Kinds: SR Linux, vEOS, CSR1000v and vSRX, cabled like this:

A side B side
SR Linux ethernet-1/1 vEOS Ethernet1/1
vSRX ge-0/0/2 vEOS Ethernet1/2
CSR1000v Gi5 vSRX ge-0/0/5
vEOS Ethernet1/3 CSR1000v Gi3

Using the ethX interface naming convention, the topology would look like this:

links:
  - endpoints: ["srl:e1-1", "vEOS:eth1"]
  - endpoints: ["vSRX:eth3", "vEOS:eth2"]
  - endpoints: ["CSR1000v:eth4", "vSRX:eth6"]
  - endpoints: ["vEOS:eth3", "CSR1000v:eth2"]

Note the four different kinds of offset used here on the four different NOSes!

Using aliased interface names, the topology definition becomes much more straightforward:

links:
  - endpoints: ["srl:ethernet-1/1", "vEOS:Ethernet1/1"]
  - endpoints: ["vSRX:ge-0/0/2", "vEOS:Ethernet1/2"]
  - endpoints: ["CSR1000v:Gi5", "vSRX:ge-0/0/5"]
  - endpoints: ["vEOS:Ethernet1/3", "CSR1000v:Gi3"]

Both topology definitions result in the same lab being deployed, but the latter is easier to write and to understand.

Many Kinds (but not all) support interface aliases and the alias names are provided in the respective kind' documentation.

Containerlab transparently maps from interface aliases to Linux interface names, and there's no additional syntax or configuration needed to specify either an interface alias or a Linux interface name in topologies.

How do aliases work?

Internally, interface aliases end up being deterministically mapped to Linux interface names, which conform to Linux interface naming standards: at most 15 characters, spaces and forward slashes (/) not permitted.

Since many NOSes use long interface names (GigabitEthernet1, that's exactly 1 character longer than permitted), and like to use slashes in their interface naming conventions, these NOS interface names cannot be directly used as interface names for the container interfaces created by Containerlab.
For example, SR Linux maps its ethernet-1/2 interface to the Linux interface e1-2. On the other hand, Juniper vSRX maps its ge-0/0/1 interface to eth2.

Brief format#

The brief format of link definition looks as follows.

# nodes configuration omitted for clarity
topology:
  nodes:
    srl:
    ceos:

  links:
    - endpoints: ["srl:ethernet-1/1", "ceos:Ethernet1/1"] #(1)!
    - endpoints: ["srl:e1-2", "ceos:eth2"]
  1. This example features two interface naming conventions: Linux interface names and interface aliases.

As you see, the topology.links element is a list of individual links. The link itself is expressed as pair of endpoints. This might sound complicated, lets use a graphical explanation:

As demonstrated on a diagram above, the links between the containers are the point-to-point links which are defined by a pair of interfaces. The link defined as:

endpoints: ["srl:e1-1", "ceos:eth1"]

will result in a creation of a p2p link between the node named srl and its e1-1 interface and the node named ceos and its eth1 interface. The p2p link is realized with a veth pair.

Extended format#

The extended link format allows a user to set every supported link parameter in a structured way. The available link parameters depend on the Link type and provided below.

veth#

The veth link is the most common link type used in containerlab. It creates a virtual ethernet link between two endpoints where each endpoint refers to a node in the topology.

links:
  - type: veth
    endpoints:
      - node: <NodeA-Name>                  # mandatory
        interface: <NodeA-Interface-Name>   # mandatory
        mac: <NodeA-Interface-Mac>          # optional
        ipv4: <NodeA-IPv4-Address>          # optional e.g. 192.168.0.1/24
        ipv6: <NodeA-IPv6-Address>          # optional e.g. 2001:db8::1/64
      - node: <NodeB-Name>                  # mandatory
        interface: <NodeB-Interface-Name>   # mandatory
        mac: <NodeB-Interface-Mac>          # optional
        ipv4: <NodeB-IPv4-Address>          # optional
        ipv6: <NodeB-IPv6-Address>          # optional
    mtu: <link-mtu>                         # optional
    vars: <link-variables>                  # optional (used in templating)
    labels: <link-labels>                   # optional (used in templating)
mgmt-net#

The mgmt-net link type represents a veth pair that is connected to a container node on one side and to the management network (usually a bridge) instantiated by the container runtime on the other.

  links:
  - type: mgmt-net
    endpoint:
      node: <NodeA-Name>                    # mandatory
      interface: <NodeA-Interface-Name>     # mandatory
      mac: <NodeA-Interface-Mac>            # optional
    host-interface: <interface-name>        # mandatory
    mtu: <link-mtu>                         # optional
    vars: <link-variables>                  # optional (used in templating)
    labels: <link-labels>                   # optional (used in templating)

The host-interface is the desired interface name that will be attached to the management network in the host namespace.

macvlan#

The macvlan link type creates a MACVlan interface with the host-interface as its parent interface. The MACVlan interface is then moved to a node's network namespace and renamed to the endpoint.interface name.

  links:
  - type: macvlan
    endpoint:
      node: <NodeA-Name>                  # mandatory
      interface: <NodeA-Interface-Name>   # mandatory
      mac: <NodeA-Interface-Mac>          # optional
    host-interface: <interface-name>        # mandatory
    mode: <macvlan-mode>                    # optional ("bridge" by default)
    vars: <link-variables>                  # optional (used in templating)
    labels: <link-labels>                   # optional (used in templating)

The host-interface is the name of the existing interface present in the host namespace.

Modes are private, vepa, bridge, passthru and source. The default is bridge.

host#

The host link type creates a veth pair between a container and the host network namespace.
In comparison to the veth type, no bridge or other namespace is required to be referenced in the link definition for a "remote" end of the veth pair.

  links:
  - type: host
    endpoint:
      node: <NodeA-Name>                  # mandatory
      interface: <NodeA-Interface-Name>   # mandatory
      mac: <NodeA-Interface-Mac>          # optional
    host-interface: <interface-name>        # mandatory
    mtu: <link-mtu>                         # optional
    vars: <link-variables>                  # optional (used in templating)
    labels: <link-labels>                   # optional (used in templating)

The host-interface parameter defines the name of the veth interface in the host's network namespace.

vxlan#

The vxlan type results in a vxlan tunnel interface that is created in the host namespace and subsequently pushed into the nodes network namespace.

  links:
    - type: vxlan
      endpoint:                              # mandatory
        node: <Node-Name>                    # mandatory
        interface: <Node-Interface-Name>     # mandatory
        mac: <Node-Interface-Mac>            # optional
      remote: <Remote-VTEP-IP>               # mandatory
      vni: <VNI>                             # mandatory
      dst-port: <VTEP-UDP-Port>              # mandatory
      src-port: <Source-UDP-Port>            # optional
      mtu: <link-mtu>                        # optional
      vars: <link-variables>                 # optional (used in templating)
      labels: <link-labels>                  # optional (used in templating)
vxlan-stitched#

The vxlan-stitched type results in a veth pair linking the host namespace and the nodes namespace and a vxlan tunnel that also terminates in the host namespace. In addition to these interfaces, tc rules are being provisioned to stitch the vxlan tunnel and the host based veth interface together.

  links:
    - type: vxlan-stitch
      endpoint:                              # mandatory
        node: <Node-Name>                    # mandatory
        interface: <Node-Interface-Name>     # mandatory
        mac: <Node-Interface-Mac>            # optional
      remote: <Remote-VTEP-IP>               # mandatory
      vni: <VNI>                             # mandatory
      dst-port: <VTEP-UDP-Port>              # mandatory
      src-port: <Source-UDP-Port>            # optional
      mtu: <link-mtu>                        # optional
      vars: <link-variables>                 # optional (used in templating)
      labels: <link-labels>                  # optional (used in templating)
dummy#

The dummy type creates a dummy interface that provides a virtual network device to route packets through without actually transmitting them.

Such interfaces are useful for testing and debugging purposes where we want to make sure that the NOS detects network ports, but doesn't actually need to send or receive packets via these ports.

  links:
  - type: dummy
    endpoint:
      node: <NodeA-Name>                    # mandatory
      interface: <NodeA-Interface-Name>     # mandatory
      mac: <NodeA-Interface-Mac>            # optional
    mtu: <link-mtu>                         # optional
    vars: <link-variables>                  # optional (used in templating)
    labels: <link-labels>                   # optional (used in templating)
Variables#

Link variables are a way to supply additional link-related information that can be passed to the configuration templates and will be rendered in the topology data json file.

You can provide link variables using link's brief and extended format. When using the brief format, the vars are defined under the link map and they will be available under the link container in the topology json file:

  links:
    - endpoints: [srl1:e1-1, srl2:e1-1]
      vars:
        foo: bar
        baz:
          - one
          - two
          - three
        three:
          a: b
          c: d
"links": [
  {
    "endpoints": {
      "a": {
        "node": "srl1",
        "interface": "e1-1",
        "mac": "aa:c1:ab:12:bb:44",
        "peer": "z"
      },
      "z": {
        "node": "srl2",
        "interface": "e1-1",
        "mac": "aa:c1:ab:96:1c:d1",
        "peer": "a"
      }
    },
    "vars": {
      "baz": [
        "one",
        "two",
        "three"
      ],
      "foo": "bar",
      "three": {
        "a": "b",
        "c": "d"
      }
    }
  }
]

In the extended format, the vars can be defined for the entire link or for each endpoint individually.

  links:
    - type: veth
      endpoints:
        - node: srl1
          interface: e1-2
          vars:
            srl1ep1var1: "val1"
            srl1ep1var2:
              a: "b"
              c: "d"
            srl1ep1var3:
              - "x"
              - "y"
              - "z"
        - node: srl2
          interface: e1-2
          vars:
            srl2ep1var1: "val2"
            srl2ep1var2:
              x: "y"
              z: "a"
            srl2ep1var3:
              - 1
              - 2
              - 3
  "links": [
    {
      "endpoints": {
        "a": {
          "node": "srl1",
          "interface": "e1-2",
          "mac": "aa:c1:ab:56:36:28",
          "vars": {
            "srl1ep1var1": "val1",
            "srl1ep1var2": {
              "a": "b",
              "c": "d"
            },
            "srl1ep1var3": [
              "x",
              "y",
              "z"
            ]
          },
          "peer": "z"
        },
        "z": {
          "node": "srl2",
          "interface": "e1-2",
          "mac": "aa:c1:ab:09:2f:ea",
          "vars": {
            "srl2ep1var1": "val2",
            "srl2ep1var2": {
              "x": "y",
              "z": "a"
            },
            "srl2ep1var3": [
              1,
              2,
              3
            ]
          },
          "peer": "a"
        }
      }
    }
  ]
IP Addresses#

The ipv4 and ipv6 fields allow for you to set the IPv4 and/or IPv6 address on an interface respectively; directly from the topology file.

Note

Currently only the Nokia SR Linux and Cisco IOL kind(s) support this feature. Contributions to add support for other kinds are welcomed.

Refer to the below example, where we configure some addressing on the node interfaces using the brief format where addresses are passed as an ordered list matching the order of which the endpoint interfaces are defined.

name: ip-addr-brief
topology:
  nodes:
    srl1:
      kind: nokia_srlinux
      image: ghcr.io/nokia/srlinux
    srl2:
      kind: nokia_srlinux
      image: ghcr.io/nokia/srlinux
  links:
    - endpoints: ["srl1:e1-1", "srl2:e1-1"]
      ipv4: ["192.168.0.1/24", "192.168.0.2/24"]
      ipv6: ["2001:db8::1/64", "2001:db8::2/64"]
    - endpoints: ["srl1:e1-2", "srl2:e1-2"]
      ipv4: ["192.168.2.1/24"] #(1)!
  1. In this case, only the srl1 node's e1-2 interface will be provided with the ipv4 variable, since the list contains only one entry and the first entry always corresponds to the first endpoint defined in the endpoints list.

    In case you only need to provide the variable to the 2nd endpoint, you keep the first element of the list empty, like so: ipv4: ["", "192.168.2.1/24"].

The extended format also supports providing the variables, in a more structured way:

name: ip-vars-extended
topology:
  nodes:
    srl1:
      kind: nokia_srlinux
      image: ghcr.io/nokia/srlinux
    srl2:
      kind: nokia_srlinux
      image: ghcr.io/nokia/srlinux
  links:
    - type: veth
      endpoints:
        - node: srl1
          interface: e1-2
          ipv4: 192.168.0.1/24
          ipv6: 2001:db8::1/64
        - node: srl2
          interface: e1-2
          ipv4: 192.168.0.2/24
          ipv6: 2001:db8::2/64
    - type: veth
      endpoints:
        - node: srl1
          interface: e1-2
          ipv4: 192.168.2.1/24
        - node: srl2
          interface: e1-2

In both examples, we configure the 192.168.0.0/24, and 2001:db8::/64 subnets on the link between srl1 and srl2's e1-1 interfaces, where the least significant value represents the host, 1 for srl1, and 2 for srl2.

We can also set the IP for only one side, which is shown using IPv4 as an example on the link between srl1 and srl2 on the e1-2 interfaces. Where the IPv4 address 192.168.2.1 is only set for srl1.

Groups#

groups sets the values for the properties of all nodes belonging to the group that you define, it's more flexible than kinds which only sets the properties for nodes of that specific kind.

It is useful to organise your topology, especially in cases where nodes of the same kind may require differing properties such as type or image version.

Values inherited from groups will take precedence over kinds and defaults. In other words, the inheritance model is as follows (from most specific to less specific):

node -> group -> kind -> defaults

For example, the We create separate groups for debian and alpine clients, as well as a group for spines where the nodes will be of type ixrd3l.

topology:
  defaults:
    kind: nokia_srlinux
    image: ghcr.io/nokia/srlinux
  groups:
    spines:
      type: ixrd3l
    apline-clients:
      kind: linux
      image: alpine
    debian-clients:
      kind: linux
      image: debian
  nodes:
    srl1:
      group: spines
    srl2:
      group: spines
    srl3:
    srl4:
    client1:
      group: alpine-clients
    client2:
      group: alpine-clients
    client3:
      group: debian-clients
    client4:
      group: debian-clients

Now with the above example, we can notice:

  • The client nodes in the alpine-clients group will be linux kind and run the alpine image.
  • The client nodes in the debian-clients group will be linux kind and run the debian image.
  • The nodes in the spines group will be ixrd3l chassis, but still inherit the kind and image from defaults.
  • The srl3 and srl4 don't belong to any group, so they will inherit their properties from the defaults.

Kinds#

Kinds define the behavior and the nature of a node, it says if the node is a specific containerized Network OS, virtualized router or something else. We go into details of kinds in its own document section, so here we will discuss what happens when kinds section appears in the topology definition:

topology:
  kinds:
    nokia_srlinux:
      type: ixr-d2l
      image: ghcr.io/nokia/srlinux
  nodes:
    srl1:
      kind: nokia_srlinux
    srl2:
      kind: nokia_srlinux
    srl3:
      kind: nokia_srlinux

In the example above the topology.kinds element has nokia_srlinux kind referenced. With this, we set some values for the properties of the nokia_srlinux kind. A configuration like that says that nodes of nokia_srlinux kind will also inherit the properties (type, image) defined on the kind level.

Essentially, what kinds section allows us to do is to shorten the lab definition in cases when we have a number of nodes of a same kind. All the nodes (srl1, srl2, srl3) will have the same values for their type and image properties.

Consider how the topology would have looked like without setting the kinds object:

topology:
  nodes:
    srl1:
      kind: nokia_srlinux
      type: ixr-d2l
      image: ghcr.io/nokia/srlinux
    srl2:
      kind: nokia_srlinux
      type: ixr-d2l
      image: ghcr.io/nokia/srlinux
    srl3:
      kind: nokia_srlinux
      type: ixr-d2l
      image: ghcr.io/nokia/srlinux

A lot of unnecessary repetition is eliminated when we set nokia_srlinux kind properties on kind level.

Defaults#

kinds set the values for the properties of a specific kind, whereas with the defaults container it is possible to set values globally.

For example, to set the environment variable for all the nodes of a topology:

topology:
  defaults:
    env:
      MYENV: VALUE
  nodes:
    srl1:
    srl2:
    srl3:

Now every node in this topology will have environment variable MYENV set to VALUE.

Settings#

Global containerlab settings are defined in settings container. The following settings are supported:

Certificate authority#

Global certificate authority settings section allows users to tune certificate management in containerlab. Refer to the Certificate management doc for more details.

Environment variables#

Topology definition file may contain environment variables anywhere in the file. The syntax is the same as in the bash shell:

name: linux

topology:
  nodes:
    l1:
      kind: linux
      image: alpine:${ALPINE_VERSION:=3}

In the example above, the ALPINE_VERSION environment variable is used to set the version of the alpine image. If the variable is not set, the value of 3 will be used. The following syntax is used to expand the environment variable:

Expression Meaning
${var} Value of var (same as $var)
${var-$DEFAULT} If var not set, evaluate expression as $DEFAULT
${var:-$DEFAULT} If var not set or is empty, evaluate expression as $DEFAULT
${var=$DEFAULT} If var not set, evaluate expression as $DEFAULT
${var:=$DEFAULT} If var not set or is empty, evaluate expression as $DEFAULT
${var+$OTHER} If var set, evaluate expression as $OTHER, otherwise as empty string
${var:+$OTHER} If var set, evaluate expression as $OTHER, otherwise as empty string
$$var Escape expressions. Result will be $var.

Magic Variables#

Magic variables are special strings that get replaced with actual values during the topology parsing.to make your lab configurations more dynamic and less verbose. These variables are surrounded by double underscores (__variable__) and can be seen in some of the advanced topology examples.

These variables can be used in startup-config paths, bind paths, and exec commands. They are replaced with actual values during lab deployment:

Variable Description Example Usage Expands To
__clabNodeName__ Current node's short name startup-config: cfg/__clabNodeName__.cfg cfg/node1.cfg (for node named "node1")
__clabNodeDir__ Path to the node's lab directory binds: __clabNodeDir__/conf:/conf clab-mylab/node1/conf:/conf
__clabDir__ Path to the lab's main directory binds: __clabDir__/data.json:/data.json:ro clab-mylab/data.json:/data.json:ro

Here are some practical examples when using magic variables can greatly simplify your topology definitions:

A common pattern found in many labs is to have a separate startup configuration file for each node. For regular network nodes it can be provided using startup-config property, for Linux containers it is often done using the file binds.

Regardless of the target node, at the end of the day, these startup configuration files are often named after the node they belong to. Using __clabNodeName__ magic variable, we can simplify the topology definition and avoid repeating the node names in the config file paths by setting the startup-config or binds property once in the defaults or kinds section and then containerlab will take care of resolving it to the actual node.

Consider the following example where we set the startup-config property in the defaults section:

name: mylab
topology:
  defaults:
    startup-config: configs/__clabNodeName__.cfg
  nodes:
    router1:
    router2:

Both router1 and router2 nodes will get their own startup configuration files: configs/router1.cfg and configs/router2.cfg respectively.

Using the __clabNodeDir__ and __clabDir__ magic variables, it is possible to bind node-specific files as well as lab-wide shared files into the nodes.

name: mylab
topology:
  nodes:
    node1:
      binds:
        # Node-specific files
        - __clabNodeDir__/custom.conf:/etc/custom.conf
        # Lab-wide shared files
        - __clabDir__/shared-data.json:/shared.json:ro

Another popular use case for the __clabNodeName__ magic variable is to customize the exec commands on a per-node basis.

name: mylab
topology:
  nodes:
    node1:
      exec:
        - echo "Node __clabNodeName__ started"  # Will output "Node node1 started"

Generated topologies#

To further simplify parametrization of the topology files, containerlab allows users to template the topology files using Go Template engine.

Using templating approach it is possible to create a lab template and instantiate different labs from it, by simply changing the variables in the variables file.

To help you get started, we created the following lab examples which demonstrate how topology templating can be used:


  1. if the filename has .clab.yml or -clab.yml suffix, the YAML file will have autocompletion and linting support in VSCode editor. 

  2. also referred to as "mapped" or "raw" interfaces in some parts of the documentation