Skip to content

sangbida/sim-ln

 
 

Repository files navigation

SimLN

SimLN is a simulation tool that can be used to generate realistic payment activity on any lightning network topology. It is intentionally environment-agnostic so that it can be used across many environments - from integration tests to public signets.

This tool is intended to serve developers who are familiar with lightning network development. It may be useful to you if you are:

  • A protocol developer looking to test proposals.
  • An application developer load-testing your application.
  • A signet operator interested in a hands-off way to run an active node.
  • A researcher generating synthetic data for a target topology.

LN Implementation Support

  • LND ✅
  • CLN ✅
  • Eclair ✅️
  • LDK-node 🏗️

See our tracking issue for updates on implementation support (contributions welcome!).

Pre-Requisites

SimLN requires you to "bring your own network" to generate activity on. You will need:

The simulator requires access details for a set of nodes that you have permission to execute commands on. Note that the current version of the simulator uses keysend to execute payments, which must be enabled as follows:

  • LND: --accept-keysend
  • CLN: enabled by default
  • Eclair: --features.keysend=optional

Getting Started

Once you have all the pre-requisites installed, clone the repo:

git clone https://github.com/bitcoin-dev-project/sim-ln
cd sim-ln

Install the CLI:

make install

To run the simulation, create a simulation file sim.json in the working directory (see setup instructions for details) and run:

sim-cli

Run sim-cli -h for details on --data-dir and --sim-file options that allow specifying custom simulation file locations.

Interested in contributing to the project? See CONTRIBUTING for more details.

Simulation File Setup

The required access details will depend on the node implementation.

  • LND:
{
  "id": <node_id>,
  "address": <ip:port or domain:port>,
  "macaroon": <path_to_selected_macaroon>,
  "cert": <path_to_tls_cert>
}
  • CLN:
{ 
  "id": <node_id>,
  "address": <ip:port or domain:port>,
  "ca_cert": <path_to_ca_cert>,
  "client_cert": <path_to_client_cert>,
  "client_key": <path_to_client_key>
}
  • Eclair:
{ 
  "id": <node_id>,
  "base_url": <ip:port or domain:port>,
  "api_username": <username_to_authorize>,
  "api_password": <password_to_authorize>
}

Payment activity can be simulated in two different ways:

  • Random activity: generate random activity on the nodes provided, using the graph topology to determine payment frequency and size.
  • Defined activity: provide descriptions of specific payments that you would like the generator to execute.

Setup - Random Activity

To run the simulator with random activity generation, you just need to provide a set of nodes and the simulator will generate activity based on the topology of the underlying graph. Note that payments will only be sent between the nodes that are provided so that liquidity does not "drain" from the simulation.

{
  "nodes": [
    {
      "id": "Alice",
      "address": "127.0.0.1:10011",
      "macaroon": "/path/admin.macaroon",
      "cert": "/path/tls.cert"
    },
    { 
      "id": "0230a16a05c5ca120136b3a770a2adfdad88a68d526e63448a9eef88bddd6a30d8",
      "address": "localhost:10013",
      "ca_cert": "/path/ca.pem",
      "client_cert": "/path/client.pem",
      "client_key": "/path/client-key.pem"
    },
    { 
      "id": "carol",
      "base_url": "127.0.0.1:8286",
      "api_username": "",
      "api_password": "eclairpw"
    }
  ]
}

Nodes can be identified by an arbitrary string ("Alice", "CLN1", etc) or by their node public key. If a valid public key is provided it must match the public key reported by the node.

