Files
bal-electrum-plugin/docs/inheritance-options.md
2026-06-20 09:50:02 -04:00

13 KiB
Raw Permalink Blame History

BAL — Inheritance Options Guide

How the Bitcoin After Life Electrum plugin reacts to every change you can make to your will: changing the date (earlier / later), adding or removing an heir, changing percentages, fees or willexecutors — and what happens to the transactions held by the willexecutor servers.

This guide describes the actual behaviour of the code (core/will.pyis_will_valid / check_willexecutors_and_heirs and gui/qt/window.pybuild_inheritance_transaction). It is meant for end users and for anyone who wants to understand the onchain consequences of each action.


1. The mental model in one paragraph

Your will is a tree of presigned Bitcoin transactions. Each leaf transaction sends your coins to your heirs and is timelocked (nLockTime) so it can only be broadcast after a future date/block. A copy of each signed transaction is handed to one or more willexecutor servers. While you are alive you periodically prove you are alive (the CheckAlive threshold). When you change anything in your will, BAL must decide between three outcomes:

  1. Do nothing — the will is still coherent.
  2. Rebuild (reprepare + resign, no onchain cost) — the will changed but nothing dangerous is already committed.
  3. Invalidate onchain first (costs a real Bitcoin fee) — a previously signed/sent transaction must be neutralised by spending its inputs, before a new will can safely replace it.

The whole point of rule 3 is safety: a willexecutor must never be able to broadcast an old transaction that would execute your inheritance too early.


2. Transaction states (status flags)

Every will item (WillItem) carries a set of boolean status flags. The most important ones:

Status Meaning Set when
VALID The item is the current, usable plan default True; cleared by INVALIDATED/REPLACED/CONFIRMED/PENDING
COMPLETE (Signed) The transaction has been signed after you press Sign
PUSHED The signed tx was sent to the willexecutor(s) after Broadcast to executors
CHECKED The willexecutor confirmed it holds the tx after a successful server Check (implies PUSHED)
CHECK_FAIL The server check failed a queried executor did not return the tx
PUSH_FAIL Sending to the executor failed cleared when PUSHED becomes true
CONFIRMED The tx is mined onchain seen onchain with height > 0
PENDING The tx is in the mempool (height 0) seen onchain, not yet mined
INVALIDATED Its inputs were spent → it can never confirm invalidation tx created / inputs gone
REPLACED Superseded by a child tx with earlier locktime a replacing child was found
EXPIRED Its locktime is already in the past relative to the check date check_will_expired

Flag transitions enforced by set_status (the safety rules baked in the code):

  • Setting INVALIDATED / REPLACED / CONFIRMED / PENDING → clears VALID.
  • Setting CONFIRMED / PENDING → clears INVALIDATED.
  • Setting PUSHED → clears PUSH_FAIL and CHECK_FAIL.
  • Setting CHECKED → implies PUSHED (and clears PUSH_FAIL).

How states map to row colour in the list

State (first match wins) Colour Hex
CHECK_FAIL red #e83845
INVALIDATED / REPLACED grey (muted)
CONFIRMED green (confirmed on server / chain)
COMPLETE (signed, not yet pushed) blue #2bc8ed
VALID (prepared, not signed) default

Note (v0.3.3 fix): a will that is signed but not yet broadcast (COMPLETE and not PUSHED) is not queried on the server, so it stays blue instead of turning red. Only PUSHED wills are serverchecked.


3. The decision flow

When you press Prepare (or on the periodic Check, or when Electrum closes), BAL runs is_will_valid. Depending on what it finds it raises a specific exception, and each exception maps to one action.

📊 A styled version of this guide (with a live diagram) is in inheritance-options.html — open it via GitHub Pages or download and open it in any browser.

The static diagram below renders everywhere; the Mermaid block after it renders live on GitHub.

BAL inheritance change decision flow

flowchart TD
    A([You change something & press Prepare / Check]) --> B{Heirs defined?}
    B -- No --> Z1[/Show: "Heirs are not defined" — stop/]
    B -- Yes --> C{Check-Alive threshold<br/>in the future?}
    C -- No, it's in the past --> INV1[[Invalidate on-chain<br/>CheckAliveError]]
    C -- Yes --> D{Any VALID tx with<br/>locktime earlier than<br/>the new date?}

    D -- "Yes (you moved the date EARLIER / anticipate)" --> E{Was that tx already<br/>signed or sent?}
    E -- "Not signed yet" --> R1[[Rebuild only<br/>no on-chain cost]]
    E -- "Signed / sent" --> INV2[[Invalidate on-chain FIRST<br/>WillExpired]]

    D -- "No" --> F{Will-executor / fee /<br/>heirs unchanged?}
    F -- "Fee changed" --> R2[[Rebuild<br/>TxFeesChanged]]
    F -- "Will-executor changed/absent" --> R3[[Rebuild<br/>WillExecutorNotPresent / Change]]
    F -- "Heir added or removed,<br/>% or address changed" --> G{Is it a POSTPONE of an<br/>already signed/sent tx?}

    G -- "Yes (date later + signed/sent)" --> INV3[[Invalidate on-chain FIRST,<br/>then rebuild — WillPostponed]]
    G -- "No (never signed, or pure heir/% change)" --> R4[[Rebuild only<br/>HeirNotFound / HeirChange]]

    F -- "Nothing changed" --> OK([Will still coherent — do nothing])

    R1 --> SIGN
    R2 --> SIGN
    R3 --> SIGN
    R4 --> SIGN
    SIGN([Re-sign the new transactions]) --> PUSH([Broadcast to will-executors])

    INV1 --> SB
    INV2 --> SB
    INV3 --> SB
    SB([Sign & broadcast the INVALIDATION tx on-chain]) --> WAIT{Invalidation<br/>confirmed?}
    WAIT -- "Yes" --> REBUILD([Press Prepare again → build the new will])
    REBUILD --> SIGN

