Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add ExpandedSignerList amendment support #2026

Merged
merged 12 commits into from
Sep 13, 2022
1 change: 1 addition & 0 deletions packages/xrpl/HISTORY.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ Subscribe to [the **xrpl-announce** mailing list](https://groups.google.com/g/xr

## Unreleased
### Added
* Add ExpandedSignerList amendment support
* When connected to nft-devnet, Client.fundWallet now defaults to using the nft-devnet faucet instead of requiring specification.

## 2.3.0 (2022-06-02)
Expand Down
21 changes: 21 additions & 0 deletions packages/xrpl/src/models/common/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -61,10 +61,31 @@ interface PathStep {

export type Path = PathStep[]

/**
* The object that describes the signer in SignerEntries.
*/
export interface SignerEntry {
/**
* The object that describes the signer in SignerEntries.
*/
SignerEntry: {
/**
* An XRP Ledger address whose signature contributes to the multi-signature.
* It does not need to be a funded address in the ledger.
*/
Account: string
/**
* 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 signer list's SignerQuorum value.
*/
SignerWeight: number
/**
* An arbitrary 256-bit (32-byte) field that can be used to identify the signer, which
* may be useful for smart contracts, or for identifying who controls a key in a large
* organization.
*/
WalletLocator?: string
JST5000 marked this conversation as resolved.
Show resolved Hide resolved
}
}

Expand Down
22 changes: 19 additions & 3 deletions packages/xrpl/src/models/transactions/signerListSet.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,15 @@ export interface SignerListSet extends BaseTransaction {
/**
* Array of SignerEntry objects, indicating the addresses and weights of
* signers in this list. This signer list must have at least 1 member and no
* more than 8 members. No address may appear more than once in the list, nor
* more than 32 members. No address may appear more than once in the list, nor
* may the Account submitting the transaction appear in the list.
*/
SignerEntries: SignerEntry[]
}

const MAX_SIGNERS = 8
const MAX_SIGNERS = 32

const HEX_WALLET_LOCATOR_REGEX = /^[0-9A-Fa-f]{64}$/u
JST5000 marked this conversation as resolved.
Show resolved Hide resolved

/**
* Verify the form and type of an SignerListSet at runtime.
Expand Down Expand Up @@ -61,7 +63,21 @@ export function validateSignerListSet(tx: Record<string, unknown>): void {

if (tx.SignerEntries.length > MAX_SIGNERS) {
throw new ValidationError(
'SignerListSet: maximum of 8 members allowed in SignerEntries',
`SignerListSet: maximum of ${MAX_SIGNERS} members allowed in SignerEntries`,
)
}

if (tx.SignerEntries.length > 0) {
for (const entry of tx.SignerEntries) {
const signerEntry = entry.SignerEntry
if (
signerEntry.WalletLocator !== undefined &&
!HEX_WALLET_LOCATOR_REGEX.test(signerEntry.WalletLocator)
) {
throw new ValidationError(
`SignerListSet: WalletLocator in SignerEntry must be a 256-bit (32-byte) hexadecimal value`,
)
}
}
}
}
123 changes: 123 additions & 0 deletions packages/xrpl/test/models/signerListSet.ts
Original file line number Diff line number Diff line change
Expand Up @@ -89,4 +89,127 @@ describe('SignerListSet', function () {
'SignerListSet: invalid SignerEntries',
)
})

it(`throws w/ maximum of 32 members allowed in SignerEntries`, function () {
signerListSetTx.SignerEntries = []
const accounts = [
'rBFBipte4nAQCTsRxd2czwvSurhCpAf4X6',
'r3ijUH32iiy9tYNj3rD7hKWYjy1BFUxngm',
'rpwq8vi4Mn3L5kDJmb8Mg59CanPFPzMCnj',
'rB72Gzqfejai46nkA4HaKYBHwAnn2yUoT4',
'rGqsJSAW71pCfUwDD5m52bLw69RzFg6kMW',
'rs8smPRA31Ym4mGxb1wzgwxtU5eVK82Gyk',
'rLrugpGxzezUQLDh7Jv1tZpouuV4MQLbU9',
'rUQ6zLXQdh1jJLGwMXp9P8rgi42kwuafzs',
'rMjY8sPdfxsyRrnVKQcutxr4mTHNXy9dEF',
'rUaxYLeFGm6SmMoa2WCqLKSyHwJyvaQmeG',
'r9wUfeVtqMfqrcDTfCpNYbNZvs5q9M9Rpo',
'rQncVNak5kvJGPUFa6fuKH7t8Usjs7Np1c',
'rnwbSSnPbVbUzuBa4etkeYrfy5v7SyhtPu',
'rDXh5D3t48MdBJyXByXq47k5P8Kuf1758B',
'rh1D4jd2mAiqUPHfAZ2cY9Nbfa3kAkaQXP',
'r9T129tXgtnyfGoLeS35c2HctaZAZSQoCH',
'rUd2uKsyCWfJP7Ve36mKoJbNCA7RYThnYk',
'r326x8PaAFtnaH7uoxaKrcDWuwpeHn4wDa',
'rpN3mkXkYhfNadcXPrY4LniM1KpM3egyQM',
'rsPKbR155hz1zrA4pSJp5Y2fxasZAatcHb',
'rsyWFLaEKTpaoSJusjpcDvGexuHCwMnqss',
'rUbc5RXfyF81oLDMgd3d7jpY9YMNMZG4XN',
'rGpYHM88BZe1iVKFHm5xiWYYxR74oxJEXf',
'rPsetWAtR1KxDtxzgHjRMD7Rc87rvXk5nD',
'rwSeNhL6Hi34igr12mCr61jY42psfTkWTq',
'r46Mygy98qjkDhVB6qs4sBnqaf7FPiA2vU',
'r4s8GmeYN4CiwVate1nMUvwMQbundqf5cW',
'rKAr4dQWDYG8cG2hSwJUVp4ry4WNaWiNgp',
'rPWXRLp1vqeUHEH3WiSKuyo9GM9XhaENQU',
'rPgmdBdRKGmndxNEYxUrrsYCZaS6go9RvW',
'rPDJZ9irzgwKRKScfEmuJMvUgrqZAJNCbL',
'rDuU2uSXMfEaoxN1qW8sj7aUNFLGEn3Hr2',
'rsbjSjA4TCB9gtm7x7SrWbZHB6g4tt9CGU',
]
signerListSetTx.SignerQuorum = accounts.length
for (const acc of accounts) {
signerListSetTx.SignerEntries.push({
SignerEntry: {
Account: acc,
SignerWeight: 1,
},
})
}

const errorMessage =
'SignerListSet: maximum of 32 members allowed in SignerEntries'
assert.throws(
() => validateSignerListSet(signerListSetTx),
ValidationError,
errorMessage,
)
assert.throws(
() => validate(signerListSetTx),
ValidationError,
errorMessage,
)
})

it(`verifies valid WalletLocator in SignerEntries`, function () {
signerListSetTx.SignerQuorum = 3
signerListSetTx.SignerEntries = [
{
SignerEntry: {
Account: 'rBFBipte4nAQCTsRxd2czwvSurhCpAf4X6',
SignerWeight: 1,
WalletLocator:
'CAFECAFECAFECAFECAFECAFECAFECAFECAFECAFECAFECAFECAFECAFECAFECAFE',
},
},
{
SignerEntry: {
Account: 'r3ijUH32iiy9tYNj3rD7hKWYjy1BFUxngm',
SignerWeight: 1,
},
},
{
SignerEntry: {
Account: 'rpwq8vi4Mn3L5kDJmb8Mg59CanPFPzMCnj',
SignerWeight: 1,
WalletLocator:
'00000000000000000000000000000000000000000000000000000000DEADBEEF',
},
},
]

assert.doesNotThrow(() => validateSignerListSet(signerListSetTx))
assert.doesNotThrow(() => validate(signerListSetTx))
})

it(`throws w/ invalid WalletLocator in SignerEntries`, function () {
signerListSetTx.SignerQuorum = 2
signerListSetTx.SignerEntries = [
{
SignerEntry: {
Account: 'rBFBipte4nAQCTsRxd2czwvSurhCpAf4X6',
SignerWeight: 1,
WalletLocator: 'not_valid',
},
},
{
SignerEntry: {
Account: 'r3ijUH32iiy9tYNj3rD7hKWYjy1BFUxngm',
SignerWeight: 1,
},
},
]
const errorMessage =
'SignerListSet: WalletLocator in SignerEntry must be a 256-bit (32-byte) hexadecimal value'
assert.throws(
() => validateSignerListSet(signerListSetTx),
ValidationError,
errorMessage,
)
assert.throws(
() => validate(signerListSetTx),
ValidationError,
errorMessage,
)
})
})