Skip to content



This feature is available through the API of Firefly III.

A webhook is a custom callback. Triggered by events in Firefly III a message will be sent to an URL of your choice.


There are currently three available triggers for webhooks in Firefly III:

  • When a transaction is created (TRIGGER_STORE_TRANSACTION)
  • When a transaction is updated (TRIGGER_UPDATE_TRANSACTION)
  • When a transaction is deleted (TRIGGER_DESTROY_TRANSACTION)


When triggered, Firefly III will respond by sending a data package to the URL of your choice. The response can be configured, depending on the trigger.

Transaction create / update / delete

  • Send the transaction involved (RESPONSE_TRANSACTIONS)
  • Send the accounts involved (RESPONSE_ACCOUNTS)
  • Send nothing (RESPONSE_NONE)


All webhook responses will be delivered in JSON. This is the only option:


Create or edit webhooks using the API

You can manage webhooks using the webhook API endpoints.


This is the only way to manage webhooks at the moment. Sorry.


Each webhook comes with a secret when created. The secret is used to generate the signature (see below). You can regenerate the secret but you can't set a value yourself.

Webhook limits

You must submit a https:// URL. HTTP or other protocols are not allowed. You must give webhooks a unique title.

It's not possible to send two or more responses at the same time, for example the transactions AND the accounts. To do this, you must create two separate webhooks. These will have two separate secrets.


When a webhook delivers its payload you can expect (more or less) the following result. The signature is important and explained later on.


These are the headers Firefly III will include with each submitted webhook. The version will match your Firefly III version. The signature has been shortened for clarity.

User-Agent = FireflyIII/5.5.0
Signature = t=1610738765,v1=d62463af1dcdcc...
Accept = application/json
Content-Type = application/json


The body will be something like this.

The UUID is unique for each webhook message. You can use it for debug purposes. The user ID matches the user who created the webhook, and the trigger + response fields tell you why the webhook was fired and what the content of the content field is. The webhook message also contains the original URL of the webhook.

The content block corresponds to the response value. It's either the transaction, an array of accounts or something else entirely. This depends on the configuration of the webhook.

Note that the result sent to you by Firefly III isn't formatted as nicely. A webhook message from Firefly III isn't indented. Note also that in this example some parts of the actual content have been cut away to save space.

    "uuid": "27db119a-c971-423f-9faf-cdae47367fc8",
    "user_id": 1,
    "response": "RESPONSE_TRANSACTIONS",
    "url": "http:\/\/\/",
    "version": "v1",
    "content": {
        "id": 246,
        "created_at": "2021-01-15T20:26:04+01:00",
        "updated_at": "2021-01-15T20:26:04+01:00",
        "user": 1,
        "group_title": null,
        "transactions": [{
            "user": 1,
            "transaction_journal_id": 251
        "links": [{
            "rel": "self",
            "uri": "\/transactions\/246"


This is a typical signature string:


The signature comes in two parts, a timestamp (t) and a versioned signature hash (v). The timestamp is prefixed by t=. Each signature is prefixed by a scheme. Schemes start with v, followed by an integer. Currently, the only valid live signature scheme is v1. Therefore, the signature is prefixed by v1=.

The signature is generated using the HMAC method. The signed string is created by concatenating:

  • The timestamp (as a string)
  • The literal character . (a dot)
  • The unformatted JSON body of the request.

Example: 123456.{"some":"json"}.

The concatenated string is hashed using SHA-3 (256 bits). The shared secret key used for generating the HMAC key is the secret of the webhook itself.

A useful little script to verify webhooks can be found on GitHub.

Last update: 2023-01-29