4. Every option, explained

4.1 Changing the CheckAlive date / heir locktime

The locktime is the future moment from which a transaction becomes spendable by the heir. BAL compares the requested locktime against the locktime frozen inside the alreadysigned transaction (w.tx.locktime), which is exactly what the willexecutors hold — not the inmemory copy.

You do… Tx already signed/sent? Result Onchain fee?
Move date LATER (postpone) No (never signed) Plain rebuild (HeirNotFound fallthrough) No
Move date LATER (postpone) Yes Invalidate first, then rebuild (WillPostponed) Yes
Move date EARLIER (anticipate) any Old tx becomes expiredinvalidate (WillExpired) Yes
CheckAlive threshold already passed Invalidate (CheckAliveError) Yes

Why postpone needs an onchain invalidation: the willexecutor still holds the old transaction with the earlier locktime. If you simply resigned a later one, a malicious or buggy executor could still broadcast the old one as soon as its earlier locktime is reached — executing your inheritance too soon. Spending the old transaction's inputs onchain makes the old tx unminable. The plugin tells you this explicitly and offers to build the invalidation tx.

Why anticipate is also onchain: moving the date earlier makes the current committed tx expired relative to the new check date; the safe path is the same — invalidate, then rebuild.

4.2 Adding an heir

check_willexecutors_and_heirs walks every heir in the current set; an heir present in heirs but not yet found in the will raises HeirNotFoundException.

  • Result: rebuild (reprepare + resign).
  • Onchain fee: Nounless the will being changed was already signed/sent and the change also moves a locktime later (then the postpone rule in 4.1 applies).

4.3 Removing an heir

The will still carries an heir that is no longer in your current heirs set → HeirNotFoundException (the removedheir branch).

  • Result: rebuild, so the removed heir disappears from the new transactions.
  • Onchain fee: No for a will that was only prepared. If the old will was already signed/sent, you must invalidate it onchain first (same safety reasoning as a postpone), then rebuild.

v0.3.2 fix: removing an heir is now correctly detected on Check and on Electrum close, not only on Prepare.

4.4 Changing an heir's percentage or address

If the stored heir [address, amount/percentage] differs from the current one, the will is no longer coherent → it is treated like an heir change (HeirChangeException / HeirNotFound).

  • Result: rebuild with the new amounts.
  • Onchain fee: No (unless the old will was signed/sent → invalidate first).

Reminder shown by the plugin: “In the inheritance process the entire wallet is always fully emptied” — the amounts across all heirs must add up so that the whole spendable balance is distributed; otherwise an AmountException warns you to adjust.

4.5 Changing the transaction fee (sat/byte)

Each will item stores the fee rate it was built with. A different rate raises TxFeesChangedException.

  • Result: rebuild at the new fee rate.
  • Onchain fee: No to rebuild (you only pay when the inheritance — or an invalidation — is actually broadcast onchain).

4.6 Changing or removing a willexecutor

  • A selected willexecutor that the will does not reference raises WillExecutorNotPresent.

  • A willexecutor whose details changed raises WillexecutorChangeException.

  • Running with “no willexecutor” but no backup transaction raises NoWillExecutorNotPresent.

  • Result: rebuild and redistribute to the (new) executor set.

  • Onchain fee: No to rebuild. The new signed transactions are simply pushed to the new/updated executors; the old executor will eventually fail its own check and drop the obsolete tx.

4.7 Nothing changed

If heirs, percentages, fees, executors and locktimes all still match the signed transactions, is_will_valid returns True and nothing happens — your will stays exactly as broadcast to the executors.


5. What happens on the willexecutor servers

Your action Effect on the servers
Prepare (rebuild) Nothing yet — new txs exist only locally until you Sign + Broadcast.
Sign Still local; tx becomes COMPLETE (blue).
Broadcast to executors The signed txs are uploaded; items become PUSHED.
Check Each PUSHED will is queried; success → CHECKED (green), failure → CHECK_FAIL (red).
Invalidate (onchain) You spend the committed inputs on the Bitcoin network. Once confirmed, the executor's stored tx can no longer be mined; on the next check it is dropped / shown invalidated.
Rebroadcast a new will Executors replace the obsolete copy with the new signed tx.

A row turning red (CHECK_FAIL) after a Check means a willexecutor that should hold your transaction did not return it — reBroadcast, or rebuild, to fix it. A row that is merely blue is signedbutnotyetsent and is perfectly normal.


6. Quick reference — does it cost a Bitcoin fee?

Change Rebuild? Onchain invalidation (real fee)?
Add heir (will only prepared)
Remove heir (will only prepared)
Change % / address (only prepared)
Change fee rate
Change / remove willexecutor
Move date earlier (anticipate) after yes
Move date later (postpone) — will signed/sent after yes
Move date later (postpone) — will only prepared
CheckAlive threshold already in the past after yes
Any change to an already signed/sent will after yes (invalidate first)
Nothing changed

7. Golden rules

  1. Before it's signed, changing anything is free — just Prepare again.
  2. After it's signed/sent, moving the date or otherwise replacing it requires an onchain invalidation first (a small Bitcoin fee) so an old transaction can never be executed early.
  3. Always finish with Sign → Broadcast → Check so the willexecutors hold the current plan (green), not an obsolete one.
  4. The wallet is always fully emptied by the inheritance, so heir amounts must add up.

This document reflects BAL plugin v0.3.3. Behaviour is derived directly from core/will.py and gui/qt/window.py.