There are a few cli flags that can be used to toggle the characteristics of the random activity that is generated:

  • --expected-payment-amount: the approximate average amount that will be sent by nodes, randomness will be introduced such that larger nodes send a wider variety of payment sizes around this expectation.
  • --capacity-multiplier: the number of times over that each node in the network sends their capacity in a calendar month, for example:
    • capacity-multiplier=2 means that each node sends double their capacity in a month.
    • capacity-multiplier=0.5 means that each node sends half their capacity in a month.
  • --fix-seed: a u64 value that allows you to generate random activities deterministically from the provided seed, albeit with some limitations. The simulations are not guaranteed to be perfectly deterministic because tasks complete in slightly different orders on each run of the simulator. With a fixed seed, we can guarantee that the order in which activities are dispatched will be deterministic.

Setup - Defined Activity

If you would like SimLN to generate a specific payments between source and destination nodes, you can provide activity descriptions of the source, destination, frequency and amount for payments that you'd like to execute. Note that source nodes must be contained in nodes, but destination nodes can be any public node in the network (though this may result in liquidity draining over time).

Required fields:

"source": the payer
"destination": the payee
"interval_secs": how often the payments should be sent
"amount_msat": the amount of each payment

Optional fields:

"start_secs": the time to start sending payments
"count": the total number of payments to send

If start_secs is not provided the payments will begin as soon as the simulation starts (default=0)

If count is not provided the payments will continue for as long as the simulation runs (default=None)

The example simulation file below sets up the following simulation:

  • Connect to Alice running LND to generate activity.
  • Connect to Bob running CLN to generate activity.
  • Dispatch 2000 msat payments from Alice to Carol every 1 seconds.
  • Dispatch 140000 msat payments from Bob to Alice every 50 seconds.
  • Dispatch 1000 msat payments from Bob to Dave every 2 seconds.
  • Dispatch 10 payments (5000 msat each) from Erin to Frank at 2 second intervals, starting 20 seconds into the sim.
{
  "nodes": [
    {
      "id": "Alice",
      "address": "localhost:10011",
      "macaroon": "/path/admin.macaroon",
      "cert": "/path/tls.cert"
    },
    {
      "id": "0230a16a05c5ca120136b3a770a2adfdad88a68d526e63448a9eef88bddd6a30d8",
      "address": "127.0.0.1:10013",
      "ca_cert": "/path/ca.pem",
      "client_cert": "/path/client.pem",
      "client_key": "/path/client-key.pem"
    },
    {
      "id": "Erin",
      "address": "localhost:10012",
      "macaroon": "/path/admin.macaroon",
      "cert": "/path/tls.cert"      
    },
    {
      "id": "Frank",
      "address": "localhost:10014",
      "macaroon": "/path/admin.macaroon",
      "cert": "/path/tls.cert"      
    }
  ],
  "activity": [
    {
      "source": "Alice",
      "destination": "02d804ad31429c8cc29e20ec43b4129553eb97623801e534ab5a66cdcd2149dbed",
      "interval_secs": 1,
      "amount_msat": 2000
    },
    {
      "source": "0230a16a05c5ca120136b3a770a2adfdad88a68d526e63448a9eef88bddd6a30d8",
      "destination": "Alice",
      "interval_secs": 50,
      "amount_msat": 140000
    },
    {
      "source": "0230a16a05c5ca120136b3a770a2adfdad88a68d526e63448a9eef88bddd6a30d8",
      "destination": "03232e245059a2e7f6e32d6c4bca97fc4cda935c553ea3693adb3265a19050c3bf",
      "interval_secs": 2,
      "amount_msat": 1000
    },
    {
      "source": "Erin",
      "destination": "Frank",
      "start_secs": 20,
      "count": 10,
      "interval_secs": 2,
      "amount_msat": 5000
    }
  ]
}

Activity sources must reference an id defined in nodes, because the simulator can only send payments from nodes that it controls. Destinations may reference either an id defined in nodes or provide a pubkey or alias of a node in the public network. If the alias provided is not unique in the public network, a pubkey must be used to identify the node.

Simulation Output

A summary of the results will be logged by the simulator. A full list of payments made with their outcomes is available in simulation_{timestamp}.csv within the configured {data_dir}/results.

