diff --git a/content/concept-reserves.md b/content/concept-reserves.md
index a115a310b38..3a1c598c732 100644
--- a/content/concept-reserves.md
+++ b/content/concept-reserves.md
@@ -25,6 +25,7 @@ Many objects in the ledger are owned by a particular address, and count toward t
- [Held Payments (Escrow)](reference-ledger-format.html#escrow) are owned by the address that placed them.
- [Payment Channels](tutorial-paychan.html) are owned by the address that created them.
- [Owner directories](reference-ledger-format.html#directorynode) list all the ledger objects that contribute to an address's owner reserve. However, the owner directory itself does not count towards the reserve.
+- Checks are owned by the address that created them (the sender, not the destination).
#### Owner Reserve Edge Cases
diff --git a/content/ledger-objects/accountroot.md b/content/ledger-objects/accountroot.md
new file mode 100644
index 00000000000..336c00fc16d
--- /dev/null
+++ b/content/ledger-objects/accountroot.md
@@ -0,0 +1,69 @@
+## AccountRoot
+[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L27 "Source")
+
+The `AccountRoot` object type describes a single _account_ object. Example `AccountRoot` object:
+
+```json
+{
+ "Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
+ "AccountTxnID": "0D5FB50FA65C9FE1538FD7E398FFFE9D1908DFA4576D8D7A020040686F93C77D",
+ "Balance": "148446663",
+ "Domain": "6D64756F31332E636F6D",
+ "EmailHash": "98B4375E1D753E5B91627516F6D70977",
+ "Flags": 8388608,
+ "LedgerEntryType": "AccountRoot",
+ "MessageKey": "0000000000000000000000070000000300",
+ "OwnerCount": 3,
+ "PreviousTxnID": "0D5FB50FA65C9FE1538FD7E398FFFE9D1908DFA4576D8D7A020040686F93C77D",
+ "PreviousTxnLgrSeq": 14091160,
+ "Sequence": 336,
+ "TransferRate": 1004999999,
+ "index": "13F1A95D7AAB7108D5CE7EEAF504B2894B8C674E6D68499076441C4837282BF8"
+}
+```
+
+The `AccountRoot` object has the following fields:
+
+| Field | JSON Type | [Internal Type][] | Description |
+|:------------------------------|:----------|:------------------|:-------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0061`, mapped to the string `AccountRoot`, indicates that this is an AccountRoot object. |
+| `Account` | String | AccountID | The identifying address of this account, such as rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn. |
+| `Balance` | String | Amount | The account's current [XRP balance in drops][XRP, in drops], represented as a string. |
+| [`Flags`](#accountroot-flags) | Number | UInt32 | A bit-map of boolean flags enabled for this account. |
+| `OwnerCount` | Number | UInt32 | The number of objects this account owns in the ledger, which contributes to its owner reserve. |
+| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
+| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
+| `Sequence` | Number | UInt32 | The sequence number of the next valid transaction for this account. (Each account starts with Sequence = 1 and increases each time a transaction is made.) |
+| `AccountTxnID` | String | Hash256 | _(Optional)_ The identifying hash of the transaction most recently submitted by this account. |
+| `Domain` | String | VariableLength | _(Optional)_ A domain associated with this account. In JSON, this is the hexadecimal for the ASCII representation of the domain. |
+| `EmailHash` | String | Hash128 | _(Optional)_ The md5 hash of an email address. Clients can use this to look up an avatar through services such as [Gravatar](https://en.gravatar.com/). |
+| `MessageKey` | String | VariableLength | _(Optional)_ A public key that may be used to send encrypted messages to this account. In JSON, uses hexadecimal. No more than 33 bytes. |
+| `RegularKey` | String | AccountID | _(Optional)_ The address of a keypair that can be used to sign transactions for this account instead of the master key. Use a [SetRegularKey transaction][] to change this value. |
+| `TickSize` | Number | UInt8 | _(Optional)_ How many significant digits to use for exchange rates of Offers involving currencies issued by this address. Valid values are `3` to `15`, inclusive. _(Requires the [TickSize amendment](reference-amendments.html#ticksize).)_ |
+| `TransferRate` | Number | UInt32 | _(Optional)_ A [transfer fee](https://ripple.com/knowledge_center/transfer-fees/) to charge other users for sending currency issued by this account to each other. |
+| `WalletLocator` | String | Hash256 | _(Optional)_ **DEPRECATED**. Do not use. |
+| `WalletSize` | Number | UInt32 | _(Optional)_ **DEPRECATED**. Do not use. |
+
+### AccountRoot Flags
+
+There are several options which can be either enabled or disabled for an account. These options can be changed with an [AccountSet transaction][]. In the ledger, flags are represented as binary values that can be combined with bitwise-or operations. The bit values for the flags in the ledger are different than the values used to enable or disable those flags in a transaction. Ledger flags have names that begin with _lsf_.
+
+AccountRoot objects can have the following flag values:
+
+| Flag Name | Hex Value | Decimal Value | Description | Corresponding [AccountSet Flag](reference-transaction-format.html#accountset-flags) |
+|-----------|-----------|---------------|-------------|-------------------------------|
+| lsfPasswordSpent | 0x00010000 | 65536 | Indicates that the account has used its free SetRegularKey transaction. | (None) |
+| lsfRequireDestTag | 0x00020000 | 131072 | Requires incoming payments to specify a Destination Tag. | asfRequireDest |
+| lsfRequireAuth | 0x00040000 | 262144 | This account must individually approve other users for those users to hold this account's issuances. | asfRequireAuth |
+| lsfDisallowXRP | 0x00080000 | 524288 | Client applications should not send XRP to this account. Not enforced by `rippled`. | asfDisallowXRP |
+| lsfDisableMaster | 0x00100000 | 1048576 | Disallows use of the master key to sign transactions for this account. | asfDisableMaster |
+| lsfNoFreeze | 0x00200000 | 2097152 | This address cannot freeze trust lines connected to it. Once enabled, cannot be disabled. | asfNoFreeze |
+| lsfGlobalFreeze | 0x00400000 | 4194304 | All assets issued by this address are frozen. | asfGlobalFreeze |
+| lsfDefaultRipple | 0x00800000 | 8388608 | Enable [rippling](concept-noripple.html) on this addresses's trust lines by default. Required for issuing addresses; discouraged for others. | asfDefaultRipple |
+
+### AccountRoot ID Format
+
+The ID of an AccountRoot object is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The Account space key (`0x0061`)
+* The AccountID of the account
diff --git a/content/ledger-objects/amendments.md b/content/ledger-objects/amendments.md
new file mode 100644
index 00000000000..562ec8ec3b3
--- /dev/null
+++ b/content/ledger-objects/amendments.md
@@ -0,0 +1,56 @@
+## Amendments
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/LedgerFormats.cpp#L110-L113 "Source")
+
+The `Amendments` object type contains a list of [Amendments](concept-amendments.html) that are currently active. Each ledger version contains **at most one** `Amendments` object.
+
+Example `Amendments` object:
+
+```json
+{
+ "Majorities": [
+ {
+ "Majority": {
+ "Amendment": "1562511F573A19AE9BD103B5D6B9E01B3B46805AEC5D3C4805C902B514399146",
+ "CloseTime": 535589001
+ }
+ }
+ ],
+ "Amendments": [
+ "42426C4D4F1009EE67080A9B7965B44656D7714D104A72F9B4369F97ABF044EE",
+ "4C97EBA926031A7CF7D7B36FDE3ED66DDA5421192D63DE53FFB46E43B9DC8373",
+ "6781F8368C4771B83E8B821D88F580202BCB4228075297B19E4FDC5233F1EFDC",
+ "740352F2412A9909880C23A559FCECEDA3BE2126FED62FC7660D628A06927F11"
+ ],
+ "Flags": 0,
+ "LedgerEntryType": "Amendments",
+ "index": "7DB0788C020F02780A673DC74757F23823FA3014C1866E72CC4CD8B226CD6EF4"
+}
+```
+
+| Name | JSON Type | [Internal Type][] | Description |
+|-------------------|-----------|-------------------|-------------|
+| `Amendments` | Array | STI_VECTOR256 | _(Optional)_ Array of 256-bit [amendment IDs](concept-amendments.html#about-amendments) for all currently-enabled amendments. If omitted, there are no enabled amendments. |
+| `Majorities` | Array | STI_ARRAY | _(Optional)_ Array of objects describing the status of amendments that have majority support but are not yet enabled. If omitted, there are no pending amendments with majority support. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags. No flags are defined for the Amendments object type, so this value is always `0`. |
+| `LedgerEntryType` | String | UInt16 | The value `0x0066`, mapped to the string `Amendments`, indicates that this object describes the status of amendments to the XRP Ledger. |
+
+Each member of the `Majorities` field, if it is present, is an object with one field, `Majority`, whose contents are a nested object with the following fields:
+
+| Name | JSON Type | [Internal Type][] | Description |
+|-------------------|-----------|-------------------|-------------|
+| `Amendment` | String | Hash256 | The Amendment ID of the pending amendment. |
+| `CloseTime` | Number | UInt32 | The [`close_time` field](#header-format) of the ledger version where this amendment most recently gained a majority. |
+
+In the [amendment process](concept-amendments.html#amendment-process), a consensus of validators adds a new amendment to the `Majorities` field using an [EnableAmendment][] pseudo-transaction with the `tfGotMajority` flag when 80% or more of validators support it. If support for a pending amendment goes below 80%, an [EnableAmendment][] pseudo-transaction with the `tfLostMajority` flag removes the amendment from the `Majorities` array. If an amendment remains in the `Majorities` field for at least 2 weeks, an [EnableAmendment][] pseudo-transaction with no flags removes it from `Majorities` and permanently adds it to the `Amendments` field.
+
+**Note:** Technically, all transactions in a ledger are processed based on which amendments are enabled in the ledger version immediately before it. While applying transactions to a ledger version where an amendment becomes enabled, the rules don't change mid-ledger. After the ledger is closed, the next ledger uses the new rules as defined by any new amendments that applied.
+
+### Amendments ID Format
+
+The `Amendments` object ID is the hash of the `Amendments` space key (`0x0066`) only. This means that the ID of the `Amendments` object in a ledger is always:
+
+```
+7DB0788C020F02780A673DC74757F23823FA3014C1866E72CC4CD8B226CD6EF4
+```
+
+(Don't mix up the ID of the `Amendments` ledger object type with the Amendment ID of an individual amendment.)
diff --git a/content/ledger-objects/check.md b/content/ledger-objects/check.md
new file mode 100644
index 00000000000..f000a1b4378
--- /dev/null
+++ b/content/ledger-objects/check.md
@@ -0,0 +1,54 @@
+## Check
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/LedgerFormats.cpp#L158 "Source")
+
+_Requires the [Checks Amendment](reference-amendments.html#checks)._
+
+A `Check` object describes a check, similar to a paper personal check, which can be cashed by its destination to get money from its sender. (The potential payment has already been approved by its sender, but no money moves until it is cashed. Unlike an [Escrow](concept-escrow.html), the money for a Check is not set aside, so cashing the Check could fail due to lack of funds.) Example `Check` object:
+
+```json
+{
+ "Account": "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination": "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationNode": "0000000000000000",
+ "DestinationTag": 1,
+ "Expiration": 570113521,
+ "Flags": 0,
+ "InvoiceID": "46060241FABCF692D4D934BA2A6C4427CD4279083E38C77CBE642243E43BE291",
+ "LedgerEntryType": "Check",
+ "OwnerNode": "0000000000000000",
+ "PreviousTxnID": "5463C6E08862A1FAE5EDAC12D70ADB16546A1F674930521295BC082494B62924",
+ "PreviousTxnLgrSeq": 6,
+ "SendMax": "100000000",
+ "Sequence": 2,
+ "index": "49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0"
+}
+```
+
+A `Check` object has the following fields:
+
+| Field | JSON Type | [Internal Type][] | Description |
+|:--------------------|:-----------------|:------------------|:----------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0043`, mapped to the string `Check`, indicates that this object is a Check object. |
+| `Account` | String | Account | The sender of the Check. Cashing the Check debits this address's balance. |
+| `Destination` | String | Account | The intended recipient of the Check. Only this address can cash the Check, using a [CheckCash transaction][]. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags. No flags are defined for Checks, so this value is always `0`. |
+| `OwnerNode` | String | UInt64 | A hint indicating which page of the sender's owner directory links to this object, in case the directory consists of multiple pages. **Note:** The object does not contain a direct link to the owner directory containing it, since that value can be derived from the `Account`. |
+| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
+| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
+| `SendMax` | String or Object | Amount | The maximum amount of currency this Check can debit the sender. If the Check is successfully cashed, the destination is credited in the same currency for up to this amount. |
+| `Sequence` | Number | UInt32 | The sequence number of the [CheckCreate transaction][] that created this check. |
+| `DestinationNode` | String | UInt64 | _(Optional)_ A hint indicating which page of the destination's owner directory links to this object, in case the directory consists of multiple pages. |
+| `DestinationTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the destination for this Check, such as a hosted recipient at the destination address. |
+| `Expiration` | Number | UInt32 | _(Optional)_ Indicates the time after which this Check is considered expired. See [Specifying Time](reference-rippled.html#specifying-time) for details. |
+| `InvoiceID` | String | Hash256 | _(Optional)_ Arbitrary 256-bit hash provided by the sender as a specific reason or identifier for this Check. |
+| `SourceTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the source for this Check, such as a hosted recipient at the sender's address. |
+
+
+### Check ID Format
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/Indexes.cpp#L193-L200 "Source")
+
+The ID of a `Check` object is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The Escrow space key (`0x0043`)
+* The AccountID of the sender of the [CheckCreate transaction][] that created the `Check` object
+* The Sequence number of the [CheckCreate transaction][] that created the `Check` object
diff --git a/content/ledger-objects/directorynode.md b/content/ledger-objects/directorynode.md
new file mode 100644
index 00000000000..9a363443d43
--- /dev/null
+++ b/content/ledger-objects/directorynode.md
@@ -0,0 +1,94 @@
+## DirectoryNode
+[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L44 "Source")
+
+The `DirectoryNode` object type provides a list of links to other objects in the ledger's state tree. A single conceptual _Directory_ takes the form of a doubly linked list, with one or more DirectoryNode objects each containing up to 32 [IDs](#tree-format) of other objects. The first object is called the root of the directory, and all objects other than the root object can be added or deleted as necessary.
+
+There are two kinds of Directories:
+
+* **Owner directories** list other objects owned by an account, such as `RippleState` or `Offer` objects.
+* **Offer directories** list the offers available in the distributed exchange. A single Offer directory contains all the offers that have the same exchange rate for the same issuances.
+
+Example Directories:
+
+
+
+*Offer Directory*
+
+```json
+{
+ "ExchangeRate": "4F069BA8FF484000",
+ "Flags": 0,
+ "Indexes": [
+ "AD7EAE148287EF12D213A251015F86E6D4BD34B3C4A0A1ED9A17198373F908AD"
+ ],
+ "LedgerEntryType": "DirectoryNode",
+ "RootIndex": "1BBEF97EDE88D40CEE2ADE6FEF121166AFE80D99EBADB01A4F069BA8FF484000",
+ "TakerGetsCurrency": "0000000000000000000000000000000000000000",
+ "TakerGetsIssuer": "0000000000000000000000000000000000000000",
+ "TakerPaysCurrency": "0000000000000000000000004A50590000000000",
+ "TakerPaysIssuer": "5BBC0F22F61D9224A110650CFE21CC0C4BE13098",
+ "index": "1BBEF97EDE88D40CEE2ADE6FEF121166AFE80D99EBADB01A4F069BA8FF484000"
+}
+```
+
+*Owner Directory*
+
+```json
+{
+ "Flags": 0,
+ "Indexes": [
+ "AD7EAE148287EF12D213A251015F86E6D4BD34B3C4A0A1ED9A17198373F908AD",
+ "E83BBB58949A8303DF07172B16FB8EFBA66B9191F3836EC27A4568ED5997BAC5"
+ ],
+ "LedgerEntryType": "DirectoryNode",
+ "Owner": "rpR95n1iFkTqpoy1e878f4Z1pVHVtWKMNQ",
+ "RootIndex": "193C591BF62482468422313F9D3274B5927CA80B4DD3707E42015DD609E39C94",
+ "index": "193C591BF62482468422313F9D3274B5927CA80B4DD3707E42015DD609E39C94"
+}
+```
+
+
+
+| Name | JSON Type | [Internal Type][] | Description |
+|-------------------|-----------|---------------|-------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0064`, mapped to the string `DirectoryNode`, indicates that this object is part of a Directory. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags enabled for this directory. Currently, the protocol defines no flags for DirectoryNode objects. |
+| `RootIndex` | Number | Hash256 | The ID of root object for this directory. |
+| `Indexes` | Array | Vector256 | The contents of this Directory: an array of IDs of other objects. |
+| `IndexNext` | Number | UInt64 | (Optional) If this Directory consists of multiple pages, this ID links to the next object in the chain, wrapping around at the end. |
+| `IndexPrevious` | Number | UInt64 | (Optional) If this Directory consists of multiple pages, this ID links to the previous object in the chain, wrapping around at the beginning. |
+| `Owner` | String | AccountID | (Owner Directories only) The address of the account that owns the objects in this directory. |
+| `ExchangeRate` | Number | UInt64 | (Offer Directories only) **DEPRECATED**. Do not use. |
+| `TakerPaysCurrency` | String | Hash160 | (Offer Directories only) The currency code of the TakerPays amount from the offers in this directory. |
+| `TakerPaysIssuer` | String | Hash160 | (Offer Directories only) The issuer of the TakerPays amount from the offers in this directory. |
+| `TakerGetsCurrency` | String | Hash160 | (Offer Directories only) The currency code of the TakerGets amount from the offers in this directory. |
+| `TakerGetsIssuer` | String | Hash160 | (Offer Directories only) The issuer of the TakerGets amount from the offers in this directory. |
+
+### Directory ID Formats
+
+There are three different formulas for creating the ID of a DirectoryNode, depending on which of the following the DirectoryNode represents:
+
+* The first page (also called the root) of an Owner Directory
+* The first page of an Offer Directory
+* Later pages of either type
+
+**The first page of an Owner Directory** has an ID that is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The Owner Directory space key (`0x004F`)
+* The AccountID from the `Owner` field.
+
+**The first page of an Offer Directory** has a special ID: the higher 192 bits define the order book, and the remaining 64 bits define the exchange rate of the offers in that directory. (The ID is big-endian, so the book is in the more significant bits, which come first, and the quality is in the less significant bits which come last.) This provides a way to iterate through an order book from best offers to worst. Specifically: the first 192 bits are the first 192 bits of the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The Book Directory space key (`0x0042`)
+* The 160-bit currency code from the `TakerPaysCurrency`
+* The 160-bit currency code from the `TakerGetsCurrency`
+* The AccountID from the `TakerPaysIssuer`
+* The AccountID from the `TakerGetsIssuer`
+
+The lower 64 bits of an Offer Directory's ID represent the TakerPays amount divided by TakerGets amount from the offer(s) in that directory as a 64-bit number in the XRP Ledger's internal amount format.
+
+**If the DirectoryNode is not the first page in the Directory** (regardless of whether it is an Owner Directory or an Offer Directory), then it has an ID that is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The DirectoryNode space key (`0x0064`)
+* The ID of the root DirectoryNode
+* The page number of this object. (Since 0 is the root DirectoryNode, this value is an integer 1 or higher.)
diff --git a/content/ledger-objects/escrow.md b/content/ledger-objects/escrow.md
new file mode 100644
index 00000000000..53a90010d62
--- /dev/null
+++ b/content/ledger-objects/escrow.md
@@ -0,0 +1,61 @@
+## Escrow
+[[Source]
](https://github.com/ripple/rippled/blob/c6b6d82a754fe449cc533e18659df483c10a5c98/src/ripple/protocol/impl/LedgerFormats.cpp#L90-L101 "Source")
+
+_(Requires the [Escrow Amendment](reference-amendments.html#escrow).)_
+
+The `Escrow` object type represents a held payment of XRP waiting to be executed or canceled. An [EscrowCreate transaction][] creates an `Escrow` object in the ledger. A successful [EscrowFinish][] or [EscrowCancel][] transaction deletes the object. If the ``Escrow`` object has a [_crypto-condition_](https://tools.ietf.org/html/draft-thomas-crypto-conditions-02), the payment can only succeed if an EscrowFinish transaction provides the corresponding _fulfillment_ that satisfies the condition. (The only supported crypto-condition type is [PREIMAGE-SHA-256](https://tools.ietf.org/html/draft-thomas-crypto-conditions-02#section-8.1).) If the `Escrow` object has a `FinishAfter` time, the held payment can only execute after that time.
+
+An `Escrow` object is associated with two addresses:
+
+- The owner, who provides the XRP when creating the `Escrow` object. If the held payment is canceled, the XRP returns to the owner.
+- The destination, where the XRP is paid when the held payment succeeds. The destination can be the same as the owner.
+
+Example `Escrow` object:
+
+```json
+{
+ "Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
+ "Amount": "10000",
+ "CancelAfter": 545440232,
+ "Condition": "A0258020A82A88B2DF843A54F58772E4A3861866ECDB4157645DD9AE528C1D3AEEDABAB6810120",
+ "Destination": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
+ "DestinationTag": 23480,
+ "FinishAfter": 545354132,
+ "Flags": 0,
+ "LedgerEntryType": "Escrow",
+ "OwnerNode": "0000000000000000",
+ "DestinationNode": "0000000000000000",
+ "PreviousTxnID": "C44F2EB84196B9AD820313DBEBA6316A15C9A2D35787579ED172B87A30131DA7",
+ "PreviousTxnLgrSeq": 28991004,
+ "SourceTag": 11747,
+ "index": "DC5F3851D8A1AB622F957761E5963BC5BD439D5C24AC6AD7AC4523F0640244AC"
+}
+```
+
+An `Escrow` object has the following fields:
+
+| Name | JSON Type | [Internal Type][] | Description |
+|-------------------|-----------|---------------|-------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0075`, mapped to the string `Escrow`, indicates that this object is an `Escrow` object. |
+| `Account` | String | AccountID | The address of the owner (sender) of this held payment. This is the account that provided the XRP, and gets it back if the held payment is canceled. |
+| `Destination` | String | AccountID | The destination address where the XRP is paid if the held payment is successful. |
+| `Amount` | String | Amount | The amount of XRP, in drops, to be delivered by the held payment. |
+| `Condition` | String | VariableLength | _(Optional)_ A [PREIMAGE-SHA-256 crypto-condition](https://tools.ietf.org/html/draft-thomas-crypto-conditions-02#section-8.1), as hexadecimal. If present, the [EscrowFinish transaction][] must contain a fulfillment that satisfies this condition. |
+| `CancelAfter` | Number | UInt32 | _(Optional)_ The held payment can be canceled if and only if this field is present _and_ the time it specifies has passed. Specifically, this is specified as [seconds since the Ripple Epoch](reference-rippled.html#specifying-time) and it "has passed" if it's earlier than the close time of the previous validated ledger. |
+| `FinishAfter` | Number | UInt32 | _(Optional)_ The time, in [seconds since the Ripple Epoch](reference-rippled.html#specifying-time), after which this held payment can be finished. Any [EscrowFinish transaction][] before this time fails. (Specifically, this is compared with the close time of the previous validated ledger.) |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags. No flags are defined for the Escrow type, so this value is always `0`. |
+| `SourceTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the source for this held payment, such as a hosted recipient at the owner's address. |
+| `DestinationTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the destination for this held payment, such as a hosted recipient at the destination address. |
+| `OwnerNode` | String | UInt64 | A hint indicating which page of the owner directory links to this object, in case the directory consists of multiple pages. **Note:** The object does not contain a direct link to the owner directory containing it, since that value can be derived from the `Account`. |
+| `DestinationNode` | String | UInt64 | _(Optional)_ A hint indicating which page of the destination's owner directory links to this object, in case the directory consists of multiple pages. Omitted on escrows created before enabling the [fix1523 amendment](reference-amendments.html#fix1523). |
+| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
+| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
+
+
+### Escrow ID Format
+
+The ID of an `Escrow` object is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The Escrow space key (`0x0075`)
+* The AccountID of the sender of the [EscrowCreate transaction][] that created the `Escrow` object
+* The Sequence number of the [EscrowCreate transaction][] that created the `Escrow` object
diff --git a/content/ledger-objects/feesettings.md b/content/ledger-objects/feesettings.md
new file mode 100644
index 00000000000..9e40be42fee
--- /dev/null
+++ b/content/ledger-objects/feesettings.md
@@ -0,0 +1,39 @@
+## FeeSettings
+[[Source]
](https://github.com/ripple/rippled/blob/master/src/ripple/protocol/impl/LedgerFormats.cpp#L115-L120 "Source")
+
+The `FeeSettings` object type contains the current base [transaction cost](concept-transaction-cost.html) and [reserve amounts](concept-reserves.html) as determined by [fee voting](concept-fee-voting.html). Each ledger version contains **at most one** `FeeSettings` object.
+
+Example `FeeSettings` object:
+
+```json
+{
+ "BaseFee": "000000000000000A",
+ "Flags": 0,
+ "LedgerEntryType": "FeeSettings",
+ "ReferenceFeeUnits": 10,
+ "ReserveBase": 20000000,
+ "ReserveIncrement": 5000000,
+ "index": "4BC50C9B0D8515D3EAAE1E74B29A95804346C491EE1A95BF25E4AAB854A6A651"
+}
+```
+
+The `FeeSettings` object has the following fields:
+
+| Name | JSON Type | [Internal Type][] | Description |
+|:--------------------|:----------|:------------------|:-----------------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0073`, mapped to the string `FeeSettings`, indicates that this object contains the ledger's fee settings. |
+| `BaseFee` | String | UInt64 | The [transaction cost](concept-transaction-cost.html) of the "reference transaction" in drops of XRP as hexadecimal. |
+| `ReferenceFeeUnits` | Number | UInt32 | The `BaseFee` translated into "fee units". |
+| `ReserveBase` | Number | UInt32 | The [base reserve](concept-reserves.html#base-reserve-and-owner-reserve) for an account in the XRP Ledger, as drops of XRP. |
+| `ReserveIncrement` | Number | UInt32 | The incremental [owner reserve](concept-reserves.html#base-reserve-and-owner-reserve) for owning objects, as drops of XRP. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags for this object. No flags are defined for this type. |
+
+**Warning:** The JSON format for this ledger object type is unusual. The `BaseFee`, `ReserveBase`, and `ReserveIncrement` indicate drops of XRP but ***not*** in the usual format for [specifying XRP](reference-rippled.html#specifying-currency-amounts).
+
+### FeeSettings ID Format
+
+The `FeeSettings` object ID is the hash of the `FeeSettings` space key (`0x0065`) only. This means that the ID of the `FeeSettings` object in a ledger is always:
+
+```
+4BC50C9B0D8515D3EAAE1E74B29A95804346C491EE1A95BF25E4AAB854A6A651
+```
diff --git a/content/ledger-objects/ledgerhashes.md b/content/ledger-objects/ledgerhashes.md
new file mode 100644
index 00000000000..d9059e3c20c
--- /dev/null
+++ b/content/ledger-objects/ledgerhashes.md
@@ -0,0 +1,71 @@
+## LedgerHashes
+[[Source]
](https://github.com/ripple/rippled/blob/master/src/ripple/protocol/impl/LedgerFormats.cpp#L104-L107 "Source")
+
+(Not to be confused with the ["ledger hash" string data type][Hash], which uniquely identifies a ledger version. This section describes the `LedgerHashes` ledger object type.)
+
+The `LedgerHashes` object type contains a history of prior ledgers that led up to this ledger version, in the form of their hashes. Objects of this ledger type are modified automatically in the process of closing a ledger. (This is the only time the ledger's "state" tree is modified without a transaction or pseudo-transaction.) The `LedgerHashes` objects exist to make it possible to look up a previous ledger's hash with only the current ledger version and at most one lookup of a previous ledger version.
+
+There are two kinds of `LedgerHashes` object. Both types have the same fields. Each ledger version contains:
+
+- Exactly one "recent history" `LedgerHashes` object
+- A number of "previous history" `LedgerHashes` objects based on the current ledger index (that is, the length of the ledger history). Specifically, the XRP Ledger adds a new "previous history" object every 65536 ledger versions.
+
+**Note:** As an exception, a new genesis ledger has no `LedgerHashes` objects at all, because it has no ledger history.
+
+Example `LedgerHashes` object (trimmed for length):
+
+```json
+{
+ "LedgerEntryType": "LedgerHashes",
+ "Flags": 0,
+ "FirstLedgerSequence": 2,
+ "LastLedgerSequence": 33872029,
+ "Hashes": [
+ "D638208ADBD04CBB10DE7B645D3AB4BA31489379411A3A347151702B6401AA78",
+ "254D690864E418DDD9BCAC93F41B1F53B1AE693FC5FE667CE40205C322D1BE3B",
+ "A2B31D28905E2DEF926362822BC412B12ABF6942B73B72A32D46ED2ABB7ACCFA",
+ "AB4014846DF818A4B43D6B1686D0DE0644FE711577C5AB6F0B2A21CCEE280140",
+ "3383784E82A8BA45F4DD5EF4EE90A1B2D3B4571317DBAC37B859836ADDE644C1",
+ ... (up to 256 ledger hashes) ...
+ ],
+ "index": "B4979A36CDC7F3D3D5C31A4EAE2AC7D7209DDA877588B9AFC66799692AB0D66B"
+}
+```
+
+A `LedgerHashes` object has the following fields:
+
+| Name | JSON Type | [Internal Type][] | Description |
+|-------------------|-----------|-------------------|-------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0068`, mapped to the string `LedgerHashes`, indicates that this object is a list of ledger hashes. |
+| `FirstLedgerSequence` | Number | UInt32 | **DEPRECATED** Do not use. (The "recent hashes" object of the production XRP Ledger has the value `2` in this field as a result of a previous `rippled` software. That value gets carried forward as the "recent hashes" object is updated. New "previous history" objects do not have this field, nor do "recent hashes" objects in [parallel networks](tutorial-rippled-setup.html#parallel-networks) started with more recent versions of `rippled`.) |
+| `LastLedgerSequence` | Number | UInt32 | The [ledger index](#ledger-index) of the last entry in this object's `Hashes` array. |
+| `Hashes` | Array of Strings | STI_VECTOR256 | An array of up to 256 ledger hashes. The contents depend on which sub-type of `LedgerHashes` object this is. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags for this object. No flags are defined for this type. |
+
+### Recent History LedgerHashes
+
+There is exactly one `LedgerHashes` object of the "recent history" sub-type in every ledger after the genesis ledger. This object contains the identifying hashes of the most recent 256 ledger versions (or fewer, if the ledger history has less than 256 ledgers total) in the `Hashes` array. Whenever a new ledger is closed, part of the process of closing it involves updating the "recent history" object with the hash of the previous ledger version this ledger version is derived from (also known as this ledger version's _parent ledger_). When there are more than 256 hashes, the oldest one is removed.
+
+Using the "recent history" `LedgerHashes` object of a given ledger, you can get the hash of any ledger index within the 256 ledger versions before the given ledger version.
+
+### Previous History LedgerHashes
+
+The "previous history" `LedgerHashes` entries collectively contain the hash of every 256th ledger version (also called "flag ledgers") in the full history of the ledger. When the child of a flag ledger closes, the flag ledger's hash is added to the `Hashes` array of the newest "previous history" `LedgerHashes` object. Every 65536 ledgers, `rippled` creates a new `LedgerHashes` object, so that each "previous history" object has the hashes of 256 flag ledgers.
+
+**Note:** The oldest "previous history" `LedgerHashes` object contains only 255 entries because the genesis ledger has ledger index 1, not 0.
+
+The "previous history" `LedgerHashes` objects act as a [skip list](https://en.wikipedia.org/wiki/Skip_list) so you can get the hash of any historical flag ledger from its index. From there, you can use that flag ledger's "recent history" object to get the hash of any other ledger.
+
+### LedgerHashes ID Formats
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/Indexes.cpp#L28-L43)
+
+There are two formats for `LedgerHashes` object IDs, depending on whether the object is a "recent history" sub-type or a "previous history" sub-type.
+
+The **"recent history"** `LedgerHashes` object has an ID that is the [SHA-512Half](#sha512half) of the `LedgerHashes` space key (`0x0073`). In other words, the "recent history" always has the ID `B4979A36CDC7F3D3D5C31A4EAE2AC7D7209DDA877588B9AFC66799692AB0D66B`.
+
+The **"previous history"** `LedgerHashes` objects have an ID that is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+- The `LedgerHashes` space key (`0x0073`)
+- The 32-bit [ledger index](#ledger-index) of a flag ledger in the object's `Hashes` array, divided by 65536.
+
+ **Tip:** Dividing by 65536 keeps the most significant 16 bits, which are the same for all the flag ledgers listed in a "previous history" object, and only those ledgers. You can use this fact to look up the `LedgerHashes` object that contains the hash of any flag ledger.
diff --git a/content/ledger-objects/offer.md b/content/ledger-objects/offer.md
new file mode 100644
index 00000000000..191a4302cc3
--- /dev/null
+++ b/content/ledger-objects/offer.md
@@ -0,0 +1,65 @@
+## Offer
+[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L57 "Source")
+
+The `Offer` object type describes an offer to exchange currencies, more traditionally known as an _order_, in the XRP Ledger's distributed exchange. An [OfferCreate transaction][] only creates an `Offer` object in the ledger when the offer cannot be fully executed immediately by consuming other offers already in the ledger.
+
+An offer can become unfunded through other activities in the network, while remaining in the ledger. However, `rippled` automatically prunes any unfunded offers it happens across in the course of transaction processing (and _only_ transaction processing, because the ledger state must only be changed by transactions). For more information, see [lifecycle of an offer](reference-transaction-format.html#lifecycle-of-an-offer).
+
+Example `Offer` object:
+
+```json
+{
+ "Account": "rBqb89MRQJnMPq8wTwEbtz4kvxrEDfcYvt",
+ "BookDirectory": "ACC27DE91DBA86FC509069EAF4BC511D73128B780F2E54BF5E07A369E2446000",
+ "BookNode": "0000000000000000",
+ "Flags": 131072,
+ "LedgerEntryType": "Offer",
+ "OwnerNode": "0000000000000000",
+ "PreviousTxnID": "F0AB71E777B2DA54B86231E19B82554EF1F8211F92ECA473121C655BFC5329BF",
+ "PreviousTxnLgrSeq": 14524914,
+ "Sequence": 866,
+ "TakerGets": {
+ "currency": "XAG",
+ "issuer": "r9Dr5xwkeLegBeXq6ujinjSBLQzQ1zQGjH",
+ "value": "37"
+ },
+ "TakerPays": "79550000000",
+ "index": "96F76F27D8A327FC48753167EC04A46AA0E382E6F57F32FD12274144D00F1797"
+}
+```
+
+An `Offer` object has the following fields:
+
+| Name | JSON Type | [Internal Type][] | Description |
+|-------------------|-----------|---------------|-------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x006F`, mapped to the string `Offer`, indicates that this object describes an order to trade currency. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags enabled for this offer. |
+| `Account` | String | AccountID | The address of the account that owns this offer. |
+| `Sequence` | Number | UInt32 | The `Sequence` value of the [OfferCreate][] transaction that created this `Offer` object. Used in combination with the `Account` to identify this Offer. |
+| `TakerPays` | String or Object | Amount | The remaining amount and type of currency requested by the offer creator. |
+| `TakerGets` | String or Object | Amount | The remaining amount and type of currency being provided by the offer creator. |
+| `BookDirectory` | String | UInt256 | The ID of the [Offer Directory](#directorynode) that links to this offer. |
+| `BookNode` | String | UInt64 | A hint indicating which page of the offer directory links to this object, in case the directory consists of multiple pages. |
+| `OwnerNode` | String | UInt64 | A hint indicating which page of the owner directory links to this object, in case the directory consists of multiple pages. **Note:** The offer does not contain a direct link to the owner directory containing it, since that value can be derived from the `Account`. |
+| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
+| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
+| `Expiration` | Number | UInt32 | (Optional) Indicates the time after which this offer is considered unfunded. See [Specifying Time](reference-rippled.html#specifying-time) for details. |
+
+### Offer Flags
+
+There are several options which can be either enabled or disabled when an [OfferCreate transaction][] creates an offer object. In the ledger, flags are represented as binary values that can be combined with bitwise-or operations. The bit values for the flags in the ledger are different than the values used to enable or disable those flags in a transaction. Ledger flags have names that begin with _lsf_.
+
+`Offer` objects can have the following flag values:
+
+| Flag Name | Hex Value | Decimal Value | Description | Corresponding [OfferCreate Flag](reference-transaction-format.html#offercreate-flags) |
+|-----------|-----------|---------------|-------------|------------------------|
+| lsfPassive | 0x00010000 | 65536 | The object was placed as a passive offer. This has no effect on the object in the ledger. | tfPassive |
+| lsfSell | 0x00020000 | 131072 | The object was placed as a sell offer. This has no effect on the object in the ledger (because tfSell only matters if you get a better rate than you asked for, which cannot happen after the object enters the ledger). | tfSell |
+
+### Offer ID Format
+
+The ID of an `Offer` object is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The Offer space key (`0x006F`)
+* The AccountID of the account placing the offer
+* The Sequence number of the [OfferCreate transaction][] that created the offer
diff --git a/content/ledger-objects/paychannel.md b/content/ledger-objects/paychannel.md
new file mode 100644
index 00000000000..63dc3b21617
--- /dev/null
+++ b/content/ledger-objects/paychannel.md
@@ -0,0 +1,97 @@
+## PayChannel
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/LedgerFormats.cpp#L134 "Source")
+
+_(Requires the [PayChan Amendment](reference-amendments.html#paychan).)_
+
+The `PayChannel` object type represents a payment channel. Payment channels enable small, rapid off-ledger payments of XRP that can be later reconciled with the consensus ledger. A payment channel holds a balance of XRP that can only be paid out to a specific destination address until the channel is closed. Any unspent XRP is returned to the channel's owner (the source address that created and funded it) when the channel closes.
+
+The [PaymentChannelCreate transaction][] type creates a `PayChannel` object. The [PaymentChannelFund][] and [PaymentChannelClaim transaction][] types modify existing `PayChannel` objects.
+
+When a payment channel expires, at first it remains on the ledger, because only new transactions can modify ledger contents. Transaction processing automatically closes a payment channel when any transaction accesses it after the expiration. To close an expired channel and return the unspent XRP to the owner, some address must send a new PaymentChannelClaim or PaymentChannelFund transaction accessing the channel.
+
+For an example of using payment channels, see the [Payment Channels Tutorial](tutorial-paychan.html).
+
+
+
+Example `PayChannel` object:
+
+```json
+{
+ "Account": "rBqb89MRQJnMPq8wTwEbtz4kvxrEDfcYvt",
+ "Destination": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
+ "Amount": "4325800",
+ "Balance": "2323423",
+ "PublicKey": "32D2471DB72B27E3310F355BB33E339BF26F8392D5A93D3BC0FC3B566612DA0F0A",
+ "SettleDelay": 3600,
+ "Expiration": 536027313,
+ "CancelAfter": 536891313,
+ "SourceTag": 0,
+ "DestinationTag": 1002341,
+ "Flags": 0,
+ "LedgerEntryType": "PayChannel",
+ "OwnerNode": "0000000000000000",
+ "PreviousTxnID": "F0AB71E777B2DA54B86231E19B82554EF1F8211F92ECA473121C655BFC5329BF",
+ "PreviousTxnLgrSeq": 14524914,
+ "index": "96F76F27D8A327FC48753167EC04A46AA0E382E6F57F32FD12274144D00F1797"
+}
+```
+
+A `PayChannel` object has the following fields:
+
+| Name | JSON Type | [Internal Type][] | Description |
+|:--------------------|:----------|:------------------|:-----------------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0078`, mapped to the string `PayChannel`, indicates that this object is a payment channel object. |
+| `Account` | String | AccountID | The source address that owns this payment channel. This comes from the sending address of the transaction that created the channel. |
+| `Destination` | String | AccountID | The destination address for this payment channel. While the payment channel is open, this address is the only one that can receive XRP from the channel. This comes from the `Destination` field of the transaction that created the channel. |
+| `Amount` | String | Amount | Total [XRP, in drops][], that has been allocated to this channel. This includes XRP that has been paid to the destination address. This is initially set by the transaction that created the channel and can be increased if the source address sends a PaymentChannelFund transaction. |
+| `Balance` | String | Amount | Total [XRP, in drops][], already paid out by the channel. The difference between this value and the `Amount` field is how much XRP can still be paid to the destination address with PaymentChannelClaim transactions. If the channel closes, the remaining difference is returned to the source address. |
+| `PublicKey` | String | PubKey | Public key, in hexadecimal, of the key pair that can be used to sign claims against this channel. This can be any valid secp256k1 or Ed25519 public key. This is set by the transaction that created the channel and must match the public key used in claims against the channel. The channel source address can also send XRP from this channel to the destination without signed claims. |
+| `SettleDelay` | Number | UInt32 | Number of seconds the source address must wait to close the channel if it still has any XRP in it. Smaller values mean that the destination address has less time to redeem any outstanding claims after the source address requests to close the channel. Can be any value that fits in a 32-bit unsigned integer (0 to 2^32-1). This is set by the transaction that creates the channel. |
+| `OwnerNode` | String | UInt64 | A hint indicating which page of the source address's owner directory links to this object, in case the directory consists of multiple pages. |
+| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
+| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags enabled for this payment channel. Currently, the protocol defines no flags for `PayChannel` objects. |
+| `Expiration` | Number | UInt32 | _(Optional)_ The mutable expiration time for this payment channel, in [seconds since the Ripple Epoch](reference-rippled.html#specifying-time). The channel is expired if this value is present and smaller than the previous ledger's [`close_time` field](#header-format). See [Setting Channel Expiration](#setting-channel-expiration) for more details. |
+| `CancelAfter` | Number | UInt32 | _(Optional)_ The immutable expiration time for this payment channel, in [seconds since the Ripple Epoch](reference-rippled.html#specifying-time). This channel is expired if this value is present and smaller than the previous ledger's [`close_time` field](#header-format). This is optionally set by the transaction that created the channel, and cannot be changed. |
+| `SourceTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the source for this payment channel, such as a hosted recipient at the owner's address. |
+| `DestinationTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the destination for this payment channel, such as a hosted recipient at the destination address. |
+
+[XRP, in drops]: reference-rippled.html#specifying-currency-amounts
+
+### Setting Channel Expiration
+
+The `Expiration` field of a payment channel is the mutable expiration time, in contrast to the immutable expiration time represented by the `CancelAfter` field. The expiration of a channel is always considered relative to the [`close_time` field](#header-format) of the previous ledger. The `Expiration` field is omitted when a `PayChannel` object is created. There are several ways the `Expiration` field of a `PayChannel` object can be updated, which can be summarized as follows: a channel's source address can set the `Expiration` of the channel freely as long as the channel always remains open at least `SettleDelay` seconds after the first attempt to close it.
+
+#### Source Address
+
+The source address can set the `Expiration` directly with the PaymentChannelFund transaction type. The new value must not be earlier than whichever of the following values is earliest:
+
+- The current `Expiration` value (if one is set)
+- The previous ledger's close time plus the `SettleDelay` of the channel
+
+In other words, the source address can always make the `Expiration` later if an expiration is already set. The source can make an `Expiration` value earlier or set an `Expiration` if one isn't currently set, as long as the new value is at least `SettleDelay` seconds in the future. If the source address attempts to set an invalid `Expiration` date, the transaction fails with the `temBAD_EXPIRATION` error code.
+
+The source address can also set the `Expiration` with the `tfClose` flag of the PaymentChannelClaim transaction type. If the flag is enabled, the ledger automatically sets the `Expiration` to whichever of the following values is earlier:
+
+- The current `Expiration` value (if one is set)
+- The previous ledger's close time plus the `SettleDelay` of the channel
+
+The source address can remove the `Expiration` with the `tfRenew` flag of the PaymentChannelClaim transaction type.
+
+#### Destination Address
+
+The destination address cannot set the `Expiration` field. However, the destination address can use the PaymentChannelClaim's `tfClose` flag to close a channel immediately.
+
+#### Other Addresses
+
+If any other address attempts to set an `Expiration` field, the transaction fails with the `tecNO_PERMISSION` error code. However, if the channel is already expired, the transaction causes the channel to close and results in `tesSUCCESS` instead.
+
+
+### PayChannel ID Format
+
+The ID of a `PayChannel` object is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The PayChannel space key (`0x0078`)
+* The AccountID of the source account
+* The AccountID of the destination account
+* The Sequence number of the transaction that created the channel
diff --git a/content/ledger-objects/ripplestate.md b/content/ledger-objects/ripplestate.md
new file mode 100644
index 00000000000..3e37b91e276
--- /dev/null
+++ b/content/ledger-objects/ripplestate.md
@@ -0,0 +1,102 @@
+## RippleState
+[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L70 "Source")
+
+The `RippleState` object type connects two accounts in a single currency. Conceptually, a `RippleState` object represents two _trust lines_ between the accounts, one from each side. Each account can change the settings for its side of the `RippleState` object, but the balance is a single shared value. A trust line that is entirely in its default state is considered the same as a trust line that does not exist, so `rippled` deletes `RippleState` objects when their properties are entirely default.
+
+Since no account is privileged in the XRP Ledger, a `RippleState` object sorts their account addresses numerically, to ensure a canonical form. Whichever address is numerically lower is deemed the "low account" and the other is the "high account".
+
+Example `RippleState` object:
+
+```json
+{
+ "Balance": {
+ "currency": "USD",
+ "issuer": "rrrrrrrrrrrrrrrrrrrrBZbvji",
+ "value": "-10"
+ },
+ "Flags": 393216,
+ "HighLimit": {
+ "currency": "USD",
+ "issuer": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
+ "value": "110"
+ },
+ "HighNode": "0000000000000000",
+ "LedgerEntryType": "RippleState",
+ "LowLimit": {
+ "currency": "USD",
+ "issuer": "rsA2LpzuawewSBQXkiju3YQTMzW13pAAdW",
+ "value": "0"
+ },
+ "LowNode": "0000000000000000",
+ "PreviousTxnID": "E3FE6EA3D48F0C2B639448020EA4F03D4F4F8FFDB243A852A0F59177921B4879",
+ "PreviousTxnLgrSeq": 14090896,
+ "index": "9CA88CDEDFF9252B3DE183CE35B038F57282BC9503CDFA1923EF9A95DF0D6F7B"
+}
+```
+
+A RippleState object has the following fields:
+
+| Name | JSON Type | Internal Type | Description |
+|-----------------|-----------|---------------|-------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0072`, mapped to the string `RippleState`, indicates that this object is a RippleState object. |
+| `Flags` | Number | UInt32 | A bit-map of boolean options enabled for this object. |
+| `Balance` | Object | Amount | The balance of the trust line, from the perspective of the low account. A negative balance indicates that the low account has issued currency to the high account. The issuer in this is always set to the neutral value [ACCOUNT_ONE](concept-accounts.html#special-addresses). |
+| `LowLimit` | Object | Amount | The limit that the low account has set on the trust line. The `issuer` is the address of the low account that set this limit. |
+| `HighLimit` | Object | Amount | The limit that the high account has set on the trust line. The `issuer` is the address of the high account that set this limit. |
+| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
+| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
+| `LowNode` | String | UInt64 | (Omitted in some historical ledgers) A hint indicating which page of the low account's owner directory links to this object, in case the directory consists of multiple pages. |
+| `HighNode` | String | UInt64 | (Omitted in some historical ledgers) A hint indicating which page of the high account's owner directory links to this object, in case the directory consists of multiple pages. |
+| `LowQualityIn` | Number | UInt32 | (Optional) The inbound quality set by the low account, as an integer in the implied ratio LowQualityIn:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
+| `LowQualityOut` | Number | UInt32 | (Optional) The outbound quality set by the low account, as an integer in the implied ratio LowQualityOut:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
+| `HighQualityIn` | Number | UInt32 | (Optional) The inbound quality set by the high account, as an integer in the implied ratio HighQualityIn:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
+| `HighQualityOut` | Number | UInt32 | (Optional) The outbound quality set by the high account, as an integer in the implied ratio HighQualityOut:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
+
+### RippleState Flags
+
+There are several options which can be either enabled or disabled for a trust line. These options can be changed with a [TrustSet transaction](reference-transaction-format.html#trustset). In the ledger, flags are represented as binary values that can be combined with bitwise-or operations. The bit values for the flags in the ledger are different than the values used to enable or disable those flags in a transaction. Ledger flags have names that begin with _lsf_.
+
+RippleState objects can have the following flag values:
+
+| Flag Name | Hex Value | Decimal Value | Description | Corresponding [TrustSet Flag](reference-transaction-format.html#trustset-flags) |
+|-----------|-----------|---------------|-------------|------------------------|
+| lsfLowReserve | 0x00010000 | 65536 | This RippleState object [contributes to the low account's owner reserve](#contributing-to-the-owner-reserve). | (None) |
+| lsfHighReserve | 0x00020000 |131072 | This RippleState object [contributes to the high account's owner reserve](#contributing-to-the-owner-reserve). | (None) |
+| lsfLowAuth | 0x00040000 | 262144 | The low account has authorized the high account to hold the low account's issuances. | tfSetAuth |
+| lsfHighAuth | 0x00080000 | 524288 | The high account has authorized the low account to hold the high account's issuances. | tfSetAuth |
+| lsfLowNoRipple | 0x00100000 | 1048576 | The low account [has disabled rippling](concept-noripple.html) from this trust line to other trust lines with the same account's NoRipple flag set. | tfSetNoRipple |
+| lsfHighNoRipple | 0x00200000 | 2097152 | The high account [has disabled rippling](concept-noripple.html) from this trust line to other trust lines with the same account's NoRipple flag set. | tfSetNoRipple |
+| lsfLowFreeze | 0x00400000 | 4194304 | The low account has frozen the trust line, preventing the high account from transferring the asset. | tfSetFreeze |
+| lsfHighFreeze | 0x00800000 | 8388608 | The high account has frozen the trust line, preventing the low account from transferring the asset. | tfSetFreeze |
+
+### Contributing to the Owner Reserve
+
+If an account modifies a trust line to put it in a non-default state, then that trust line counts towards the account's [owner reserve](concept-reserves.html#owner-reserves). In a RippleState object, the `lsfLowReserve` and `lsfHighReserve` flags indicate which account(s) are responsible for the owner reserve. The `rippled` server automatically sets these flags when it modifies a trust line.
+
+The values that count towards a trust line's non-default state are as follows:
+
+| High account responsible if... | Low account responsible if... |
+|-----------------------|----------------------|
+| `Balance` is negative (the high account holds currency) | `Balance` is positive (the low account holds currency) |
+| `HighLimit` is not `0` | `LowLimit` is not `0` |
+| `LowQualityIn` is not `0` and not `1000000000` | `HighQualityIn` is not `0` and not `1000000000` |
+| `LowQualityOut` is not `0` and not `1000000000` | `HighQualityOut` is not `0` and not `1000000000` |
+| **lsfHighNoRipple** flag is not in its default state | **lsfLowNoRipple** flag is not in its default state |
+| **lsfHighFreeze** flag is enabled | **lsfLowFreeze** flag is enabled |
+
+The **lsfLowAuth** and **lsfHighAuth** flags do not count against the default state, because they cannot be disabled.
+
+The default state of the two NoRipple flags depends on the state of the [lsfDefaultRipple flag](#accountroot-flags) in their corresponding AccountRoot objects. If DefaultRipple is disabled (the default), then the default state of the lsfNoRipple flag is _enabled_ for all of an account's trust lines. If an account enables DefaultRipple, then the lsfNoRipple flag is _disabled_ (rippling is enabled) for an account's trust lines by default.
+
+**Note:** Prior to the introduction of the DefaultRipple flags in `rippled` version 0.27.3 (March 10, 2015), the default state for all trust lines was with both NoRipple flags disabled (rippling enabled).
+
+Fortunately, `rippled` uses lazy evaluation to calculate the owner reserve. This means that even if an account changes the default state of all its trust lines by changing the DefaultRipple flag, that account's reserve stays the same initially. If an account modifies a trust line, `rippled` re-evaluates whether that individual trust line is in its default state and should contribute to the owner reserve.
+
+### RippleState ID Format
+
+The ID of a RippleState object is the [SHA-512Half](#sha512half) of the following values, concatenated in order:
+
+* The RippleState space key (`0x0072`)
+* The AccountID of the low account
+* The AccountID of the high account
+* The 160-bit currency code of the trust line(s)
diff --git a/content/ledger-objects/signerlist.md b/content/ledger-objects/signerlist.md
new file mode 100644
index 00000000000..880b9550173
--- /dev/null
+++ b/content/ledger-objects/signerlist.md
@@ -0,0 +1,77 @@
+## SignerList
+[[Source]
](https://github.com/ripple/rippled/blob/6d2e3da30696bd10e3bb11a5ff6d45d2c4dae90f/src/ripple/protocol/impl/LedgerFormats.cpp#L127 "Source")
+
+The `SignerList` object type represents a list of parties that, as a group, are authorized to sign a transaction in place of an individual account. You can create, replace, or remove a SignerList using the [SignerListSet transaction type](reference-transaction-format.html#signerlistset). This object type is introduced by the [MultiSign amendment](reference-amendments.html#multisign). [New in: rippled 0.31.0][]
+
+Example SignerList object:
+
+```json
+{
+ "Flags": 0,
+ "LedgerEntryType": "SignerList",
+ "OwnerNode": "0000000000000000",
+ "PreviousTxnID": "5904C0DC72C58A83AEFED2FFC5386356AA83FCA6A88C89D00646E51E687CDBE4",
+ "PreviousTxnLgrSeq": 16061435,
+ "SignerEntries": [
+ {
+ "SignerEntry": {
+ "Account": "rsA2LpzuawewSBQXkiju3YQTMzW13pAAdW",
+ "SignerWeight": 2
+ }
+ },
+ {
+ "SignerEntry": {
+ "Account": "raKEEVSGnKSD9Zyvxu4z6Pqpm4ABH8FS6n",
+ "SignerWeight": 1
+ }
+ },
+ {
+ "SignerEntry": {
+ "Account": "rUpy3eEg8rqjqfUoLeBnZkscbKbFsKXC3v",
+ "SignerWeight": 1
+ }
+ }
+ ],
+ "SignerListID": 0,
+ "SignerQuorum": 3,
+ "index": "A9C28A28B85CD533217F5C0A0C7767666B093FA58A0F2D80026FCC4CD932DDC7"
+}
+```
+
+A SignerList object has the following fields:
+
+| Name | JSON Type | Internal Type | Description |
+|-----------------|-----------|---------------|-------------|
+| `LedgerEntryType` | String | UInt16 | The value `0x0053`, mapped to the string `SignerList`, indicates that this object is a SignerList object. |
+| `Flags` | Number | UInt32 | A bit-map of boolean flags. No flags are defined for the SignerList type, so this value is always `0`. |
+| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
+| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
+| `OwnerNode` | String | UInt64 | A hint indicating which page of the owner directory links to this object, in case the directory consists of multiple pages. |
+| `SignerEntries` | Array | Array | An array of SignerEntry objects representing the parties who are part of this signer list. |
+| `SignerListID` | Number | UInt32 | An ID for this signer list. Currently always set to `0`. If a future [amendment](concept-amendments.html) allows multiple signer lists for an account, this may change. |
+| `SignerQuorum` | Number | UInt32 | A target number for signer weights. To produce a valid signature for the owner of this SignerList, the signers must provide valid signatures whose weights sum to this value or more. |
+
+The `SignerEntries` may be any combination of funded and unfunded addresses that use either secp256k1 or ed25519 keys.
+
+### SignerEntry Object
+
+Each member of the `SignerEntries` field is an object that describes that signer in the list. A SignerEntry has the following fields:
+
+| Name | JSON Type | Internal Type | Description |
+|-----------------|-----------|---------------|-------------|
+| `Account` | String | AccountID | An XRP Ledger address whose signature contributes to the multi-signature. It does not need to be a funded address in the ledger. |
+| `SignerWeight` | Number | UInt16 | The weight of a signature from this signer. A multi-signature is only valid if the sum weight of the signatures provided meets or exceeds the SignerList's `SignerQuorum` value. |
+
+When processing a multi-signed transaction, the server dereferences the `Account` values with respect to the ledger at the time of transaction execution. If the address _does not_ correspond to a funded [AccountRoot object](#accountroot), then only the master secret associated with that address can be used to produce a valid signature. If the account _does_ exist in the ledger, then it depends on the state of that account. If the account has a Regular Key configured, the Regular Key can be used. The account's master key can only be used if it is not disabled. A multi-signature cannot be used as part of another multi-signature.
+
+### SignerLists and Reserves
+
+A SignerList contributes to its owner's [reserve requirement](concept-reserves.html). The SignerList itself counts as two objects, and each member of the list counts as one. As a result, the total owner reserve associated with a SignerList is anywhere from 3 times to 10 times the reserve required by a single trust line ([RippleState](#ripplestate)) or [Offer](#offer) object in the ledger.
+
+### SignerList ID Format
+
+The ID of a SignerList object is the SHA-512Half of the following values, concatenated in order:
+
+* The RippleState space key (`0x0053`)
+* The AccountID of the owner of the SignerList
+* The SignerListID (currently always `0`)
diff --git a/content/reference-ledger-format.md b/content/reference-ledger-format.md
index e2c61e4ff5e..683e09c05f0 100644
--- a/content/reference-ledger-format.md
+++ b/content/reference-ledger-format.md
@@ -70,6 +70,7 @@ There are several different kinds of objects that can appear in the ledger's sta
* [**AccountRoot** - The settings, XRP balance, and other metadata for one account.](#accountroot)
* [**Amendments** - Singleton object with status of enabled and pending amendments.](#amendments)
+* [**Check** - A check that can be redeemed for money by its destination](#check)
* [**DirectoryNode** - Contains links to other objects.](#directorynode)
* [**Escrow** - Contains XRP held for a conditional payment.](#escrow)
* [**FeeSettings** - Singleton object with consensus-approved base transaction cost and reserve requirements.](#feesettings)
@@ -81,755 +82,27 @@ There are several different kinds of objects that can appear in the ledger's sta
Each ledger object consists of several fields. In the peer protocol that `rippled` servers use to communicate with each other, ledger objects are represented in their raw binary format. In other [`rippled` APIs](reference-rippled.html), ledger objects are represented as JSON objects.
-## AccountRoot
-[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L27 "Source")
-
-The `AccountRoot` object type describes a single _account_ object. Example `AccountRoot` object:
-
-```json
-{
- "Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
- "AccountTxnID": "0D5FB50FA65C9FE1538FD7E398FFFE9D1908DFA4576D8D7A020040686F93C77D",
- "Balance": "148446663",
- "Domain": "6D64756F31332E636F6D",
- "EmailHash": "98B4375E1D753E5B91627516F6D70977",
- "Flags": 8388608,
- "LedgerEntryType": "AccountRoot",
- "MessageKey": "0000000000000000000000070000000300",
- "OwnerCount": 3,
- "PreviousTxnID": "0D5FB50FA65C9FE1538FD7E398FFFE9D1908DFA4576D8D7A020040686F93C77D",
- "PreviousTxnLgrSeq": 14091160,
- "Sequence": 336,
- "TransferRate": 1004999999,
- "index": "13F1A95D7AAB7108D5CE7EEAF504B2894B8C674E6D68499076441C4837282BF8"
-}
-```
-
-The `AccountRoot` object has the following fields:
+{% include 'ledger-objects/accountroot.md' %}
-| Field | JSON Type | [Internal Type][] | Description |
-|-----------------|-----------|---------------|-------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0061`, mapped to the string `AccountRoot`, indicates that this object is an AccountRoot object. |
-| `Account` | String | AccountID | The identifying address of this account, such as rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn. |
-| [Flags](#accountroot-flags) | Number | UInt32 | A bit-map of boolean flags enabled for this account. |
-| `Sequence` | Number | UInt32 | The sequence number of the next valid transaction for this account. (Each account starts with Sequence = 1 and increases each time a transaction is made.) |
-| `Balance` | String | Amount | The account's current [XRP balance in drops][XRP, in drops], represented as a string. |
-| `OwnerCount` | Number | UInt32 | The number of objects this account owns in the ledger, which contributes to its owner reserve. |
-| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
-| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
-| `AccountTxnID` | String | Hash256 | (Optional) The identifying hash of the transaction most recently submitted by this account. |
-| `RegularKey` | String | AccountID | (Optional) The address of a keypair that can be used to sign transactions for this account instead of the master key. Use a [SetRegularKey transaction][] to change this value. |
-| `EmailHash` | String | Hash128 | (Optional) The md5 hash of an email address. Clients can use this to look up an avatar through services such as [Gravatar](https://en.gravatar.com/). |
-| `WalletLocator` | String | Hash256 | (Optional) **DEPRECATED**. Do not use. |
-| `WalletSize` | Number | UInt32 | (Optional) **DEPRECATED**. Do not use. |
-| `MessageKey` | String | VariableLength | (Optional) A public key that may be used to send encrypted messages to this account. In JSON, uses hexadecimal. No more than 33 bytes. |
-| `TickSize` | Number | UInt8 | (Optional) How many significant digits to use for exchange rates of Offers involving currencies issued by this address. Valid values are `3` to `15`, inclusive. _(Requires the [TickSize amendment](reference-amendments.html#ticksize).)_ |
-| `TransferRate` | Number | UInt32 | (Optional) A [transfer fee](https://ripple.com/knowledge_center/transfer-fees/) to charge other users for sending currency issued by this account to each other. |
-| `Domain` | String | VariableLength | (Optional) A domain associated with this account. In JSON, this is the hexadecimal for the ASCII representation of the domain. |
-
-### AccountRoot Flags
-
-There are several options which can be either enabled or disabled for an account. These options can be changed with an [AccountSet transaction][]. In the ledger, flags are represented as binary values that can be combined with bitwise-or operations. The bit values for the flags in the ledger are different than the values used to enable or disable those flags in a transaction. Ledger flags have names that begin with _lsf_.
-
-AccountRoot objects can have the following flag values:
-
-| Flag Name | Hex Value | Decimal Value | Description | Corresponding [AccountSet Flag](reference-transaction-format.html#accountset-flags) |
-|-----------|-----------|---------------|-------------|-------------------------------|
-| lsfPasswordSpent | 0x00010000 | 65536 | Indicates that the account has used its free SetRegularKey transaction. | (None) |
-| lsfRequireDestTag | 0x00020000 | 131072 | Requires incoming payments to specify a Destination Tag. | asfRequireDest |
-| lsfRequireAuth | 0x00040000 | 262144 | This account must individually approve other users for those users to hold this account's issuances. | asfRequireAuth |
-| lsfDisallowXRP | 0x00080000 | 524288 | Client applications should not send XRP to this account. Not enforced by `rippled`. | asfDisallowXRP |
-| lsfDisableMaster | 0x00100000 | 1048576 | Disallows use of the master key to sign transactions for this account. | asfDisableMaster |
-| lsfNoFreeze | 0x00200000 | 2097152 | This address cannot freeze trust lines connected to it. Once enabled, cannot be disabled. | asfNoFreeze |
-| lsfGlobalFreeze | 0x00400000 | 4194304 | All assets issued by this address are frozen. | asfGlobalFreeze |
-| lsfDefaultRipple | 0x00800000 | 8388608 | Enable [rippling](concept-noripple.html) on this addresses's trust lines by default. Required for issuing addresses; discouraged for others. | asfDefaultRipple |
-
-### AccountRoot ID Format
-
-The ID of an AccountRoot object is the [SHA-512Half](#sha512half) of the following values put together:
-
-* The Account space key (`0x0061`)
-* The AccountID of the account
-
-
-## Amendments
-[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/LedgerFormats.cpp#L110-L113 "Source")
-
-The `Amendments` object type contains a list of [Amendments](concept-amendments.html) that are currently active. Each ledger version contains **at most one** `Amendments` object.
-
-Example `Amendments` object:
-
-```json
-{
- "Majorities": [
- {
- "Majority": {
- "Amendment": "1562511F573A19AE9BD103B5D6B9E01B3B46805AEC5D3C4805C902B514399146",
- "CloseTime": 535589001
- }
- }
- ],
- "Amendments": [
- "42426C4D4F1009EE67080A9B7965B44656D7714D104A72F9B4369F97ABF044EE",
- "4C97EBA926031A7CF7D7B36FDE3ED66DDA5421192D63DE53FFB46E43B9DC8373",
- "6781F8368C4771B83E8B821D88F580202BCB4228075297B19E4FDC5233F1EFDC",
- "740352F2412A9909880C23A559FCECEDA3BE2126FED62FC7660D628A06927F11"
- ],
- "Flags": 0,
- "LedgerEntryType": "Amendments",
- "index": "7DB0788C020F02780A673DC74757F23823FA3014C1866E72CC4CD8B226CD6EF4"
-}
-```
-
-| Name | JSON Type | [Internal Type][] | Description |
-|-------------------|-----------|-------------------|-------------|
-| `Amendments` | Array | STI_VECTOR256 | _(Optional)_ Array of 256-bit [amendment IDs](concept-amendments.html#about-amendments) for all currently-enabled amendments. If omitted, there are no enabled amendments. |
-| `Majorities` | Array | STI_ARRAY | _(Optional)_ Array of objects describing the status of amendments that have majority support but are not yet enabled. If omitted, there are no pending amendments with majority support. |
-| `Flags` | Number | UInt32 | Not used. |
-| `LedgerEntryType` | String | UInt16 | The value `0x0066`, mapped to the string `Amendments`, indicates that this is object describes status of amendments to the XRP Ledger. |
-
-Each member of the `Majorities` field, if it is present, is an object with a one field, `Majority`, whose contents are a nested object with the following fields:
-
-| Name | JSON Type | [Internal Type][] | Description |
-|-------------------|-----------|-------------------|-------------|
-| `Amendment` | String | Hash256 | The Amendment ID of the pending amendment. |
-| `CloseTime` | Number | UInt32 | The [`close_time` field](#header-format) of the ledger version where this amendment most recently gained a majority. |
-
-In the [amendment process](concept-amendments.html#amendment-process), a consensus of validators adds a new amendment to the `Majorities` field using an [EnableAmendment][] pseudo-transaction with the `tfGotMajority` flag when 80% or more of validators support it. If support for a pending amendment goes below 80%, an [EnableAmendment][] pseudo-transaction with the `tfLostMajority` flag removes the amendment from the `Majorities` array. If an amendment remains in the `Majorities` field for at least 2 weeks, an [EnableAmendment][] pseudo-transaction with no flags removes it from `Majorities` and permanently adds it to the `Amendments` field.
-
-**Note:** Technically, all transactions in a ledger are processed based on which amendments are enabled in the ledger version immediately before it. While applying transactions to a ledger version where an amendment becomes enabled, the rules don't change mid-ledger. After the ledger is closed, the next ledger uses the new rules as defined by any new amendments that applied.
-
-### Amendments ID Format
-
-The `Amendments` object ID is the hash of the `Amendments` space key (`0x0066`) only. This means that the ID of the `Amendments` object in a ledger is always:
-
-```
-7DB0788C020F02780A673DC74757F23823FA3014C1866E72CC4CD8B226CD6EF4
-```
-
-(Don't mix up the ID of the `Amendments` ledger object type with the Amendment ID of an individual amendment.)
-
-
-## DirectoryNode
-[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L44 "Source")
-
-The `DirectoryNode` object type provides a list of links to other objects in the ledger's state tree. A single conceptual _Directory_ takes the form of a doubly linked list, with one or more DirectoryNode objects each containing up to 32 [IDs](#tree-format) of other objects. The first object is called the root of the directory, and all objects other than the root object can be added or deleted as necessary.
-
-There are two kinds of Directories:
-
-* **Owner directories** list other objects owned by an account, such as `RippleState` or `Offer` objects.
-* **Offer directories** list the offers available in the distributed exchange. A single Offer directory contains all the offers that have the same exchange rate for the same issuances.
-
-Example Directories:
-
-
-
-*Offer Directory*
-
-```json
-{
- "ExchangeRate": "4F069BA8FF484000",
- "Flags": 0,
- "Indexes": [
- "AD7EAE148287EF12D213A251015F86E6D4BD34B3C4A0A1ED9A17198373F908AD"
- ],
- "LedgerEntryType": "DirectoryNode",
- "RootIndex": "1BBEF97EDE88D40CEE2ADE6FEF121166AFE80D99EBADB01A4F069BA8FF484000",
- "TakerGetsCurrency": "0000000000000000000000000000000000000000",
- "TakerGetsIssuer": "0000000000000000000000000000000000000000",
- "TakerPaysCurrency": "0000000000000000000000004A50590000000000",
- "TakerPaysIssuer": "5BBC0F22F61D9224A110650CFE21CC0C4BE13098",
- "index": "1BBEF97EDE88D40CEE2ADE6FEF121166AFE80D99EBADB01A4F069BA8FF484000"
-}
-```
-
-*Owner Directory*
-
-```json
-{
- "Flags": 0,
- "Indexes": [
- "AD7EAE148287EF12D213A251015F86E6D4BD34B3C4A0A1ED9A17198373F908AD",
- "E83BBB58949A8303DF07172B16FB8EFBA66B9191F3836EC27A4568ED5997BAC5"
- ],
- "LedgerEntryType": "DirectoryNode",
- "Owner": "rpR95n1iFkTqpoy1e878f4Z1pVHVtWKMNQ",
- "RootIndex": "193C591BF62482468422313F9D3274B5927CA80B4DD3707E42015DD609E39C94",
- "index": "193C591BF62482468422313F9D3274B5927CA80B4DD3707E42015DD609E39C94"
-}
-```
-
-
-
-| Name | JSON Type | [Internal Type][] | Description |
-|-------------------|-----------|---------------|-------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0064`, mapped to the string `DirectoryNode`, indicates that this object is part of a Directory. |
-| `Flags` | Number | UInt32 | A bit-map of boolean flags enabled for this directory. Currently, the protocol defines no flags for DirectoryNode objects. |
-| `RootIndex` | Number | Hash256 | The ID of root object for this directory. |
-| `Indexes` | Array | Vector256 | The contents of this Directory: an array of IDs of other objects. |
-| `IndexNext` | Number | UInt64 | (Optional) If this Directory consists of multiple pages, this ID links to the next object in the chain, wrapping around at the end. |
-| `IndexPrevious` | Number | UInt64 | (Optional) If this Directory consists of multiple pages, this ID links to the previous object in the chain, wrapping around at the beginning. |
-| `Owner` | String | AccountID | (Owner Directories only) The address of the account that owns the objects in this directory. |
-| `ExchangeRate` | Number | UInt64 | (Offer Directories only) **DEPRECATED**. Do not use. |
-| `TakerPaysCurrency` | String | Hash160 | (Offer Directories only) The currency code of the TakerPays amount from the offers in this directory. |
-| `TakerPaysIssuer` | String | Hash160 | (Offer Directories only) The issuer of the TakerPays amount from the offers in this directory. |
-| `TakerGetsCurrency` | String | Hash160 | (Offer Directories only) The currency code of the TakerGets amount from the offers in this directory. |
-| `TakerGetsIssuer` | String | Hash160 | (Offer Directories only) The issuer of the TakerGets amount from the offers in this directory. |
-
-### Directory ID Formats
-
-There are three different formulas for creating the ID of a DirectoryNode, depending on which of the following the DirectoryNode represents:
-
-* The first page (also called the root) of an Owner Directory
-* The first page of an Offer Directory
-* Later pages of either type
-
-**The first page of an Owner Directory** has an ID that is the [SHA-512Half](#sha512half) of the following values put together:
-
-* The Owner Directory space key (`0x004F`)
-* The AccountID from the `Owner` field.
-
-**The first page of an Offer Directory** has a special ID: the higher 192 bits define the order book, and the remaining 64 bits define the exchange rate of the offers in that directory. (The ID is big-endian, so the book is in the more significant bits, which come first, and the quality is in the less significant bits which come last.) This provides a way to iterate through an order book from best offers to worst. Specifically: the first 192 bits are the first 192 bits of the [SHA-512Half](#sha512half) of the following values put together:
-
-* The Book Directory space key (`0x0042`)
-* The 160-bit currency code from the `TakerPaysCurrency`
-* The 160-bit currency code from the `TakerGetsCurrency`
-* The AccountID from the `TakerPaysIssuer`
-* The AccountID from the `TakerGetsIssuer`
-
-The lower 64 bits of an Offer Directory's ID represent the TakerPays amount divided by TakerGets amount from the offer(s) in that directory as a 64-bit number in the XRP Ledger's internal amount format.
-
-**If the DirectoryNode is not the first page in the Directory** (regardless of whether it is an Owner Directory or an Offer Directory), then it has an ID that is the [SHA-512Half](#sha512half) of the following values put together:
-
-* The DirectoryNode space key (`0x0064`)
-* The ID of the root DirectoryNode
-* The page number of this object. (Since 0 is the root DirectoryNode, this value is an integer 1 or higher.)
-
-
-## Escrow
-[[Source]
](https://github.com/ripple/rippled/blob/c6b6d82a754fe449cc533e18659df483c10a5c98/src/ripple/protocol/impl/LedgerFormats.cpp#L90-L101 "Source")
-
-_(Requires the [Escrow Amendment](reference-amendments.html#escrow).)_
-
-The `Escrow` object type represents a held payment of XRP waiting to be executed or canceled. An [EscrowCreate transaction][] creates an `Escrow` object in the ledger. A successful [EscrowFinish][] or [EscrowCancel][] transaction deletes the object. If the ``Escrow`` object has a [_crypto-condition_](https://tools.ietf.org/html/draft-thomas-crypto-conditions-02), the payment can only succeed if an EscrowFinish transaction provides the corresponding _fulfillment_ that satisfies the condition. (The only supported crypto-condition type is [PREIMAGE-SHA-256](https://tools.ietf.org/html/draft-thomas-crypto-conditions-02#section-8.1).) If the `Escrow` object has a `FinishAfter` time, the held payment can only execute after that time.
-
-An `Escrow` object is associated with two addresses:
-
-- The owner, who provides the XRP when creating the `Escrow` object. If the held payment is canceled, the XRP returns to the owner.
-- The destination, where the XRP is paid when the held payment succeeds. The destination can be the same as the owner.
-
-Example `Escrow` object:
-
-```json
-{
- "Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
- "Amount": "10000",
- "CancelAfter": 545440232,
- "Condition": "A0258020A82A88B2DF843A54F58772E4A3861866ECDB4157645DD9AE528C1D3AEEDABAB6810120",
- "Destination": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
- "DestinationTag": 23480,
- "FinishAfter": 545354132,
- "Flags": 0,
- "LedgerEntryType": "Escrow",
- "OwnerNode": "0000000000000000",
- "DestinationNode": "0000000000000000",
- "PreviousTxnID": "C44F2EB84196B9AD820313DBEBA6316A15C9A2D35787579ED172B87A30131DA7",
- "PreviousTxnLgrSeq": 28991004,
- "SourceTag": 11747,
- "index": "DC5F3851D8A1AB622F957761E5963BC5BD439D5C24AC6AD7AC4523F0640244AC"
-}
-```
-
-An `Escrow` object has the following fields:
-
-| Name | JSON Type | [Internal Type][] | Description |
-|-------------------|-----------|---------------|-------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0075`, mapped to the string `Escrow`, indicates that this object is an `Escrow` object. |
-| `Account` | String | AccountID | The address of the owner (sender) of this held payment. This is the account that provided the XRP, and gets it back if the held payment is canceled. |
-| `Destination` | String | AccountID | The destination address where the XRP is paid if the held payment is successful. |
-| `Amount` | String | Amount | The amount of XRP, in drops, to be delivered by the held payment. |
-| `Condition` | String | VariableLength | _(Optional)_ A [PREIMAGE-SHA-256 crypto-condition](https://tools.ietf.org/html/draft-thomas-crypto-conditions-02#section-8.1), as hexadecimal. If present, the [EscrowFinish transaction][] must contain a fulfillment that satisfies this condition. |
-| `CancelAfter` | Number | UInt32 | _(Optional)_ The held payment can be canceled if and only if this field is present _and_ the time it specifies has passed. Specifically, this is specified as [seconds since the Ripple Epoch](reference-rippled.html#specifying-time) and it "has passed" if it's earlier than the close time of the previous validated ledger. |
-| `FinishAfter` | Number | UInt32 | _(Optional)_ The time, in [seconds since the Ripple Epoch](reference-rippled.html#specifying-time), after which this held payment can be finished. Any [EscrowFinish transaction][] before this time fails. (Specifically, this is compared with the close time of the previous validated ledger.) |
-| `SourceTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the source for this held payment, such as a hosted recipient at the owner's address. |
-| `DestinationTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the destination for this held payment, such as a hosted recipient at the destination address. |
-| `OwnerNode` | String | UInt64 | A hint indicating which page of the owner directory links to this object, in case the directory consists of multiple pages. **Note:** The object does not contain a direct link to the owner directory containing it, since that value can be derived from the `Account`. |
-| `DestinationNode` | String | UInt64 | _(Optional)_ A hint indicating which page of the destination's owner directory links to this object, in case the directory consists of multiple pages. Omitted on escrows created before enabling the [fix1523 amendment](reference-amendments.html#fix1523). |
-| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
-| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
-
-
-### Escrow ID Format
-
-The ID of an `Escrow` object is the [SHA-512Half](#sha512half) of the following values put together:
-
-* The Escrow space key (`0x0075`)
-* The AccountID of the sender of the [EscrowCreate transaction][] that created the `Escrow` object
-* The Sequence number of the [EscrowCreate transaction][] that created the `Escrow` object
-
-## FeeSettings
-[[Source]
](https://github.com/ripple/rippled/blob/master/src/ripple/protocol/impl/LedgerFormats.cpp#L115-L120 "Source")
-
-The `FeeSettings` object type contains the current base [transaction cost](concept-transaction-cost.html) and [reserve amounts](concept-reserves.html) as determined by [fee voting](concept-fee-voting.html). Each ledger version contains **at most one** `FeeSettings` object.
-
-Example `FeeSettings` object:
-
-```json
-{
- "BaseFee": "000000000000000A",
- "Flags": 0,
- "LedgerEntryType": "FeeSettings",
- "ReferenceFeeUnits": 10,
- "ReserveBase": 20000000,
- "ReserveIncrement": 5000000,
- "index": "4BC50C9B0D8515D3EAAE1E74B29A95804346C491EE1A95BF25E4AAB854A6A651"
-}
-```
-
-The `FeeSettings` object has the following fields:
-
-| Name | JSON Type | [Internal Type][] | Description |
-|:--------------------|:----------|:------------------|:-----------------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0073`, mapped to the string `FeeSettings`, indicates that this object contains the ledger's fee settings. |
-| `BaseFee` | String | UInt64 | The [transaction cost](concept-transaction-cost.html) of the "reference transaction" in drops of XRP as hexadecimal. |
-| `ReferenceFeeUnits` | Number | UInt32 | The `BaseFee` translated into "fee units". |
-| `ReserveBase` | Number | UInt32 | The [base reserve](concept-reserves.html#base-reserve-and-owner-reserve) for an account in the XRP Ledger, as drops of XRP. |
-| `ReserveIncrement` | Number | UInt32 | The incremental [owner reserve](concept-reserves.html#base-reserve-and-owner-reserve) for owning objects, as drops of XRP. |
-| `Flags` | Number | UInt32 | A bit-map of boolean flags for this object. No flags are defined for this type. |
-
-**Warning:** The JSON format for this ledger object type is unusual. The `BaseFee`, `ReserveBase`, and `ReserveIncrement` indicate drops of XRP but ***not*** in the usual format for [specifying XRP](reference-rippled.html#specifying-currency-amounts).
-
-### FeeSettings ID Format
-
-The `FeeSettings` object ID is the hash of the `FeeSettings` space key (`0x0065`) only. This means that the ID of the `FeeSettings` object in a ledger is always:
-
-```
-4BC50C9B0D8515D3EAAE1E74B29A95804346C491EE1A95BF25E4AAB854A6A651
-```
-
-
-## LedgerHashes
-[[Source]
](https://github.com/ripple/rippled/blob/master/src/ripple/protocol/impl/LedgerFormats.cpp#L104-L107 "Source")
-
-(Not to be confused with the ["ledger hash" string data type][Hash], which uniquely identifies a ledger version. This section describes the `LedgerHashes` ledger object type.)
-
-The `LedgerHashes` object type contains a history of prior ledgers that led up to this ledger version, in the form of their hashes. Objects of this ledger type are modified automatically in the process of closing a ledger. (This is the only time the ledger's "state" tree is modified without a transaction or pseudo-transaction.) The `LedgerHashes` objects exist to make it possible to look up a previous ledger's hash with only the current ledger version and at most one lookup of a previous ledger version.
-
-There are two kinds of `LedgerHashes` object. Both types have the same fields. Each ledger version contains:
-
-- Exactly one "recent history" `LedgerHashes` object
-- A number of "previous history" `LedgerHashes` objects based on the current ledger index (that is, the length of the ledger history). Specifically, the XRP Ledger adds a new "previous history" object every 65536 ledger versions.
-
-**Note:** As an exception, a new genesis ledger has no `LedgerHashes` objects at all, because it has no ledger history.
-
-Example `LedgerHashes` object (trimmed for length):
-
-```json
-{
- "LedgerEntryType": "LedgerHashes",
- "Flags": 0,
- "FirstLedgerSequence": 2,
- "LastLedgerSequence": 33872029,
- "Hashes": [
- "D638208ADBD04CBB10DE7B645D3AB4BA31489379411A3A347151702B6401AA78",
- "254D690864E418DDD9BCAC93F41B1F53B1AE693FC5FE667CE40205C322D1BE3B",
- "A2B31D28905E2DEF926362822BC412B12ABF6942B73B72A32D46ED2ABB7ACCFA",
- "AB4014846DF818A4B43D6B1686D0DE0644FE711577C5AB6F0B2A21CCEE280140",
- "3383784E82A8BA45F4DD5EF4EE90A1B2D3B4571317DBAC37B859836ADDE644C1",
- ... (up to 256 ledger hashes) ...
- ],
- "index": "B4979A36CDC7F3D3D5C31A4EAE2AC7D7209DDA877588B9AFC66799692AB0D66B"
-}
-```
-
-A `LedgerHashes` object has the following fields:
-
-| Name | JSON Type | [Internal Type][] | Description |
-|-------------------|-----------|-------------------|-------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0068`, mapped to the string `LedgerHashes`, indicates that this object is a list of ledger hashes. |
-| `FirstLedgerSequence` | Number | UInt32 | **DEPRECATED** Do not use. (The "recent hashes" object of the production XRP Ledger has the value `2` in this field as a result of a previous `rippled` software. That value gets carried forward as the "recent hashes" object is updated. New "previous history" objects do not have this field, nor do "recent hashes" objects in [parallel networks](tutorial-rippled-setup.html#parallel-networks) started with more recent versions of `rippled`.) |
-| `LastLedgerSequence` | Number | UInt32 | The [ledger index](#ledger-index) of the last entry in this object's `Hashes` array. |
-| `Hashes` | Array of Strings | STI_VECTOR256 | An array of up to 256 ledger hashes. The contents depend on which sub-type of `LedgerHashes` object this is. |
-| `Flags` | Number | UInt32 | A bit-map of boolean flags for this object. No flags are defined for this type. |
-
-### Recent History LedgerHashes
-
-There is exactly one `LedgerHashes` object of the "recent history" sub-type in every ledger after the genesis ledger. This object contains the identifying hashes of the most recent 256 ledger versions (or fewer, if the ledger history has less than 256 ledgers total) in the `Hashes` array. Whenever a new ledger is closed, part of the process of closing it involves updating the "recent history" object with the hash of the previous ledger version this ledger version is derived from (also known as this ledger version's _parent ledger_). When there are more than 256 hashes, the oldest one is removed.
-
-Using the "recent history" `LedgerHashes` object of a given ledger, you can get the hash of any ledger index within the 256 ledger versions before the given ledger version.
-
-### Previous History LedgerHashes
-
-The "previous history" `LedgerHashes` entries collectively contain the hash of every 256th ledger version (also called "flag ledgers") in the full history of the ledger. When the child of a flag ledger closes, the flag ledger's hash is added to the `Hashes` array of the newest "previous history" `LedgerHashes` object. Every 65536 ledgers, `rippled` creates a new `LedgerHashes` object, so that each "previous history" object has the hashes of 256 flag ledgers.
-
-**Note:** The oldest "previous history" `LedgerHashes` object contains only 255 entries because the genesis ledger has ledger index 1, not 0.
-
-The "previous history" `LedgerHashes` objects act as a [skip list](https://en.wikipedia.org/wiki/Skip_list) so you can get the hash of any historical flag ledger from its index. From there, you can use that flag ledger's "recent history" object to get the hash of any other ledger.
-
-### LedgerHashes ID Formats
-[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/Indexes.cpp#L28-L43)
-
-There are two formats for `LedgerHashes` object IDs, depending on whether the object is a "recent history" sub-type or a "previous history" sub-type.
-
-The **"recent history"** `LedgerHashes` object has an ID that is the [SHA-512Half](#sha512half) of the `LedgerHashes` space key (`0x0073`). In other words, the "recent history" always has the ID `B4979A36CDC7F3D3D5C31A4EAE2AC7D7209DDA877588B9AFC66799692AB0D66B`.
-
-The **"previous history"** `LedgerHashes` objects have an ID that is the [SHA-512Half](#sha512half) of the following values put together:
-
-- The `LedgerHashes` space key (`0x0073`)
-- The 32-bit [ledger index](#ledger-index) of a flag ledger in the object's `Hashes` array, divided by 65536.
-
- **Tip:** Dividing by 65536 keeps the most significant 16 bits, which are the same for all the flag ledgers listed in a "previous history" object, and only those ledgers. You can use this fact to look up the `LedgerHashes` object that contains the hash of any flag ledger.
-
-
-## Offer
-[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L57 "Source")
-
-The `Offer` object type describes an offer to exchange currencies, more traditionally known as an _order_, in the XRP Ledger's distributed exchange. An [OfferCreate transaction][] only creates an `Offer` object in the ledger when the offer cannot be fully executed immediately by consuming other offers already in the ledger.
-
-An offer can become unfunded through other activities in the network, while remaining in the ledger. However, `rippled` automatically prunes any unfunded offers it happens across in the course of transaction processing (and _only_ transaction processing, because the ledger state must only be changed by transactions). For more information, see [lifecycle of an offer](reference-transaction-format.html#lifecycle-of-an-offer).
-
-Example `Offer` object:
-
-```json
-{
- "Account": "rBqb89MRQJnMPq8wTwEbtz4kvxrEDfcYvt",
- "BookDirectory": "ACC27DE91DBA86FC509069EAF4BC511D73128B780F2E54BF5E07A369E2446000",
- "BookNode": "0000000000000000",
- "Flags": 131072,
- "LedgerEntryType": "Offer",
- "OwnerNode": "0000000000000000",
- "PreviousTxnID": "F0AB71E777B2DA54B86231E19B82554EF1F8211F92ECA473121C655BFC5329BF",
- "PreviousTxnLgrSeq": 14524914,
- "Sequence": 866,
- "TakerGets": {
- "currency": "XAG",
- "issuer": "r9Dr5xwkeLegBeXq6ujinjSBLQzQ1zQGjH",
- "value": "37"
- },
- "TakerPays": "79550000000",
- "index": "96F76F27D8A327FC48753167EC04A46AA0E382E6F57F32FD12274144D00F1797"
-}
-```
-
-An `Offer` object has the following fields:
-
-| Name | JSON Type | [Internal Type][] | Description |
-|-------------------|-----------|---------------|-------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x006F`, mapped to the string `Offer`, indicates that this object describes an order to trade currency. |
-| `Flags` | Number | UInt32 | A bit-map of boolean flags enabled for this offer. |
-| `Account` | String | AccountID | The address of the account that owns this offer. |
-| `Sequence` | Number | UInt32 | The `Sequence` value of the [OfferCreate][] transaction that created this `Offer` object. Used in combination with the `Account` to identify this Offer. |
-| `TakerPays` | String or Object | Amount | The remaining amount and type of currency requested by the offer creator. |
-| `TakerGets` | String or Object | Amount | The remaining amount and type of currency being provided by the offer creator. |
-| `BookDirectory` | String | UInt256 | The ID of the [Offer Directory](#directorynode) that links to this offer. |
-| `BookNode` | String | UInt64 | A hint indicating which page of the offer directory links to this object, in case the directory consists of multiple pages. |
-| `OwnerNode` | String | UInt64 | A hint indicating which page of the owner directory links to this object, in case the directory consists of multiple pages. **Note:** The offer does not contain a direct link to the owner directory containing it, since that value can be derived from the `Account`. |
-| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
-| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
-| `Expiration` | Number | UInt32 | (Optional) Indicates the time after which this offer is considered unfunded. See [Specifying Time](reference-rippled.html#specifying-time) for details. |
-
-### Offer Flags
-
-There are several options which can be either enabled or disabled when an [OfferCreate transaction][] creates an offer object. In the ledger, flags are represented as binary values that can be combined with bitwise-or operations. The bit values for the flags in the ledger are different than the values used to enable or disable those flags in a transaction. Ledger flags have names that begin with _lsf_.
-
-`Offer` objects can have the following flag values:
-
-| Flag Name | Hex Value | Decimal Value | Description | Corresponding [OfferCreate Flag](reference-transaction-format.html#offercreate-flags) |
-|-----------|-----------|---------------|-------------|------------------------|
-| lsfPassive | 0x00010000 | 65536 | The object was placed as a passive offer. This has no effect on the object in the ledger. | tfPassive |
-| lsfSell | 0x00020000 | 131072 | The object was placed as a sell offer. This has no effect on the object in the ledger (because tfSell only matters if you get a better rate than you asked for, which cannot happen after the object enters the ledger). | tfSell |
-
-### Offer ID Format
-
-The ID of an `Offer` object is the [SHA-512Half](#sha512half) of the following values put together:
-
-* The Offer space key (`0x006F`)
-* The AccountID of the account placing the offer
-* The Sequence number of the [OfferCreate transaction][] that created the offer
-
-
-
-## PayChannel
-[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/protocol/impl/LedgerFormats.cpp#L134 "Source")
-
-_(Requires the [PayChan Amendment](reference-amendments.html#paychan).)_
-
-The `PayChannel` object type represents a payment channel. Payment channels enable small, rapid off-ledger payments of XRP that can be later reconciled with the consensus ledger. A payment channel holds a balance of XRP that can only be paid out to a specific destination address until the channel is closed. Any unspent XRP is returned to the channel's owner (the source address that created and funded it) when the channel closes.
-
-The [PaymentChannelCreate transaction][] type creates a `PayChannel` object. The [PaymentChannelFund][] and [PaymentChannelClaim transaction][] types modify existing `PayChannel` objects.
-
-When a payment channel expires, at first it remains on the ledger, because only new transactions can modify ledger contents. Transaction processing automatically closes a payment channel when any transaction accesses it after the expiration. To close an expired channel and return the unspent XRP to the owner, some address must send a new PaymentChannelClaim or PaymentChannelFund transaction accessing the channel.
-
-For an example of using payment channels, see the [Payment Channels Tutorial](tutorial-paychan.html).
-
-
-
-Example `PayChannel` object:
-
-```json
-{
- "Account": "rBqb89MRQJnMPq8wTwEbtz4kvxrEDfcYvt",
- "Destination": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
- "Amount": "4325800",
- "Balance": "2323423",
- "PublicKey": "32D2471DB72B27E3310F355BB33E339BF26F8392D5A93D3BC0FC3B566612DA0F0A",
- "SettleDelay": 3600,
- "Expiration": 536027313,
- "CancelAfter": 536891313,
- "SourceTag": 0,
- "DestinationTag": 1002341,
- "Flags": 0,
- "LedgerEntryType": "PayChannel",
- "OwnerNode": "0000000000000000",
- "PreviousTxnID": "F0AB71E777B2DA54B86231E19B82554EF1F8211F92ECA473121C655BFC5329BF",
- "PreviousTxnLgrSeq": 14524914,
- "index": "96F76F27D8A327FC48753167EC04A46AA0E382E6F57F32FD12274144D00F1797"
-}
-```
-
-A `PayChannel` object has the following fields:
-
-| Name | JSON Type | [Internal Type][] | Description |
-|:--------------------|:----------|:------------------|:-----------------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0078`, mapped to the string `PayChannel`, indicates that this object is a payment channel object. |
-| `Account` | String | AccountID | The source address that owns this payment channel. This comes from the sending address of the transaction that created the channel. |
-| `Destination` | String | AccountID | The destination address for this payment channel. While the payment channel is open, this address is the only one that can receive XRP from the channel. This comes from the `Destination` field of the transaction that created the channel. |
-| `Amount` | String | Amount | Total [XRP, in drops][], that has been allocated to this channel. This includes XRP that has been paid to the destination address. This is initially set by the transaction that created the channel and can be increased if the source address sends a PaymentChannelFund transaction. |
-| `Balance` | String | Amount | Total [XRP, in drops][], already paid out by the channel. The difference between this value and the `Amount` field is how much XRP can still be paid to the destination address with PaymentChannelClaim transactions. If the channel closes, the remaining difference is returned to the source address. |
-| `PublicKey` | String | PubKey | Public key, in hexadecimal, of the key pair that can be used to sign claims against this channel. This can be any valid secp256k1 or Ed25519 public key. This is set by the transaction that created the channel and must match the public key used in claims against the channel. The channel source address can also send XRP from this channel to the destination without signed claims. |
-| `SettleDelay` | Number | UInt32 | Number of seconds the source address must wait to close the channel if it still has any XRP in it. Smaller values mean that the destination address has less time to redeem any outstanding claims after the source address requests to close the channel. Can be any value that fits in a 32-bit unsigned integer (0 to 2^32-1). This is set by the transaction that creates the channel. |
-| `OwnerNode` | String | UInt64 | A hint indicating which page of the source address's owner directory links to this object, in case the directory consists of multiple pages. |
-| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
-| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
-| `Flags` | Number | UInt32 | A bit-map of boolean flags enabled for this payment channel. Currently, the protocol defines no flags for `PayChannel` objects. |
-| `Expiration` | Number | UInt32 | _(Optional)_ The mutable expiration time for this payment channel, in [seconds since the Ripple Epoch](reference-rippled.html#specifying-time). The channel is expired if this value is present and smaller than the previous ledger's [`close_time` field](#header-format). See [Setting Channel Expiration](#setting-channel-expiration) for more details. |
-| `CancelAfter` | Number | UInt32 | _(Optional)_ The immutable expiration time for this payment channel, in [seconds since the Ripple Epoch](reference-rippled.html#specifying-time). This channel is expired if this value is present and smaller than the previous ledger's [`close_time` field](#header-format). This is optionally set by the transaction that created the channel, and cannot be changed. |
-| `SourceTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the source for this payment channel, such as a hosted recipient at the owner's address. |
-| `DestinationTag` | Number | UInt32 | _(Optional)_ An arbitrary tag to further specify the destination for this payment channel, such as a hosted recipient at the destination address. |
-
-[XRP, in drops]: reference-rippled.html#specifying-currency-amounts
-
-### Setting Channel Expiration
-
-The `Expiration` field of a payment channel is the mutable expiration time, in contrast to the immutable expiration time represented by the `CancelAfter` field. The expiration of a channel is always considered relative to the [`close_time` field](#header-format) of the previous ledger. The `Expiration` field is omitted when a `PayChannel` object is created. There are several ways the `Expiration` field of a `PayChannel` object can be updated, which can be summarized as follows: a channel's source address can set the `Expiration` of the channel freely as long as the channel always remains open at least `SettleDelay` seconds after the first attempt to close it.
-
-#### Source Address
-
-The source address can set the `Expiration` directly with the PaymentChannelFund transaction type. The new value must not be earlier than whichever of the following values is earliest:
-
-- The current `Expiration` value (if one is set)
-- The previous ledger's close time plus the `SettleDelay` of the channel
-
-In other words, the source address can always make the `Expiration` later if an expiration is already set. The source can make an `Expiration` value earlier or set an `Expiration` if one isn't currently set, as long as the new value is at least `SettleDelay` seconds in the future. If the source address attempts to set an invalid `Expiration` date, the transaction fails with the `temBAD_EXPIRATION` error code.
-
-The source address can also set the `Expiration` with the `tfClose` flag of the PaymentChannelClaim transaction type. If the flag is enabled, the ledger automatically sets the `Expiration` to whichever of the following values is earlier:
-
-- The current `Expiration` value (if one is set)
-- The previous ledger's close time plus the `SettleDelay` of the channel
-
-The source address can remove the `Expiration` with the `tfRenew` flag of the PaymentChannelClaim transaction type.
-
-#### Destination Address
-
-The destination address cannot set the `Expiration` field. However, the destination address can use the PaymentChannelClaim's `tfClose` flag to close a channel immediately.
-
-#### Other Addresses
-
-If any other address attempts to set an `Expiration` field, the transaction fails with the `tecNO_PERMISSION` error code. However, if the channel is already expired, the transaction causes the channel to close and results in `tesSUCCESS` instead.
-
-
-### PayChannel ID Format
-
-The ID of a `PayChannel` object is the [SHA-512Half](#sha512half) of the following values put together:
-
-* The PayChannel space key (`0x0078`)
-* The AccountID of the source account
-* The AccountID of the destination account
-* The Sequence number of the transaction that created the channel
-
-
-
-## RippleState
-[[Source]
](https://github.com/ripple/rippled/blob/5d2d88209f1732a0f8d592012094e345cbe3e675/src/ripple/protocol/impl/LedgerFormats.cpp#L70 "Source")
-
-The `RippleState` object type connects two accounts in a single currency. Conceptually, a `RippleState` object represents two _trust lines_ between the accounts, one from each side. Each account can change the settings for its side of the `RippleState` object, but the balance is a single shared value. A trust line that is entirely in its default state is considered the same as trust line that does not exist, so `rippled` deletes `RippleState` objects when their properties are entirely default.
-
-Since no account is privileged in the XRP Ledger, a `RippleState` object sorts their account addresses numerically, to ensure a canonical form. Whichever address is numerically lower is deemed the "low account" and the other is the "high account".
-
-Example `RippleState` object:
-
-```json
-{
- "Balance": {
- "currency": "USD",
- "issuer": "rrrrrrrrrrrrrrrrrrrrBZbvji",
- "value": "-10"
- },
- "Flags": 393216,
- "HighLimit": {
- "currency": "USD",
- "issuer": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
- "value": "110"
- },
- "HighNode": "0000000000000000",
- "LedgerEntryType": "RippleState",
- "LowLimit": {
- "currency": "USD",
- "issuer": "rsA2LpzuawewSBQXkiju3YQTMzW13pAAdW",
- "value": "0"
- },
- "LowNode": "0000000000000000",
- "PreviousTxnID": "E3FE6EA3D48F0C2B639448020EA4F03D4F4F8FFDB243A852A0F59177921B4879",
- "PreviousTxnLgrSeq": 14090896,
- "index": "9CA88CDEDFF9252B3DE183CE35B038F57282BC9503CDFA1923EF9A95DF0D6F7B"
-}
-```
-
-A RippleState object has the following fields:
-
-| Name | JSON Type | Internal Type | Description |
-|-----------------|-----------|---------------|-------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0072`, mapped to the string `RippleState`, indicates that this object is a RippleState object. |
-| `Flags` | Number | UInt32 | A bit-map of boolean options enabled for this object. |
-| `Balance` | Object | Amount | The balance of the trust line, from the perspective of the low account. A negative balance indicates that the low account has issued currency to the high account. The issuer in this is always set to the neutral value [ACCOUNT_ONE](concept-accounts.html#special-addresses). |
-| `LowLimit` | Object | Amount | The limit that the low account has set on the trust line. The `issuer` is the address of the low account that set this limit. |
-| `HighLimit` | Object | Amount | The limit that the high account has set on the trust line. The `issuer` is the address of the high account that set this limit. |
-| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
-| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
-| `LowNode` | String | UInt64 | (Omitted in some historical ledgers) A hint indicating which page of the low account's owner directory links to this object, in case the directory consists of multiple pages. |
-| `HighNode` | String | UInt64 | (Omitted in some historical ledgers) A hint indicating which page of the high account's owner directory links to this object, in case the directory consists of multiple pages. |
-| `LowQualityIn` | Number | UInt32 | (Optional) The inbound quality set by the low account, as an integer in the implied ratio LowQualityIn:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
-| `LowQualityOut` | Number | UInt32 | (Optional) The outbound quality set by the low account, as an integer in the implied ratio LowQualityOut:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
-| `HighQualityIn` | Number | UInt32 | (Optional) The inbound quality set by the high account, as an integer in the implied ratio HighQualityIn:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
-| `HighQualityOut` | Number | UInt32 | (Optional) The outbound quality set by the high account, as an integer in the implied ratio HighQualityOut:1,000,000,000. The value 0 is equivalent to 1 billion, or face value. |
-
-### RippleState Flags
-
-There are several options which can be either enabled or disabled for a trust line. These options can be changed with a [TrustSet transaction](reference-transaction-format.html#trustset). In the ledger, flags are represented as binary values that can be combined with bitwise-or operations. The bit values for the flags in the ledger are different than the values used to enable or disable those flags in a transaction. Ledger flags have names that begin with _lsf_.
-
-RippleState objects can have the following flag values:
-
-| Flag Name | Hex Value | Decimal Value | Description | Corresponding [TrustSet Flag](reference-transaction-format.html#trustset-flags) |
-|-----------|-----------|---------------|-------------|------------------------|
-| lsfLowReserve | 0x00010000 | 65536 | This RippleState object [contributes to the low account's owner reserve](#contributing-to-the-owner-reserve). | (None) |
-| lsfHighReserve | 0x00020000 |131072 | This RippleState object [contributes to the high account's owner reserve](#contributing-to-the-owner-reserve). | (None) |
-| lsfLowAuth | 0x00040000 | 262144 | The low account has authorized the high account to hold the low account's issuances. | tfSetAuth |
-| lsfHighAuth | 0x00080000 | 524288 | The high account has authorized the low account to hold the high account's issuances. | tfSetAuth |
-| lsfLowNoRipple | 0x00100000 | 1048576 | The low account [has disabled rippling](concept-noripple.html) from this trust line to other trust lines with the same account's NoRipple flag set. | tfSetNoRipple |
-| lsfHighNoRipple | 0x00200000 | 2097152 | The high account [has disabled rippling](concept-noripple.html) from this trust line to other trust lines with the same account's NoRipple flag set. | tfSetNoRipple |
-| lsfLowFreeze | 0x00400000 | 4194304 | The low account has frozen the trust line, preventing the high account from transferring the asset. | tfSetFreeze |
-| lsfHighFreeze | 0x00800000 | 8388608 | The high account has frozen the trust line, preventing the low account from transferring the asset. | tfSetFreeze |
-
-### Contributing to the Owner Reserve
-
-If an account modifies a trust line to put it in a non-default state, then that trust line counts towards the account's [owner reserve](concept-reserves.html#owner-reserves). In a RippleState object, the `lsfLowReserve` and `lsfHighReserve` flags indicate which account(s) are responsible for the owner reserve. The `rippled` server automatically sets these flags when it modifies a trust line.
-
-The values that count towards a trust line's non-default state are as follows:
-
-| High account responsible if... | Low account responsible if... |
-|-----------------------|----------------------|
-| `Balance` is negative (the high account holds currency) | `Balance` is positive (the low account holds currency) |
-| `HighLimit` is not `0` | `LowLimit` is not `0` |
-| `LowQualityIn` is not `0` and not `1000000000` | `HighQualityIn` is not `0` and not `1000000000` |
-| `LowQualityOut` is not `0` and not `1000000000` | `HighQualityOut` is not `0` and not `1000000000` |
-| **lsfHighNoRipple** flag is not in its default state | **lsfLowNoRipple** flag is not in its default state |
-| **lsfHighFreeze** flag is enabled | **lsfLowFreeze** flag is enabled |
-
-The **lsfLowAuth** and **lsfHighAuth** flags do not count against the default state, because they cannot be disabled.
-
-The default state of the two NoRipple flags depends on the state of the [lsfDefaultRipple flag](#accountroot-flags) in their corresponding AccountRoot objects. If DefaultRipple is disabled (the default), then the default state of the lsfNoRipple flag is _enabled_ for all of an account's trust lines. If an account enables DefaultRipple, then the lsfNoRipple flag is _disabled_ (rippling is enabled) for an account's trust lines by default.
-
-**Note:** Prior to the introduction of the DefaultRipple flags in `rippled` version 0.27.3 (March 10, 2015), the default state for all trust lines was with both NoRipple flags disabled (rippling enabled).
-
-Fortunately, `rippled` uses lazy evaluation to calculate the owner reserve. This means that even if an account changes the default state of all its trust lines by changing the DefaultRipple flag, that account's reserve stays the same initially. If an account modifies a trust line, `rippled` re-evaluates whether that individual trust line is in its default state and should contribute the owner reserve.
-
-### RippleState ID Format
-
-The ID of a RippleState object is the [SHA-512Half](#sha512half) of the following values put together:
-
-* The RippleState space key (`0x0072`)
-* The AccountID of the low account
-* The AccountID of the high account
-* The 160-bit currency code of the trust line(s)
-
-
-## SignerList
-[[Source]
](https://github.com/ripple/rippled/blob/6d2e3da30696bd10e3bb11a5ff6d45d2c4dae90f/src/ripple/protocol/impl/LedgerFormats.cpp#L127 "Source")
-
-The `SignerList` object type represents a list of parties that, as a group, are authorized to sign a transaction in place of an individual account. You can create, replace, or remove a SignerList using the [SignerListSet transaction type](reference-transaction-format.html#signerlistset). This object type is introduced by the [MultiSign amendment](reference-amendments.html#multisign). [New in: rippled 0.31.0][]
-
-Example SignerList object:
-
-```json
-{
- "Flags": 0,
- "LedgerEntryType": "SignerList",
- "OwnerNode": "0000000000000000",
- "PreviousTxnID": "5904C0DC72C58A83AEFED2FFC5386356AA83FCA6A88C89D00646E51E687CDBE4",
- "PreviousTxnLgrSeq": 16061435,
- "SignerEntries": [
- {
- "SignerEntry": {
- "Account": "rsA2LpzuawewSBQXkiju3YQTMzW13pAAdW",
- "SignerWeight": 2
- }
- },
- {
- "SignerEntry": {
- "Account": "raKEEVSGnKSD9Zyvxu4z6Pqpm4ABH8FS6n",
- "SignerWeight": 1
- }
- },
- {
- "SignerEntry": {
- "Account": "rUpy3eEg8rqjqfUoLeBnZkscbKbFsKXC3v",
- "SignerWeight": 1
- }
- }
- ],
- "SignerListID": 0,
- "SignerQuorum": 3,
- "index": "A9C28A28B85CD533217F5C0A0C7767666B093FA58A0F2D80026FCC4CD932DDC7"
-}
-```
-
-A SignerList object has the following fields:
-
-| Name | JSON Type | Internal Type | Description |
-|-----------------|-----------|---------------|-------------|
-| `LedgerEntryType` | String | UInt16 | The value `0x0053`, mapped to the string `SignerList`, indicates that this object is a SignerList object. |
-| `OwnerNode` | String | UInt64 | A hint indicating which page of the owner directory links to this object, in case the directory consists of multiple pages. |
-| `SignerQuorum` | Number | UInt32 | A target number for signer weights. To produce a valid signature for the owner of this SignerList, the signers must provide valid signatures whose weights sum to this value or more. |
-| `SignerEntries` | Array | Array | An array of SignerEntry objects representing the parties who are part of this signer list. |
-| `SignerListID` | Number | UInt32 | An ID for this signer list. Currently always set to `0`. If a future [amendment](concept-amendments.html) allows multiple signer lists for an account, this may change. |
-| `PreviousTxnID` | String | Hash256 | The identifying hash of the transaction that most recently modified this object. |
-| `PreviousTxnLgrSeq` | Number | UInt32 | The [index of the ledger](#ledger-index) that contains the transaction that most recently modified this object. |
-
-The `SignerEntries` may be any combination of funded and unfunded addresses that use either secp256k1 or ed25519 keys.
+{% include 'ledger-objects/amendments.md' %}
-### SignerEntry Object
+{% include 'ledger-objects/check.md' %}
-Each member of the `SignerEntries` field is an object that describes that signer in the list. A SignerEntry has the following fields:
-
-| Name | JSON Type | Internal Type | Description |
-|-----------------|-----------|---------------|-------------|
-| `Account` | String | AccountID | An XRP Ledger address whose signature contributes to the multi-signature. It does not need to be a funded address in the ledger. |
-| `SignerWeight` | Number | UInt16 | The weight of a signature from this signer. A multi-signature is only valid if the sum weight of the signatures provided meets or exceeds the SignerList's `SignerQuorum` value. |
+{% include 'ledger-objects/directorynode.md' %}
-When processing a multi-signed transaction, the server dereferences the `Account` values with respect to the ledger at the time of transaction execution. If the address _does not_ correspond to a funded [AccountRoot object](#accountroot), then only the master secret associated with that address can be used to produce a valid signature. If the account _does_ exist in the ledger, then it depends on the state of that account. If the account has a Regular Key configured, the Regular Key can be used. The account's master key can only be used if it is not disabled. A multi-signature cannot be used as part of another multi-signature.
+{% include 'ledger-objects/escrow.md' %}
-### SignerLists and Reserves
+{% include 'ledger-objects/feesettings.md' %}
-A SignerList contributes to its owner's [reserve requirement](concept-reserves.html). The SignerList itself counts as two objects, and each member of the list counts as one. As a result, the total owner reserve associated with a SignerList is anywhere from 3 times to 10 times the reserve required by a single trust line ([RippleState](#ripplestate)) or [Offer](#offer) object in the ledger.
+{% include 'ledger-objects/ledgerhashes.md' %}
-### SignerList ID Format
+{% include 'ledger-objects/offer.md' %}
-The ID of a SignerList object is the SHA-512Half of the following values put together:
+{% include 'ledger-objects/paychannel.md' %}
-* The RippleState space key (`0x0053`)
-* The AccountID of the owner of the SignerList
-* The SignerListID (currently always `0`)
+{% include 'ledger-objects/ripplestate.md' %}
+{% include 'ledger-objects/signerlist.md' %}
{% include 'snippets/rippled_versions.md' %}
{% include 'snippets/tx-type-links.md' %}
diff --git a/content/reference-transaction-format.md b/content/reference-transaction-format.md
index fc305fff0ff..9ef7a7c633e 100644
--- a/content/reference-transaction-format.md
+++ b/content/reference-transaction-format.md
@@ -362,6 +362,9 @@ All transactions have certain fields in common:
Each transaction type has additional fields relevant to the type of action it causes:
* [AccountSet - Set options on an account](#accountset)
+* [CheckCancel - Cancel a check](#checkcancel)
+* [CheckCash - Redeem a check](#checkcash)
+* [CheckCreate - Create a check](#checkcreate)
* [EscrowCancel - Reclaim escrowed XRP](#escrowcancel)
* [EscrowCreate - Create an escrowed XRP payment](#escrowcreate)
* [EscrowFinish - Deliver escrowed XRP to recipient](#escrowfinish)
@@ -375,11 +378,6 @@ Each transaction type has additional fields relevant to the type of action it ca
* [SignerListSet - Set multi-signing settings](#signerlistset)
* [TrustSet - Add or modify a trust line](#trustset)
-
_Pseudo-Transactions_ that are not created and submitted in the usual way, but may be added to open ledgers according to ledger rules. They still must be approved by consensus to be included in a validated ledger. Pseudo-transactions have their own unique transaction types:
@@ -389,6 +387,12 @@ _Pseudo-Transactions_ that are not created and submitted in the usual way, but m
{% include 'transactions/accountset.md' %}
+{% include 'transactions/checkcancel.md' %}
+
+{% include 'transactions/checkcash.md' %}
+
+{% include 'transactions/checkcreate.md' %}
+
{% include 'transactions/escrowcancel.md' %}
{% include 'transactions/escrowcreate.md' %}
@@ -557,6 +561,8 @@ See also: [Partial Payments](concept-partial-payments.html)
These codes indicate an error in the local server processing the transaction; it is possible that another server with a different configuration or load level could process the transaction successfully. They have numerical values in the range -399 to -300. The exact code for any given error is subject to change, so don't rely on it.
+**Caution:** Transactions with `tel` codes are not applied to ledgers and cannot cause any changes to the XRP Ledger state. However, a transaction that provisionally failed may still succeed or fail with a different code after being reapplied. For more information, see [Finality of Results](#finality-of-results) and [Reliable Transaction Submission](tutorial-reliable-transaction-submission.html).
+
| Code | Explanation |
|:----------------------|:-----------------------------------------------------|
| `telBAD_DOMAIN` | The transaction specified a domain value (for example, the `Domain` field of an [AccountSet transaction][]) that cannot be used, probably because it is too long to store in the ledger. |
@@ -577,6 +583,8 @@ These codes indicate an error in the local server processing the transaction; it
These codes indicate that the transaction was malformed, and cannot succeed according to the XRP Ledger protocol. They have numerical values in the range -299 to -200. The exact code for any given error is subject to change, so don't rely on it.
+**Tip:** Transactions with `tem` codes are not applied to ledgers, and cannot cause any changes to XRP Ledger state. A `tem` result is final unless the rules for a valid transaction change. (For example, using functionality from an [Amendment](concept-amendments.html) before that amendment is enabled results in `temDISABLED`; such a transaction could succeed later if it becomes valid when the amendment is enabled.)
+
| Code | Explanation |
|:-----------------------------|:----------------------------------------------|
| `temBAD_AMOUNT` | An amount specified by the transaction (for example the destination `Amount` or `SendMax` values of a [Payment](#payment)) was invalid, possibly because it was a negative number. |
@@ -618,6 +626,8 @@ These codes indicate that the transaction was malformed, and cannot succeed acco
These codes indicate that the transaction failed and was not included in a ledger, but the transaction could have succeeded in some theoretical ledger. Typically this means that the transaction can no longer succeed in any future ledger. They have numerical values in the range -199 to -100. The exact code for any given error is subject to change, so don't rely on it.
+**Caution:** Transactions with `tef` codes are not applied to ledgers and cannot cause any changes to the XRP Ledger state. However, a transaction that provisionally failed may still succeed or fail with a different code after being reapplied. For more information, see [Finality of Results](#finality-of-results) and [Reliable Transaction Submission](tutorial-reliable-transaction-submission.html).
+
| Code | Explanation |
|:-----------------------|:----------------------------------------------------|
| `tefALREADY` | The same exact transaction has already been applied. |
@@ -643,6 +653,8 @@ These codes indicate that the transaction failed and was not included in a ledge
These codes indicate that the transaction failed, but it could apply successfully in the future, usually if some other hypothetical transaction applies first. They have numerical values in the range -99 to -1. The exact code for any given error is subject to change, so don't rely on it.
+**Caution:** Transactions with `ter` codes are not applied to ledgers and cannot cause any changes to the XRP Ledger state. However, a transaction that provisionally failed may still succeed or fail with a different code after being reapplied. For more information, see [Finality of Results](#finality-of-results) and [Reliable Transaction Submission](tutorial-reliable-transaction-submission.html).
+
| Code | Explanation |
|:-----------------|:----------------------------------------------------------|
| `terFUNDS_SPENT` | **DEPRECATED.** |
@@ -659,7 +671,7 @@ These codes indicate that the transaction failed, but it could apply successfull
### tes Success
-The code `tesSUCCESS` is the only code that indicates a transaction succeeded. This does not always mean it did what it was supposed to do. (For example, an [OfferCancel][] can "succeed" even if there is no offer for it to cancel.) Success uses the numerical value 0.
+The code `tesSUCCESS` is the only code that indicates a transaction succeeded. This does not always mean it accomplished what you expected it to do. (For example, an [OfferCancel][] can "succeed" even if there is no offer for it to cancel.) The `tesSUCCESS` result uses the numerical value 0.
| Code | Explanation |
|:-----------|:----------------------------------------------------------------|
@@ -667,14 +679,19 @@ The code `tesSUCCESS` is the only code that indicates a transaction succeeded. T
### tec Codes
-These codes indicate that the transaction failed, but it was applied to a ledger to apply the [transaction cost](concept-transaction-cost.html). They have numerical values in the range 100 to 199. The exact codes sometimes appear in ledger data, so they do not change, but we recommend not relying on the numeric value regardless.
+These codes indicate that the transaction failed, but it was applied to a ledger to apply the [transaction cost](concept-transaction-cost.html). They have numerical values in the range 100 to 199. Ripple recommends using the text code, not the numeric value.
+
+For the most part, transactions with `tec` codes take no action other than to destroy the XRP paid as a [transaction cost](concept-transaction-cost.html), but there are some exceptions. As an exception, a transaction that results in `tecOVERSIZE` still cleans up some [unfunded offers](#lifecycle-of-an-offer). Always look at the [transaction metadata](#understanding-transaction-metadata) to see precisely what a transaction did.
+
+**Caution:** A transaction that provisionally failed with a `tec` code may still succeed or fail with a different code after being reapplied. The result is final when it appears in a validated ledger version. For more information, see [Finality of Results](#finality-of-results) and [Reliable Transaction Submission](tutorial-reliable-transaction-submission.html).
| Code | Value | Explanation |
|:---------------------------|:------|:----------------------------------------|
| `tecCLAIM` | 100 | Unspecified failure, with transaction cost destroyed. |
| `tecCRYPTOCONDITION_ERROR` | 146 | This [EscrowCreate][] or [EscrowFinish][] transaction contained a malformed or mismatched crypto-condition. |
-| `tecDIR_FULL` | 121 | The address sending the transaction cannot own any more objects in the ledger. |
+| `tecDIR_FULL` | 121 | The transaction tried to add an object (such as a trust line, Check, Escrow, or Payment Channel) to an account's owner directory, but that account cannot own any more objects in the ledger. |
| `tecDST_TAG_NEEDED` | 143 | The [Payment](#payment) transaction omitted a destination tag, but the destination account has the `lsfRequireDestTag` flag enabled. [New in: rippled 0.28.0][] |
+| `tecEXPIRED` | 148 | The transaction tried to create an object (such as an Offer or a Check) whose provided Expiration time has already passed. |
| `tecFAILED_PROCESSING` | 105 | An unspecified error occurred when processing the transaction. |
| `tecFROZEN` | 137 | The [OfferCreate transaction][] failed because one or both of the assets involved are subject to a [global freeze](concept-freeze.html). |
| `tecINSUF_RESERVE_LINE` | 122 | The transaction failed because the sending account does not have enough XRP to create a new trust line. (See: [Reserves](concept-reserves.html)) This error occurs when the counterparty already has a trust line in a non-default state to the sending account for the same currency. (See `tecNO_LINE_INSUF_RESERVE` for the other case.) |
diff --git a/content/rippled-api-methods/account_objects.md b/content/rippled-api-methods/account_objects.md
index 67ef4e48db8..5c3fc8328ce 100644
--- a/content/rippled-api-methods/account_objects.md
+++ b/content/rippled-api-methods/account_objects.md
@@ -12,6 +12,7 @@ The types of objects that may appear in the `account_objects` response for an ac
- The account's [SignerList](reference-ledger-format.html#signerlist), if the account has [multi-signing](reference-transaction-format.html#multi-signing) enabled.
- [Escrow objects](reference-ledger-format.html#escrow) for held payments that have not yet been executed or canceled.
- [PayChannel objects](reference-ledger-format.html#paychannel) for open payment channels.
+- [Check objects](reference-ledger-format.html#check) for pending checks.
#### Request Format
@@ -63,7 +64,7 @@ The request includes the following parameters:
| `Field` | Type | Description |
|:---------------|:-------------------------------------------|:---------------|
| `account` | String | A unique identifier for the account, most commonly the account's address. |
-| `type` | String | _(Optional)_ If included, filter results to include only this type of ledger object. The valid types are: `offer`, `signer_list`, `state` (trust line), `escrow`, and `payment_channel`. |
+| `type` | String | _(Optional)_ If included, filter results to include only this type of ledger object. The valid types are: `check`, `escrow`, `offer`, `payment_channel`, `signer_list`, and `state` (trust line). |
| `ledger_hash` | String | _(Optional)_ A 20-byte hex string for the ledger version to use. (See [Specifying a Ledger](#specifying-ledgers)) |
| `ledger_index` | String or Unsigned Integer | _(Optional)_ The sequence number of the ledger to use, or a shortcut string to choose a ledger automatically. (See [Specifying a Ledger](#specifying-ledgers)) |
| `limit` | Unsigned Integer | _(Optional)_ The maximum number of objects to include in the results. Must be within the inclusive range 10 to 400 on non-admin connections. Defaults to 200. |
diff --git a/content/snippets/tx-type-links.md b/content/snippets/tx-type-links.md
index 71d11eec584..fffa56f1dfa 100644
--- a/content/snippets/tx-type-links.md
+++ b/content/snippets/tx-type-links.md
@@ -1,5 +1,8 @@
{% set txtypes = [
"AccountSet",
+ "CheckCancel",
+ "CheckCash",
+ "CheckCreate",
"EscrowCancel",
"EscrowCreate",
"EscrowFinish",
diff --git a/content/transactions/checkcancel.md b/content/transactions/checkcancel.md
new file mode 100644
index 00000000000..d69a7b8f9aa
--- /dev/null
+++ b/content/transactions/checkcancel.md
@@ -0,0 +1,28 @@
+## CheckCancel
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/app/tx/impl/CancelCheck.cpp "Source")
+
+_Requires the [Checks Amendment](reference-amendments.html#checks)._
+
+Cancels an unredeemed Check, removing it from the ledger without sending any money. The source or the destination of the check can cancel a Check at any time using this transaction type. If the Check has expired, any address can cancel it.
+
+Example CheckCancel:
+
+```json
+{
+ "Account": "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "TransactionType": "CheckCancel",
+ "CheckID": "49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0",
+ "Fee": "12"
+}
+```
+
+In addition to the [common fields](#common-fields), a CheckCancel transaction has the following:
+
+| Field | JSON Type | [Internal Type][] | Description |
+|:------------|:----------|:------------------|:-------------------------------|
+| `CheckID` | String | Hash256 | The ID of the [Check ledger object](reference-ledger-format.html#check) to cancel, as a 64-character hexadecimal string. |
+
+### Error Cases
+
+- If the object identified by the `CheckID` does not exist or is not a Check, the transaction fails with the result `tecNO_ENTRY`.
+- If the Check is not expired and the sender of the CheckCancel transaction is not the source or destination of the Check, the transaction fails with the result `tecNO_PERMISSION`.
diff --git a/content/transactions/checkcash.md b/content/transactions/checkcash.md
new file mode 100644
index 00000000000..6339994963e
--- /dev/null
+++ b/content/transactions/checkcash.md
@@ -0,0 +1,39 @@
+## CheckCash
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/app/tx/impl/CashCheck.cpp "Source")
+
+_Requires the [Checks Amendment](reference-amendments.html#checks)._
+
+Attempts to redeem a Check object in the ledger to receive up to the amount authorized by the corresponding [CheckCreate transaction][]. Only the `Destination` address of a Check can cash it with a CheckCash transaction. Cashing a check this way is similar to executing a [Payment][] initiated by the destination.
+
+Since the funds for a check are not guaranteed, redeeming a Check can fail because the sender does not have a high enough balance or because there is not enough liquidity to deliver the funds. If this happens, the Check remains in the ledger and the destination can try to cash it again later, or for a different amount.
+
+Example CheckCash:
+
+```json
+{
+ "Account": "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "TransactionType": "CheckCash",
+ "Amount": "100000000",
+ "CheckID": "838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334",
+ "Fee": "12"
+}
+```
+
+In addition to the [common fields](#common-fields), a CheckCancel transaction has the following:
+
+| Field | JSON Type | [Internal Type][] | Description |
+|:------------|:----------|:------------------|:-------------------------------|
+| `CheckID` | String | Hash256 | The ID of the [Check ledger object](reference-ledger-format.html#check) to cash, as a 64-character hexadecimal string. |
+| `Amount` | [Currency Amount][] | Amount | _(Optional)_ Redeem the Check for exactly this amount, if possible. The currency must match the `SendMax` of the corresponding CheckCreate transaction. You must provide either this field or `DeliverMin`. |
+| `DeliverMin` | [Currency Amount][] | Amount | _(Optional)_ Redeem the Check for at least this amount and as much as possible. The currency must match the `SendMax` of the corresponding CheckCreate transaction. You must provide either this field or `Amount`. |
+
+The transaction ***must*** include either `Amount` or `DeliverMin`, but not both.
+
+### Error Cases
+
+- If the sender of the CheckCash transaction is not the `Destination` of the check, the transaction fails with the result code `tecNO_PERMISSION`.
+- If the Check identified by the `CheckID` field does not exist, the transaction fails with the result `tecNO_ENTRY`.
+- If the Check identified by the `CheckID` field has already expired, the transaction fails with the result `tecEXPIRED`.
+- If the destination of the Check has the RequireDest flag enabled but the Check, as created, does not have a destination tag, the transaction fails with the result code `tecDST_TAG_NEEDED`.
+- If the transaction specifies both `Amount` and `DeliverMin`, or omits both, the transaction fails with the result `temMALFORMED`.
+- If the `Amount` or `DeliverMin` does not match the currency (and issuer, if not XRP) of the Check, the transaction fails with the result `temBAD_CURRENCY`
diff --git a/content/transactions/checkcreate.md b/content/transactions/checkcreate.md
new file mode 100644
index 00000000000..b816819b63f
--- /dev/null
+++ b/content/transactions/checkcreate.md
@@ -0,0 +1,41 @@
+## CheckCreate
+[[Source]
](https://github.com/ripple/rippled/blob/develop/src/ripple/app/tx/impl/CreateCheck.cpp "Source")
+
+_Requires the [Checks Amendment](reference-amendments.html#checks)._
+
+Create a Check object in the ledger, which is a deferred payment that can be cashed by its intended destination. The sender of this transaction is the sender of the Check.
+
+Example CheckCreate:
+
+```json
+{
+ "TransactionType": "CheckCreate",
+ "Account": "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination": "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "SendMax": "100000000",
+ "Expiration": 570113521,
+ "InvoiceID": "6F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B",
+ "DestinationTag": 1,
+ "Fee": "12"
+}
+```
+
+In addition to the [common fields](#common-fields), a CheckCancel transaction has the following:
+
+| Field | JSON Type | [Internal Type][] | Description |
+|:-----------------|:--------------------|:------------------|:----------------|
+| `Destination` | String | Account | The unique address of the [account](concept-accounts.html) that can cash the check. |
+| `SendMax` | [Currency Amount][] | Amount | Maximum amount of source currency the Check is allowed to debit the sender, including [transfer fees](concept-transfer-fees.html) on non-XRP currencies. The Check can only credit the destination with the same currency (from the same issuer, for non-XRP currencies). For non-XRP amounts, the nested field names MUST be lower-case. |
+| `DestinationTag` | Unsigned Integer | UInt32 | _(Optional)_ Arbitrary tag that identifies the reason for the check, or a hosted recipient to pay. |
+| `Expiration` | Unsigned Integer | UInt32 | _(Optional)_ Time after which the check is no longer valid, in [seconds since the Ripple Epoch](reference-rippled.html#specifying-time). |
+| `InvoiceID` | String | Hash256 | _(Optional)_ Arbitrary 256-bit hash representing a specific reason or identifier for this check. |
+
+### Error Cases
+
+- If the `Destination` is the sender of the transaction, the transaction fails with the result code `temREDUNDANT`.
+- If the `Destination` [account](concept-accounts.html) does not exist in the ledger, the transaction fails with the result code `tecNO_DST`.
+- If the `Destination` account has the RequireDest flag enabled but the transaction does not include a `DestinationTag` field, the transaction fails with the result code `tecDST_TAG_NEEDED`.
+- If `SendMax` specifies an issued currency which is [frozen](concept-freeze.html), the transaction fails with the result `tecFROZEN`.
+- If the `Expiration` of the transaction is in the past, the transaction fails with the result `tecEXPIRED`.
+- If the sender does not have enough XRP to meet the [owner reserve](concept-reserves.html#owner-reserves) after adding the Check, the transaction fails with the result `tecINSUFFICIENT_RESERVE`.
+- If either the sender or the destination of the Check cannot own more objects in the ledger, the transaction fails with the result `tecDIR_FULL`.
diff --git a/content/transactions/offercreate.md b/content/transactions/offercreate.md
index 5fee513bbfd..a8bacb0cf52 100644
--- a/content/transactions/offercreate.md
+++ b/content/transactions/offercreate.md
@@ -96,7 +96,7 @@ You can determine the final disposition of an offer with an `Expiration` as soon
**Note:** Since only new transactions can modify the ledger, an expired offer can stay on the ledger after it becomes inactive. The offer is treated as unfunded and has no effect, but it can continue to appear in results (for example, from the [ledger_entry](reference-rippled.html#ledger-entry) command). Later on, the expired offer can get finally deleted as a result of another transaction (such as another OfferCreate) if the server finds it while processing.
-If an OfferCreate transaction has an `Expiration` time that has already passed when the transaction first gets included in a ledger, the transaction does not execute the offer but still results in a `tesSUCCESS` transaction code. (This is because such a transaction could still successfully cancel another offer.)
+If an OfferCreate transaction has an `Expiration` time that has already passed when the transaction first gets included in a ledger, the transaction does not execute the offer. The result code of such a transaction depends on whether the [Checks amendment](reference-amendments.html#checks) is enabled. With the Checks amendment enabled, the transaction has the `tecEXPIRED` result code. Otherwise, the transaction has the `tesSUCCESS` transaction code. In either case, the transaction has no effect except to destroy the XRP paid as a [transaction cost](concept-transaction-cost.html).
### Auto-Bridging
diff --git a/content/tutorial-checks.md b/content/tutorial-checks.md
new file mode 100644
index 00000000000..43805bf3930
--- /dev/null
+++ b/content/tutorial-checks.md
@@ -0,0 +1,588 @@
+# Checks Tutorial (draft)
+
+***TODO: This whole page is just example requests and responses for now, not a finished page.***
+
+
+
+
+
+
+## Create check
+
+CheckCreate tx rUn84→rfkE1
+
+```
+$ ./rippled submit snkuWqxoqt6aeykTbkEWrTMJHrWGM '{ "TransactionType": "CheckCreate",
+ "Account": "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination": "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "SendMax": "100000000",
+ "Expiration": 570113521,
+ "InvoiceID": "46060241FABCF692D4D934BA2A6C4427CD4279083E38C77CBE642243E43BE291",
+ "DestinationTag": 1,
+ "Fee": "12"
+}'
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:01:20 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "engine_result" : "tesSUCCESS",
+ "engine_result_code" : 0,
+ "engine_result_message" : "The transaction was applied. Only final in a validated ledger.",
+ "status" : "success",
+ "tx_blob" : "120010228000000024000000022A21FB3DF12E00000001501146060241FABCF692D4D934BA2A6C4427CD4279083E38C77CBE642243E43BE29168400000000000000C694000000005F5E1007321022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF7874473045022100CC5B7069DF8133E91216F49933A685194DDB9BDCFF8241A7EF2F838993B98BEB022016DF6D746DF13AEA0D4BC867149BFEFFAF724AB0842A823A440D0EC684D876D181147990EC5D1D8DF69E070A968D4B186986FDF06ED0831449FF0C73CA6AF9733DA805F76CA2C37776B7C46B",
+ "tx_json" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "InvoiceID" : "46060241FABCF692D4D934BA2A6C4427CD4279083E38C77CBE642243E43BE291",
+ "SendMax" : "100000000",
+ "Sequence" : 2,
+ "SigningPubKey" : "022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF78",
+ "TransactionType" : "CheckCreate",
+ "TxnSignature" : "3045022100CC5B7069DF8133E91216F49933A685194DDB9BDCFF8241A7EF2F838993B98BEB022016DF6D746DF13AEA0D4BC867149BFEFFAF724AB0842A823A440D0EC684D876D1",
+ "hash" : "5463C6E08862A1FAE5EDAC12D70ADB16546A1F674930521295BC082494B62924"
+ }
+ }
+}
+```
+
+## wait for ledger to close
+
+or use `ledger_accept` if in standalone mode
+
+## Confirm CheckCreate by hash with tx command
+
+```
+$ ./rippled tx 5463C6E08862A1FAE5EDAC12D70ADB16546A1F674930521295BC082494B62924
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:01:53 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "InvoiceID" : "46060241FABCF692D4D934BA2A6C4427CD4279083E38C77CBE642243E43BE291",
+ "SendMax" : "100000000",
+ "Sequence" : 2,
+ "SigningPubKey" : "022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF78",
+ "TransactionType" : "CheckCreate",
+ "TxnSignature" : "3045022100CC5B7069DF8133E91216F49933A685194DDB9BDCFF8241A7EF2F838993B98BEB022016DF6D746DF13AEA0D4BC867149BFEFFAF724AB0842A823A440D0EC684D876D1",
+ "date" : 570070890,
+ "hash" : "5463C6E08862A1FAE5EDAC12D70ADB16546A1F674930521295BC082494B62924",
+ "inLedger" : 6,
+ "ledger_index" : 6,
+ "meta" : {
+ "AffectedNodes" : [
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "RootIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ }
+ },
+ {
+ "CreatedNode" : {
+ "LedgerEntryType" : "Check",
+ "LedgerIndex" : "49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0",
+ "NewFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "InvoiceID" : "46060241FABCF692D4D934BA2A6C4427CD4279083E38C77CBE642243E43BE291",
+ "SendMax" : "100000000",
+ "Sequence" : 2
+ }
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "RootIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Balance" : "4999999999976",
+ "Flags" : 0,
+ "OwnerCount" : 2,
+ "Sequence" : 3
+ },
+ "LedgerEntryType" : "AccountRoot",
+ "LedgerIndex" : "D3A1DBAA28717975A9119EC4CBC891BA9A66236C484F03C9911F463AD3B66DE0",
+ "PreviousFields" : {
+ "Balance" : "4999999999988",
+ "OwnerCount" : 1,
+ "Sequence" : 2
+ },
+ "PreviousTxnID" : "5168CE1F7FB10EED87E6DD94DAE5DD910B120869E5DCE682F6995BDAEA40DACA",
+ "PreviousTxnLgrSeq" : 5
+ }
+ }
+ ],
+ "TransactionIndex" : 0,
+ "TransactionResult" : "tesSUCCESS"
+ },
+ "status" : "success",
+ "validated" : true
+ }
+}
+```
+
+find CreatedNode of type Check, LedgerIndex is its ID: `49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0`
+
+## Cancel Check
+
+```
+$ ./rippled submit snkuWqxoqt6aeykTbkEWrTMJHrWGM '{
+> "Account": "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+> "TransactionType": "CheckCancel",
+> "CheckID": "49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0",
+> "Fee": "12"
+> }'
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:11:07 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "engine_result" : "tesSUCCESS",
+ "engine_result_code" : 0,
+ "engine_result_message" : "The transaction was applied. Only final in a validated ledger.",
+ "status" : "success",
+ "tx_blob" : "12001222800000002400000003501849647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB068400000000000000C7321022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF78744630440220615F9D19FA182F08530CD978A4C216C8676D0BA9EDB53A620AC909AA0EF0FE7E02203A09CC34C3DB85CCCB3137E78081F8F2B441FB0A3B9E40901F312D3CBA0A67A181147990EC5D1D8DF69E070A968D4B186986FDF06ED0",
+ "tx_json" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "CheckID" : "49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0",
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "Sequence" : 3,
+ "SigningPubKey" : "022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF78",
+ "TransactionType" : "CheckCancel",
+ "TxnSignature" : "30440220615F9D19FA182F08530CD978A4C216C8676D0BA9EDB53A620AC909AA0EF0FE7E02203A09CC34C3DB85CCCB3137E78081F8F2B441FB0A3B9E40901F312D3CBA0A67A1",
+ "hash" : "414558223CA8595916BB1FEF238B3BB601B7C0E52659292251CE613E6B4370F9"
+ }
+ }
+}
+```
+
+## wait for validation / ledger_accept
+
+## tx on checkcancel
+
+```
+$ ./rippled tx 414558223CA8595916BB1FEF238B3BB601B7C0E52659292251CE613E6B4370F9
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:11:53 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "CheckID" : "49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0",
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "Sequence" : 3,
+ "SigningPubKey" : "022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF78",
+ "TransactionType" : "CheckCancel",
+ "TxnSignature" : "30440220615F9D19FA182F08530CD978A4C216C8676D0BA9EDB53A620AC909AA0EF0FE7E02203A09CC34C3DB85CCCB3137E78081F8F2B441FB0A3B9E40901F312D3CBA0A67A1",
+ "date" : 570071520,
+ "hash" : "414558223CA8595916BB1FEF238B3BB601B7C0E52659292251CE613E6B4370F9",
+ "inLedger" : 7,
+ "ledger_index" : 7,
+ "meta" : {
+ "AffectedNodes" : [
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "RootIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ }
+ },
+ {
+ "DeletedNode" : {
+ "FinalFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationNode" : "0000000000000000",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "Flags" : 0,
+ "InvoiceID" : "46060241FABCF692D4D934BA2A6C4427CD4279083E38C77CBE642243E43BE291",
+ "OwnerNode" : "0000000000000000",
+ "PreviousTxnID" : "5463C6E08862A1FAE5EDAC12D70ADB16546A1F674930521295BC082494B62924",
+ "PreviousTxnLgrSeq" : 6,
+ "SendMax" : "100000000",
+ "Sequence" : 2
+ },
+ "LedgerEntryType" : "Check",
+ "LedgerIndex" : "49647F0D748DC3FE26BDACBC57F251AADEFFF391403EC9BF87C97F67E9977FB0"
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "RootIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Balance" : "4999999999964",
+ "Flags" : 0,
+ "OwnerCount" : 1,
+ "Sequence" : 4
+ },
+ "LedgerEntryType" : "AccountRoot",
+ "LedgerIndex" : "D3A1DBAA28717975A9119EC4CBC891BA9A66236C484F03C9911F463AD3B66DE0",
+ "PreviousFields" : {
+ "Balance" : "4999999999976",
+ "OwnerCount" : 2,
+ "Sequence" : 3
+ },
+ "PreviousTxnID" : "5463C6E08862A1FAE5EDAC12D70ADB16546A1F674930521295BC082494B62924",
+ "PreviousTxnLgrSeq" : 6
+ }
+ }
+ ],
+ "TransactionIndex" : 0,
+ "TransactionResult" : "tesSUCCESS"
+ },
+ "status" : "success",
+ "validated" : true
+ }
+}
+```
+
+deleted LedgerEntryType Check in metadata: √
+
+## new checkcreate
+
+```
+$ ./rippled submit snkuWqxoqt6aeykTbkEWrTMJHrWGM '{
+> "TransactionType": "CheckCreate",
+> "Account": "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+> "Destination": "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+> "SendMax": "100000000",
+> "Expiration": 570113521,
+> "InvoiceID": "6F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B",
+> "DestinationTag": 1,
+> "Fee": "12"
+> }'
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:14:21 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "engine_result" : "tesSUCCESS",
+ "engine_result_code" : 0,
+ "engine_result_message" : "The transaction was applied. Only final in a validated ledger.",
+ "status" : "success",
+ "tx_blob" : "120010228000000024000000042A21FB3DF12E0000000150116F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B68400000000000000C694000000005F5E1007321022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF787446304402206F4500FF59F5DC8684B5715941DCD0985A4107EE7BE11795D051C47E9F14B1C402201DA5634CD532DE77A3BE2C89F30F945514F93F3689A95DE7E42FC3072B10A5B181147990EC5D1D8DF69E070A968D4B186986FDF06ED0831449FF0C73CA6AF9733DA805F76CA2C37776B7C46B",
+ "tx_json" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "InvoiceID" : "6F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B",
+ "SendMax" : "100000000",
+ "Sequence" : 4,
+ "SigningPubKey" : "022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF78",
+ "TransactionType" : "CheckCreate",
+ "TxnSignature" : "304402206F4500FF59F5DC8684B5715941DCD0985A4107EE7BE11795D051C47E9F14B1C402201DA5634CD532DE77A3BE2C89F30F945514F93F3689A95DE7E42FC3072B10A5B1",
+ "hash" : "0FD9F719CDE29E6F6DF752B93EB9AC6FBB493BF989F2CB63B8C0E73A8DCDF61A"
+ }
+ }
+}
+```
+
+## validation/ledger_accept
+
+## and tx again
+
+```
+$ ./rippled tx 0FD9F719CDE29E6F6DF752B93EB9AC6FBB493BF989F2CB63B8C0E73A8DCDF61A
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:14:56 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "InvoiceID" : "6F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B",
+ "SendMax" : "100000000",
+ "Sequence" : 4,
+ "SigningPubKey" : "022C53CD19049F32F31848DD3B3BE5CEF6A2DD1EFDA7971AB3FA49B1BAF12AEF78",
+ "TransactionType" : "CheckCreate",
+ "TxnSignature" : "304402206F4500FF59F5DC8684B5715941DCD0985A4107EE7BE11795D051C47E9F14B1C402201DA5634CD532DE77A3BE2C89F30F945514F93F3689A95DE7E42FC3072B10A5B1",
+ "date" : 570071700,
+ "hash" : "0FD9F719CDE29E6F6DF752B93EB9AC6FBB493BF989F2CB63B8C0E73A8DCDF61A",
+ "inLedger" : 8,
+ "ledger_index" : 8,
+ "meta" : {
+ "AffectedNodes" : [
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "RootIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ }
+ },
+ {
+ "CreatedNode" : {
+ "LedgerEntryType" : "Check",
+ "LedgerIndex" : "838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334",
+ "NewFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "InvoiceID" : "6F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B",
+ "SendMax" : "100000000",
+ "Sequence" : 4
+ }
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "RootIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Balance" : "4999999999952",
+ "Flags" : 0,
+ "OwnerCount" : 2,
+ "Sequence" : 5
+ },
+ "LedgerEntryType" : "AccountRoot",
+ "LedgerIndex" : "D3A1DBAA28717975A9119EC4CBC891BA9A66236C484F03C9911F463AD3B66DE0",
+ "PreviousFields" : {
+ "Balance" : "4999999999964",
+ "OwnerCount" : 1,
+ "Sequence" : 4
+ },
+ "PreviousTxnID" : "414558223CA8595916BB1FEF238B3BB601B7C0E52659292251CE613E6B4370F9",
+ "PreviousTxnLgrSeq" : 7
+ }
+ }
+ ],
+ "TransactionIndex" : 0,
+ "TransactionResult" : "tesSUCCESS"
+ },
+ "status" : "success",
+ "validated" : true
+ }
+}
+```
+
+new check's id: `838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334`
+
+## Cash check (exact amount)
+
+rfkE1 sends this one
+
+```
+$ ./rippled submit sn2Zh1tRZyodU9qNy9tMnQr9UbBss '{
+> "Account": "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+> "TransactionType": "CheckCash",
+> "Amount": "100000000",
+> "CheckID": "838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334",
+> "Fee": "12"
+> }'
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:17:54 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "engine_result" : "tesSUCCESS",
+ "engine_result_code" : 0,
+ "engine_result_message" : "The transaction was applied. Only final in a validated ledger.",
+ "status" : "success",
+ "tx_blob" : "120011228000000024000000015018838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334614000000005F5E10068400000000000000C732102F135B14C552968B0ABE8493CC4C5795A7484D73F6BFD01379F73456F725F66ED74473045022100C64278AC90B841CD3EA9889A4847CAB3AC9927057A34130810FAA7FAC0C6E3290220347260A4C0A6DC9B699DA12510795B2B3414E1FA222AF743226345FBAAEF937C811449FF0C73CA6AF9733DA805F76CA2C37776B7C46B",
+ "tx_json" : {
+ "Account" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "Amount" : "100000000",
+ "CheckID" : "838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334",
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "Sequence" : 1,
+ "SigningPubKey" : "02F135B14C552968B0ABE8493CC4C5795A7484D73F6BFD01379F73456F725F66ED",
+ "TransactionType" : "CheckCash",
+ "TxnSignature" : "3045022100C64278AC90B841CD3EA9889A4847CAB3AC9927057A34130810FAA7FAC0C6E3290220347260A4C0A6DC9B699DA12510795B2B3414E1FA222AF743226345FBAAEF937C",
+ "hash" : "0521707D510858BC8AF69D2227E1D1ADA7DB7C5B4B74115BCD0D91B62AFA8EDC"
+ }
+ }
+}
+```
+
+## validate
+
+## tx the checkcash for final result
+
+```
+$ ./rippled tx 0521707D510858BC8AF69D2227E1D1ADA7DB7C5B4B74115BCD0D91B62AFA8EDC
+Loading: "/home/mduo13/.config/ripple/rippled.cfg"
+2018-Jan-24 01:18:39 HTTPClient:NFO Connecting to 127.0.0.1:5005
+
+{
+ "result" : {
+ "Account" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "Amount" : "100000000",
+ "CheckID" : "838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334",
+ "Fee" : "12",
+ "Flags" : 2147483648,
+ "Sequence" : 1,
+ "SigningPubKey" : "02F135B14C552968B0ABE8493CC4C5795A7484D73F6BFD01379F73456F725F66ED",
+ "TransactionType" : "CheckCash",
+ "TxnSignature" : "3045022100C64278AC90B841CD3EA9889A4847CAB3AC9927057A34130810FAA7FAC0C6E3290220347260A4C0A6DC9B699DA12510795B2B3414E1FA222AF743226345FBAAEF937C",
+ "date" : 570071920,
+ "hash" : "0521707D510858BC8AF69D2227E1D1ADA7DB7C5B4B74115BCD0D91B62AFA8EDC",
+ "inLedger" : 9,
+ "ledger_index" : 9,
+ "meta" : {
+ "AffectedNodes" : [
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "RootIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "032D861D151E38E86F46805ED1896D1A50144F65459717B6D12470A9E6E3B66E"
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Account" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "Balance" : "1000099999988",
+ "Flags" : 0,
+ "OwnerCount" : 0,
+ "Sequence" : 2
+ },
+ "LedgerEntryType" : "AccountRoot",
+ "LedgerIndex" : "38E1EF3284A45B090D549EFFB014ACF68927FE0884CDAF01CE3629DF90542D66",
+ "PreviousFields" : {
+ "Balance" : "1000000000000",
+ "Sequence" : 1
+ },
+ "PreviousTxnID" : "3E14D859F6B4BE923323EFC94571606455921E65173147A89BC6EDDA4374B294",
+ "PreviousTxnLgrSeq" : 5
+ }
+ },
+ {
+ "DeletedNode" : {
+ "FinalFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Destination" : "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
+ "DestinationNode" : "0000000000000000",
+ "DestinationTag" : 1,
+ "Expiration" : 570113521,
+ "Flags" : 0,
+ "InvoiceID" : "6F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B",
+ "OwnerNode" : "0000000000000000",
+ "PreviousTxnID" : "0FD9F719CDE29E6F6DF752B93EB9AC6FBB493BF989F2CB63B8C0E73A8DCDF61A",
+ "PreviousTxnLgrSeq" : 8,
+ "SendMax" : "100000000",
+ "Sequence" : 4
+ },
+ "LedgerEntryType" : "Check",
+ "LedgerIndex" : "838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334"
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Flags" : 0,
+ "Owner" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "RootIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ },
+ "LedgerEntryType" : "DirectoryNode",
+ "LedgerIndex" : "AD136EC2A266027D8F202C97D294BBE32F6FC2AD5501D9853F785FE77AB94C94"
+ }
+ },
+ {
+ "ModifiedNode" : {
+ "FinalFields" : {
+ "Account" : "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
+ "Balance" : "4999899999952",
+ "Flags" : 0,
+ "OwnerCount" : 1,
+ "Sequence" : 5
+ },
+ "LedgerEntryType" : "AccountRoot",
+ "LedgerIndex" : "D3A1DBAA28717975A9119EC4CBC891BA9A66236C484F03C9911F463AD3B66DE0",
+ "PreviousFields" : {
+ "Balance" : "4999999999952",
+ "OwnerCount" : 2
+ },
+ "PreviousTxnID" : "0FD9F719CDE29E6F6DF752B93EB9AC6FBB493BF989F2CB63B8C0E73A8DCDF61A",
+ "PreviousTxnLgrSeq" : 8
+ }
+ }
+ ],
+ "TransactionIndex" : 0,
+ "TransactionResult" : "tesSUCCESS"
+ },
+ "status" : "success",
+ "validated" : true
+ }
+}
+```
+
+## cash check (delivermin)
+
+***TODO: alternative version of cashing a check with delivermin instead of amount)