Run sim-cli -h for details on data directory (--data-dir) and other options including --print-batch-size which affect how simulation outputs are persisted

Lightning Environments

If you're looking to get started with local lightning development, we recommend polar. For larger deployments, see the Scaling Lightning project.

Docker

If you want to run the cli in a containerized environment, see the docker set up docs here

Advanced Usage - Network Simulation

If you are looking to simulate payments are large lightning networks without the resource consumption of setting up a large cluster of nodes, you may be interested in dispatching payments on a simulated network.

To run on a simulated network, you will need to provide the desired topology and channel policies for each edge in the graph. The nodes in the network will be inferred from the edges provided. Simulation of payments on both a mocked and real lightning network is not supported, so the sim_network field is mutually exclusive with the nodes section described above.

The example that follows will execute random payments on a network consisting of three nodes and two channels. You may specify defined activity to execute on the mocked network, though you must refer to nodes by their pubkey (aliases are not yet supported).

{
  "sim_network": [
    {
      "scid": 1,
      "capacity_msat": 250000,
      "node_1": {
        "pubkey": "0344f37d544896dcc95a08ddd9bdfc2b756bf3f91b3f65bce588bd9d0228c24977",
        "max_htlc_count": 483,
        "max_in_flight_msat": 250000,
        "min_htlc_size_msat": 1,
        "max_htlc_size_msat": 100000,
        "cltv_expiry_delta": 40,
        "base_fee": 1000,
        "fee_rate_prop": 100
      },
      "node_2": {
        "pubkey": "020a30431ce58843eedf8051214dbfadb65b107cc598b8277f14bb9b33c9cd026f",
        "max_htlc_count": 15,
        "max_in_flight_msat": 100000,
        "min_htlc_size_msat": 1,
        "max_htlc_size_msat": 50000,
        "cltv_expiry_delta": 80,
        "base_fee": 2000,
        "fee_rate_prop": 500
      }
    },
    {
      "scid": 2,
      "capacity_msat": 100000,
      "node_1": {
        "pubkey": "020a30431ce58843eedf8051214dbfadb65b107cc598b8277f14bb9b33c9cd026f",
        "max_htlc_count": 200,
        "max_in_flight_msat": 100000,
        "min_htlc_size_msat": 1,
        "max_htlc_size_msat": 25000,
        "cltv_expiry_delta": 40,
        "base_fee": 1750,
        "fee_rate_prop": 100
      },
      "node_2": {
        "pubkey": "035c0b392725bb7298d56bf1bcb23634fc509d86a39a8141d435f9d4d6cd4b12eb",
        "max_htlc_count": 15,
        "max_in_flight_msat": 50000,
        "min_htlc_size_msat": 1,
        "max_htlc_size_msat": 50000,
        "cltv_expiry_delta": 80,
        "base_fee": 3000,
        "fee_rate_prop": 5
      }
    }
  ]
}

Note that you need to provide forwarding policies in each direction, because each participant in the channel sets their own forwarding policy and restrictions on their counterparty.

Inclusions and Limitations

The simulator will execute payments on the mocked out network as it would for a network of real nodes. See the inclusions and exclusions listed below for a description of the functionality covered by the simulated network.

Included:

  • Routing Policy Enforcement: mocked channels enforce fee and CLTV requirements.
  • Channel restrictions: mocked channels abide by the in-flight count and value limitations set on channel creation.
  • Liquidity checks: HTLCs are only forwarded if the node has sufficient liquidity in the mocked channel.

Not included:

  • Channel reserve: the required minimum reserve balance is not subtracted from a node's available balance.
  • On chain fees: the simulation does not subtract on chain fees from available liquidity.
  • Dust limits: the simulation node not account for restrictions on dust HTLCs.

Developers

About

Payment activity generator for the lightning network

Resources

License

Contributing

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Rust 96.9%
  • Makefile 1.2%
  • Shell 1.1%
  • Dockerfile 0.8%