Compare commits
57 Commits
feat/testn
...
hooks-xaha
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
26aeea5a0d | ||
|
|
4a1bd98bd4 | ||
|
|
4171e618a2 | ||
|
|
1860b2d9ce | ||
|
|
488828f9d6 | ||
|
|
9f2105f6d3 | ||
|
|
6beccaef68 | ||
|
|
215af7258c | ||
|
|
ac3137088b | ||
|
|
1b8debda87 | ||
|
|
c50c7a5860 | ||
|
|
d21cda21d0 | ||
|
|
1cae0f161e | ||
|
|
412e3f2bbf | ||
|
|
dc37b1911a | ||
|
|
c348868c89 | ||
|
|
2c3cfebe3a | ||
|
|
6265a9cdbf | ||
|
|
1321b498cf | ||
|
|
801d9778cb | ||
|
|
2cf18ef61c | ||
|
|
4d2dc16ce5 | ||
|
|
9ecf5478e6 | ||
|
|
6d1ef110b7 | ||
|
|
b9edfcd63b | ||
|
|
b653d9a9cb | ||
|
|
da28e0a7d1 | ||
|
|
8a5b83d57f | ||
|
|
025eff6cf2 | ||
|
|
62d521b2cc | ||
|
|
7aafca21df | ||
|
|
80f58e903c | ||
|
|
c4af3df017 | ||
|
|
5d8d142bc4 | ||
|
|
e27a71d713 | ||
|
|
e08b07cbeb | ||
|
|
e4936c03ef | ||
|
|
21a69ac8ea | ||
|
|
52e4f219f7 | ||
|
|
e1f34c4beb | ||
|
|
54a89c969e | ||
|
|
ded867d997 | ||
|
|
0fce9af77c | ||
|
|
55c68c580a | ||
|
|
832a7997d1 | ||
|
|
4528e5a16e | ||
|
|
38f064c6d8 | ||
|
|
fbf4565dbc | ||
|
|
9001c64fed | ||
|
|
03b768db4e | ||
|
|
825af0db89 | ||
|
|
31043f33ab | ||
|
|
39699a1cb9 | ||
|
|
b50b300307 | ||
|
|
82c06cbb12 | ||
|
|
423ee18e6a | ||
|
|
3bb26d0c9b |
@@ -5,8 +5,8 @@ GITHUB_ID=""
|
||||
NEXT_PUBLIC_COMPILE_API_ENDPOINT="http://localhost:9000/api/build"
|
||||
NEXT_PUBLIC_COMPILE_API_BASE_URL="http://localhost:9000"
|
||||
NEXT_PUBLIC_LANGUAGE_SERVER_API_ENDPOINT="ws://localhost:9000/language-server/c"
|
||||
NEXT_PUBLIC_TESTNET_URL="hooks-testnet-v3.xrpl-labs.com"
|
||||
NEXT_PUBLIC_DEBUG_STREAM_URL="hooks-testnet-v3-debugstream.xrpl-labs.com"
|
||||
NEXT_PUBLIC_EXPLORER_URL="hooks-testnet-v3-explorer.xrpl-labs.com"
|
||||
NEXT_PUBLIC_TESTNET_URL="xahau-test.net"
|
||||
NEXT_PUBLIC_DEBUG_STREAM_URL="xahau-test.net/debugstream"
|
||||
NEXT_PUBLIC_EXPLORER_URL="explorer.xahau-test.net"
|
||||
NEXT_PUBLIC_NETWORK_ID="21338"
|
||||
NEXT_PUBLIC_SITE_URL="http://localhost:3000"
|
||||
@@ -231,7 +231,7 @@ const Navigation = () => {
|
||||
as="a"
|
||||
rel="noreferrer noopener"
|
||||
target="_blank"
|
||||
href="https://xrpl-hooks.readme.io/v2.0/docs"
|
||||
href="https://xrpl-hooks.readme.io/docs"
|
||||
>
|
||||
<ArrowUpRight size="15px" /> Hooks documentation
|
||||
</Text>
|
||||
@@ -352,7 +352,7 @@ const Navigation = () => {
|
||||
</Button>
|
||||
</Link>
|
||||
</ButtonGroup>
|
||||
<Link href="https://xrpl-hooks.readme.io/v2.0" passHref>
|
||||
<Link href="https://xrpl-hooks.readme.io/" passHref>
|
||||
<a target="_blank" rel="noreferrer noopener">
|
||||
<Button outline>
|
||||
<BookOpen size="15px" />
|
||||
|
||||
@@ -3,13 +3,11 @@ import { mauve, mauveDark, purple, purpleDark } from '@radix-ui/colors'
|
||||
import { useTheme } from 'next-themes'
|
||||
import { styled } from '../stitches.config'
|
||||
import dynamic from 'next/dynamic'
|
||||
import type { Props } from 'react-select'
|
||||
import type { Props, StylesConfig } from 'react-select'
|
||||
const SelectInput = dynamic(() => import('react-select'), { ssr: false })
|
||||
const CreatableSelectInput = dynamic(() => import('react-select/creatable'), { ssr: false })
|
||||
|
||||
// eslint-disable-next-line react/display-name
|
||||
const Select = forwardRef<any, Props>((props, ref) => {
|
||||
const { theme } = useTheme()
|
||||
const isDark = theme === 'dark'
|
||||
const getColors = (isDark: boolean) => {
|
||||
const colors: any = {
|
||||
// primary: pink.pink9,
|
||||
active: isDark ? purpleDark.purple9 : purple.purple9,
|
||||
@@ -30,15 +28,17 @@ const Select = forwardRef<any, Props>((props, ref) => {
|
||||
}
|
||||
colors.outline = colors.background
|
||||
colors.selected = colors.secondary
|
||||
return (
|
||||
<SelectInput
|
||||
ref={ref}
|
||||
menuPosition={props.menuPosition || 'fixed'}
|
||||
styles={{
|
||||
return colors
|
||||
}
|
||||
|
||||
const getStyles = (isDark: boolean) => {
|
||||
const colors = getColors(isDark)
|
||||
const styles: StylesConfig = {
|
||||
container: provided => {
|
||||
return {
|
||||
...provided,
|
||||
position: 'relative'
|
||||
position: 'relative',
|
||||
width: '100%'
|
||||
}
|
||||
},
|
||||
singleValue: provided => ({
|
||||
@@ -107,16 +107,57 @@ const Select = forwardRef<any, Props>((props, ref) => {
|
||||
dropdownIndicator: (provided, state) => {
|
||||
return {
|
||||
...provided,
|
||||
padding: 6,
|
||||
color: state.isFocused ? colors.border : colors.secondary,
|
||||
':hover': {
|
||||
color: colors.border
|
||||
}
|
||||
}
|
||||
},
|
||||
clearIndicator: provided => {
|
||||
return {
|
||||
...provided,
|
||||
padding: 6,
|
||||
color: colors.secondary,
|
||||
':hover': {
|
||||
color: colors.border
|
||||
}
|
||||
}}
|
||||
}
|
||||
}
|
||||
}
|
||||
return styles
|
||||
}
|
||||
|
||||
// eslint-disable-next-line react/display-name
|
||||
const Select = forwardRef<any, Props>((props, ref) => {
|
||||
const { theme } = useTheme()
|
||||
const isDark = theme === 'dark'
|
||||
const styles = getStyles(isDark)
|
||||
return (
|
||||
<SelectInput
|
||||
ref={ref}
|
||||
menuPosition={props.menuPosition || 'fixed'}
|
||||
styles={styles}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
})
|
||||
|
||||
// eslint-disable-next-line react/display-name
|
||||
const Creatable = forwardRef<any, Props>((props, ref) => {
|
||||
const { theme } = useTheme()
|
||||
const isDark = theme === 'dark'
|
||||
const styles = getStyles(isDark)
|
||||
return (
|
||||
<CreatableSelectInput
|
||||
ref={ref}
|
||||
formatCreateLabel={label => `Enter "${label}"`}
|
||||
menuPosition={props.menuPosition || 'fixed'}
|
||||
styles={styles}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
})
|
||||
|
||||
export default styled(Select, {})
|
||||
export const CreatableSelect = styled(Creatable, {})
|
||||
|
||||
71
components/Sequence.tsx
Normal file
71
components/Sequence.tsx
Normal file
@@ -0,0 +1,71 @@
|
||||
import { FC, useCallback, useState } from 'react'
|
||||
import state from '../state'
|
||||
import { Flex, Input, Button } from '.'
|
||||
import fetchAccountInfo from '../utils/accountInfo'
|
||||
import { useSnapshot } from 'valtio'
|
||||
|
||||
interface AccountSequenceProps {
|
||||
address?: string
|
||||
}
|
||||
const AccountSequence: FC<AccountSequenceProps> = ({ address }) => {
|
||||
const { accounts } = useSnapshot(state)
|
||||
const account = accounts.find(acc => acc.address === address)
|
||||
const [isLoading, setIsLoading] = useState(false)
|
||||
const setSequence = useCallback(
|
||||
(sequence: number) => {
|
||||
const acc = state.accounts.find(acc => acc.address == address)
|
||||
if (!acc) return
|
||||
acc.sequence = sequence
|
||||
},
|
||||
[address]
|
||||
)
|
||||
const handleUpdateSequence = useCallback(
|
||||
async (silent?: boolean) => {
|
||||
if (!account) return
|
||||
setIsLoading(true)
|
||||
|
||||
const info = await fetchAccountInfo(account.address, { silent })
|
||||
if (info) {
|
||||
setSequence(info.Sequence)
|
||||
}
|
||||
|
||||
setIsLoading(false)
|
||||
},
|
||||
[account, setSequence]
|
||||
)
|
||||
const disabled = !account
|
||||
return (
|
||||
<Flex row align="center" fluid>
|
||||
<Input
|
||||
placeholder="Account sequence"
|
||||
value={account?.sequence || ""}
|
||||
disabled={!account}
|
||||
type="number"
|
||||
readOnly={true}
|
||||
/>
|
||||
<Button
|
||||
size="xs"
|
||||
variant="primary"
|
||||
type="button"
|
||||
outline
|
||||
disabled={disabled}
|
||||
isDisabled={disabled}
|
||||
isLoading={isLoading}
|
||||
css={{
|
||||
background: '$backgroundAlt',
|
||||
position: 'absolute',
|
||||
right: '$2',
|
||||
fontSize: '$xs',
|
||||
cursor: 'pointer',
|
||||
alignContent: 'center',
|
||||
display: 'flex'
|
||||
}}
|
||||
onClick={() => handleUpdateSequence()}
|
||||
>
|
||||
Update
|
||||
</Button>
|
||||
</Flex>
|
||||
)
|
||||
}
|
||||
|
||||
export default AccountSequence
|
||||
@@ -21,6 +21,7 @@ import { prepareDeployHookTx, sha256 } from '../state/actions/deployHook'
|
||||
import estimateFee from '../utils/estimateFee'
|
||||
import { getParameters, getInvokeOptions, transactionOptions, SetHookData } from '../utils/setHook'
|
||||
import { capitalize } from '../utils/helpers'
|
||||
import AccountSequence from './Sequence'
|
||||
|
||||
export const SetHookDialog: React.FC<{ accountAddress: string }> = React.memo(
|
||||
({ accountAddress }) => {
|
||||
@@ -190,6 +191,10 @@ export const SetHookDialog: React.FC<{ accountAddress: string }> = React.memo(
|
||||
onChange={(acc: any) => setSelectedAccount(acc)}
|
||||
/>
|
||||
</Box>
|
||||
<Box css={{ width: '100%', position: 'relative' }}>
|
||||
<Label>Sequence</Label>
|
||||
<AccountSequence address={selectedAccount?.value} />
|
||||
</Box>
|
||||
<Box css={{ width: '100%' }}>
|
||||
<Label>Invoke on transactions</Label>
|
||||
<Controller
|
||||
|
||||
@@ -20,6 +20,7 @@ import { TxUI } from './ui'
|
||||
import { default as _estimateFee } from '../../utils/estimateFee'
|
||||
import toast from 'react-hot-toast'
|
||||
import { combineFlags, extractFlags, transactionFlags } from '../../state/constants/flags'
|
||||
import { SetHookData, toHex } from '../../utils/setHook'
|
||||
|
||||
export interface TransactionProps {
|
||||
header: string
|
||||
@@ -40,20 +41,40 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
|
||||
const prepareOptions = useCallback(
|
||||
(state: Partial<TransactionState> = txState) => {
|
||||
const { selectedTransaction, selectedDestAccount, selectedAccount, txFields, selectedFlags } =
|
||||
state
|
||||
const {
|
||||
selectedTransaction,
|
||||
selectedAccount,
|
||||
txFields,
|
||||
selectedFlags,
|
||||
hookParameters,
|
||||
memos
|
||||
} = state
|
||||
|
||||
const TransactionType = selectedTransaction?.value || null
|
||||
const Destination = selectedDestAccount?.value || txFields?.Destination
|
||||
const Account = selectedAccount?.value || null
|
||||
const Flags = combineFlags(selectedFlags?.map(flag => flag.value)) || txFields?.Flags
|
||||
const HookParameters = Object.entries(hookParameters || {}).reduce<
|
||||
SetHookData['HookParameters']
|
||||
>((acc, [_, { label, value }]) => {
|
||||
return acc.concat({
|
||||
HookParameter: { HookParameterName: toHex(label), HookParameterValue: value }
|
||||
})
|
||||
}, [])
|
||||
const Memos = memos
|
||||
? Object.entries(memos).reduce<SetHookData['Memos']>((acc, [_, { format, data, type }]) => {
|
||||
return acc?.concat({
|
||||
Memo: { MemoData: data, MemoFormat: toHex(format), MemoType: toHex(type) }
|
||||
})
|
||||
}, [])
|
||||
: undefined
|
||||
|
||||
return prepareTransaction({
|
||||
...txFields,
|
||||
HookParameters,
|
||||
Flags,
|
||||
TransactionType,
|
||||
Destination,
|
||||
Account
|
||||
Account,
|
||||
Memos
|
||||
})
|
||||
},
|
||||
[txState]
|
||||
@@ -69,15 +90,29 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
}
|
||||
}, [selectedAccount?.value, selectedTransaction?.value, setState, txIsLoading])
|
||||
|
||||
const getJsonString = useCallback(
|
||||
(state?: Partial<TransactionState>) =>
|
||||
JSON.stringify(prepareOptions?.(state) || {}, null, editorSettings.tabSize),
|
||||
[editorSettings.tabSize, prepareOptions]
|
||||
)
|
||||
|
||||
const saveEditorState = useCallback(
|
||||
(value: string = '', transactionType?: string) => {
|
||||
const pTx = prepareState(value, transactionType)
|
||||
if (pTx) {
|
||||
pTx.editorValue = getJsonString(pTx)
|
||||
return setState(pTx)
|
||||
}
|
||||
},
|
||||
[getJsonString, setState]
|
||||
)
|
||||
|
||||
const submitTest = useCallback(async () => {
|
||||
let st: TransactionState | undefined
|
||||
const tt = txState.selectedTransaction?.value
|
||||
if (viewType === 'json') {
|
||||
// save the editor state first
|
||||
const pst = prepareState(editorValue || '', tt)
|
||||
if (!pst) return
|
||||
|
||||
st = setState(pst)
|
||||
st = saveEditorState(editorValue, tt)
|
||||
if (!st) return
|
||||
}
|
||||
|
||||
const account = accounts.find(acc => acc.address === selectedAccount?.value)
|
||||
@@ -90,11 +125,12 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
throw Error('Account must be selected from imported accounts!')
|
||||
}
|
||||
const options = prepareOptions(st)
|
||||
|
||||
const fields = getTxFields(options.TransactionType)
|
||||
if (fields.Destination && !options.Destination) {
|
||||
throw Error('Destination account is required!')
|
||||
// delete unnecessary fields
|
||||
Object.keys(options).forEach(field => {
|
||||
if (!options[field]) {
|
||||
delete options[field]
|
||||
}
|
||||
})
|
||||
|
||||
await sendTransaction(account, options, { logPrefix })
|
||||
} catch (error) {
|
||||
@@ -108,23 +144,18 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
}
|
||||
setState({ txIsLoading: false })
|
||||
}, [
|
||||
txState.selectedTransaction?.value,
|
||||
viewType,
|
||||
accounts,
|
||||
txIsDisabled,
|
||||
setState,
|
||||
header,
|
||||
saveEditorState,
|
||||
editorValue,
|
||||
txState,
|
||||
selectedAccount?.value,
|
||||
prepareOptions
|
||||
])
|
||||
|
||||
const getJsonString = useCallback(
|
||||
(state?: Partial<TransactionState>) =>
|
||||
JSON.stringify(prepareOptions?.(state) || {}, null, editorSettings.tabSize),
|
||||
[editorSettings.tabSize, prepareOptions]
|
||||
)
|
||||
|
||||
const resetState = useCallback(
|
||||
(transactionType: SelectOption | undefined = defaultTransactionType) => {
|
||||
const fields = getTxFields(transactionType?.value)
|
||||
@@ -134,13 +165,6 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
selectedTransaction: transactionType
|
||||
}
|
||||
|
||||
if (fields.Destination !== undefined) {
|
||||
nwState.selectedDestAccount = null
|
||||
fields.Destination = ''
|
||||
} else {
|
||||
fields.Destination = undefined
|
||||
}
|
||||
|
||||
if (transactionType?.value && transactionFlags[transactionType?.value] && fields.Flags) {
|
||||
nwState.selectedFlags = extractFlags(transactionType.value, fields.Flags)
|
||||
fields.Flags = undefined
|
||||
@@ -177,11 +201,21 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
[accounts, prepareOptions, setState, txState]
|
||||
)
|
||||
|
||||
const switchToJson = useCallback(() => {
|
||||
const editorValue = getJsonString()
|
||||
setState({ viewType: 'json', editorValue })
|
||||
}, [getJsonString, setState])
|
||||
|
||||
const switchToUI = useCallback(() => {
|
||||
setState({ viewType: 'ui' })
|
||||
}, [setState])
|
||||
|
||||
return (
|
||||
<Box css={{ position: 'relative', height: 'calc(100% - 28px)' }} {...props}>
|
||||
{viewType === 'json' ? (
|
||||
<TxJson
|
||||
getJsonString={getJsonString}
|
||||
saveEditorState={saveEditorState}
|
||||
header={header}
|
||||
state={txState}
|
||||
setState={setState}
|
||||
@@ -189,6 +223,7 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
/>
|
||||
) : (
|
||||
<TxUI
|
||||
switchToJson={switchToJson}
|
||||
state={txState}
|
||||
resetState={resetState}
|
||||
setState={setState}
|
||||
@@ -209,8 +244,8 @@ const Transaction: FC<TransactionProps> = ({ header, state: txState, ...props })
|
||||
<Button
|
||||
onClick={() => {
|
||||
if (viewType === 'ui') {
|
||||
setState({ viewType: 'json' })
|
||||
} else setState({ viewType: 'ui' })
|
||||
switchToJson()
|
||||
} else switchToUI()
|
||||
}}
|
||||
outline
|
||||
>
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import { FC, useCallback, useEffect, useMemo, useState } from 'react'
|
||||
import { FC, useCallback, useEffect, useState } from 'react'
|
||||
import { useSnapshot } from 'valtio'
|
||||
import state, { prepareState, transactionsData, TransactionState } from '../../state'
|
||||
import state, { transactionsData, TransactionState } from '../../state'
|
||||
import Text from '../Text'
|
||||
import { Flex, Link } from '..'
|
||||
import { showAlert } from '../../state/actions/showAlert'
|
||||
@@ -11,27 +11,27 @@ import Monaco from '../Monaco'
|
||||
import type monaco from 'monaco-editor'
|
||||
|
||||
interface JsonProps {
|
||||
getJsonString?: (state?: Partial<TransactionState>) => string
|
||||
getJsonString: (st?: Partial<TransactionState>) => string
|
||||
saveEditorState: (val?: string, tt?: string) => TransactionState | undefined
|
||||
header?: string
|
||||
setState: (pTx?: Partial<TransactionState> | undefined) => void
|
||||
state: TransactionState
|
||||
estimateFee?: () => Promise<string | undefined>
|
||||
}
|
||||
|
||||
export const TxJson: FC<JsonProps> = ({ getJsonString, state: txState, header, setState }) => {
|
||||
export const TxJson: FC<JsonProps> = ({
|
||||
getJsonString,
|
||||
state: txState,
|
||||
header,
|
||||
setState,
|
||||
saveEditorState
|
||||
}) => {
|
||||
const { editorSettings, accounts } = useSnapshot(state)
|
||||
const { editorValue, estimatedFee } = txState
|
||||
const { editorValue, estimatedFee, editorIsSaved } = txState
|
||||
const [currTxType, setCurrTxType] = useState<string | undefined>(
|
||||
txState.selectedTransaction?.value
|
||||
)
|
||||
|
||||
useEffect(() => {
|
||||
setState({
|
||||
editorValue: getJsonString?.()
|
||||
})
|
||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||
}, [])
|
||||
|
||||
useEffect(() => {
|
||||
const parsed = parseJSON(editorValue)
|
||||
if (!parsed) return
|
||||
@@ -44,29 +44,19 @@ export const TxJson: FC<JsonProps> = ({ getJsonString, state: txState, header, s
|
||||
}
|
||||
}, [editorValue])
|
||||
|
||||
const saveState = (value: string, transactionType?: string) => {
|
||||
const tx = prepareState(value, transactionType)
|
||||
if (tx) {
|
||||
setState(tx)
|
||||
setState({
|
||||
editorValue: getJsonString?.(tx)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
const discardChanges = () => {
|
||||
showAlert('Confirm', {
|
||||
body: 'Are you sure to discard these changes?',
|
||||
confirmText: 'Yes',
|
||||
onCancel: () => {},
|
||||
onConfirm: () => setState({ editorValue: getJsonString?.() })
|
||||
onConfirm: () => setState({ editorValue: getJsonString() })
|
||||
})
|
||||
}
|
||||
|
||||
const onExit = (value: string) => {
|
||||
const options = parseJSON(value)
|
||||
if (options) {
|
||||
saveState(value, currTxType)
|
||||
saveEditorState(value, currTxType)
|
||||
return
|
||||
}
|
||||
showAlert('Error!', {
|
||||
@@ -163,8 +153,6 @@ export const TxJson: FC<JsonProps> = ({ getJsonString, state: txState, header, s
|
||||
})
|
||||
}, [getSchemas, monacoInst])
|
||||
|
||||
const hasUnsaved = useMemo(() => editorValue !== getJsonString?.(), [editorValue, getJsonString])
|
||||
|
||||
return (
|
||||
<Monaco
|
||||
rootProps={{
|
||||
@@ -174,7 +162,7 @@ export const TxJson: FC<JsonProps> = ({ getJsonString, state: txState, header, s
|
||||
id={header}
|
||||
height="100%"
|
||||
value={editorValue}
|
||||
onChange={val => setState({ editorValue: val })}
|
||||
onChange={val => setState({ editorValue: val, editorIsSaved: false })}
|
||||
onMount={(editor, monaco) => {
|
||||
editor.updateOptions({
|
||||
minimap: { enabled: false },
|
||||
@@ -190,12 +178,12 @@ export const TxJson: FC<JsonProps> = ({ getJsonString, state: txState, header, s
|
||||
model?.onWillDispose(() => onExit(model.getValue()))
|
||||
}}
|
||||
overlay={
|
||||
hasUnsaved ? (
|
||||
!editorIsSaved ? (
|
||||
<Flex row align="center" css={{ fontSize: '$xs', color: '$textMuted', ml: 'auto' }}>
|
||||
<Text muted small>
|
||||
This file has unsaved changes.
|
||||
</Text>
|
||||
<Link css={{ ml: '$1' }} onClick={() => saveState(editorValue || '', currTxType)}>
|
||||
<Link css={{ ml: '$1' }} onClick={() => saveEditorState(editorValue, currTxType)}>
|
||||
save
|
||||
</Link>
|
||||
<Link css={{ ml: '$1' }} onClick={discardChanges}>
|
||||
|
||||
@@ -1,34 +1,43 @@
|
||||
import { FC, useCallback, useEffect, useMemo, useState } from 'react'
|
||||
import { FC, ReactNode, useCallback, useEffect, useState } from 'react'
|
||||
import Container from '../Container'
|
||||
import Flex from '../Flex'
|
||||
import Input from '../Input'
|
||||
import Select from '../Select'
|
||||
import Select, { CreatableSelect } from '../Select'
|
||||
import Text from '../Text'
|
||||
import {
|
||||
SelectOption,
|
||||
TransactionState,
|
||||
transactionsOptions,
|
||||
TxFields,
|
||||
getTxFields,
|
||||
defaultTransactionType
|
||||
} from '../../state/transactions'
|
||||
import { useSnapshot } from 'valtio'
|
||||
import state from '../../state'
|
||||
import { streamState } from '../DebugStream'
|
||||
import { Button } from '..'
|
||||
import { Box, Button } from '..'
|
||||
import Textarea from '../Textarea'
|
||||
import { getFlags } from '../../state/constants/flags'
|
||||
import { Plus, Trash } from 'phosphor-react'
|
||||
import AccountSequence from '../Sequence'
|
||||
import { capitalize, typeIs } from '../../utils/helpers'
|
||||
|
||||
interface UIProps {
|
||||
setState: (pTx?: Partial<TransactionState> | undefined) => TransactionState | undefined
|
||||
resetState: (tt?: SelectOption) => TransactionState | undefined
|
||||
state: TransactionState
|
||||
estimateFee?: (...arg: any) => Promise<string | undefined>
|
||||
switchToJson: () => void
|
||||
}
|
||||
|
||||
export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estimateFee }) => {
|
||||
export const TxUI: FC<UIProps> = ({
|
||||
state: txState,
|
||||
setState,
|
||||
resetState,
|
||||
estimateFee,
|
||||
switchToJson
|
||||
}) => {
|
||||
const { accounts } = useSnapshot(state)
|
||||
const { selectedAccount, selectedDestAccount, selectedTransaction, txFields, selectedFlags } =
|
||||
const { selectedAccount, selectedTransaction, txFields, selectedFlags, hookParameters, memos } =
|
||||
txState
|
||||
|
||||
const accountOptions: SelectOption[] = accounts.map(acc => ({
|
||||
@@ -36,13 +45,6 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
value: acc.address
|
||||
}))
|
||||
|
||||
const destAccountOptions: SelectOption[] = accounts
|
||||
.map(acc => ({
|
||||
label: acc.name,
|
||||
value: acc.address
|
||||
}))
|
||||
.filter(acc => acc.value !== selectedAccount?.value)
|
||||
|
||||
const flagsOptions: SelectOption[] = Object.entries(
|
||||
getFlags(selectedTransaction?.value) || {}
|
||||
).map(([label, value]) => ({
|
||||
@@ -71,6 +73,22 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
[setState, txFields]
|
||||
)
|
||||
|
||||
const setRawField = useCallback(
|
||||
(field: keyof TxFields, type: string, value: any) => {
|
||||
// TODO $type should be a narrowed type
|
||||
setState({
|
||||
txFields: {
|
||||
...txFields,
|
||||
[field]: {
|
||||
$type: type,
|
||||
$value: value
|
||||
}
|
||||
}
|
||||
})
|
||||
},
|
||||
[setState, txFields]
|
||||
)
|
||||
|
||||
const handleEstimateFee = useCallback(
|
||||
async (state?: TransactionState, silent?: boolean) => {
|
||||
setFeeLoading(true)
|
||||
@@ -94,8 +112,6 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
[handleEstimateFee, resetState, setState]
|
||||
)
|
||||
|
||||
const switchToJson = () => setState({ viewType: 'json' })
|
||||
|
||||
// default tx
|
||||
useEffect(() => {
|
||||
if (selectedTransaction?.value) return
|
||||
@@ -105,22 +121,23 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
}
|
||||
}, [handleChangeTxType, selectedTransaction?.value])
|
||||
|
||||
const fields = useMemo(
|
||||
() => getTxFields(selectedTransaction?.value),
|
||||
[selectedTransaction?.value]
|
||||
)
|
||||
|
||||
const richFields = ['TransactionType', 'Account']
|
||||
if (fields.Destination !== undefined) {
|
||||
richFields.push('Destination')
|
||||
}
|
||||
const richFields = ['TransactionType', 'Account', 'HookParameters', 'Memos']
|
||||
|
||||
if (flagsOptions.length) {
|
||||
richFields.push('Flags')
|
||||
}
|
||||
|
||||
const otherFields = Object.keys(txFields).filter(k => !richFields.includes(k)) as [keyof TxFields]
|
||||
const amountOptions = [
|
||||
{ label: 'XRP', value: 'xrp' },
|
||||
{ label: 'Token', value: 'token' }
|
||||
] as const
|
||||
|
||||
const defaultTokenAmount = {
|
||||
value: '0',
|
||||
currency: '',
|
||||
issuer: ''
|
||||
}
|
||||
return (
|
||||
<Container
|
||||
css={{
|
||||
@@ -130,94 +147,32 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
}}
|
||||
>
|
||||
<Flex column fluid css={{ height: '100%', overflowY: 'auto', pr: '$1' }}>
|
||||
<Flex
|
||||
row
|
||||
fluid
|
||||
css={{
|
||||
justifyContent: 'flex-end',
|
||||
alignItems: 'center',
|
||||
mb: '$3',
|
||||
mt: '1px',
|
||||
pr: '1px'
|
||||
}}
|
||||
>
|
||||
<Text muted css={{ mr: '$3' }}>
|
||||
Transaction type:{' '}
|
||||
</Text>
|
||||
<TxField label="Transaction type">
|
||||
<Select
|
||||
instanceId="transactionsType"
|
||||
placeholder="Select transaction type"
|
||||
options={transactionsOptions}
|
||||
hideSelectedOptions
|
||||
css={{ width: '70%' }}
|
||||
value={selectedTransaction}
|
||||
onChange={(tt: any) => handleChangeTxType(tt)}
|
||||
/>
|
||||
</Flex>
|
||||
<Flex
|
||||
row
|
||||
fluid
|
||||
css={{
|
||||
justifyContent: 'flex-end',
|
||||
alignItems: 'center',
|
||||
mb: '$3',
|
||||
pr: '1px'
|
||||
}}
|
||||
>
|
||||
<Text muted css={{ mr: '$3' }}>
|
||||
Account:{' '}
|
||||
</Text>
|
||||
</TxField>
|
||||
<TxField label="Account">
|
||||
<Select
|
||||
instanceId="from-account"
|
||||
placeholder="Select your account"
|
||||
css={{ width: '70%' }}
|
||||
options={accountOptions}
|
||||
value={selectedAccount}
|
||||
onChange={(acc: any) => handleSetAccount(acc)} // TODO make react-select have correct types for acc
|
||||
/>
|
||||
</Flex>
|
||||
{richFields.includes('Destination') && (
|
||||
<Flex
|
||||
row
|
||||
fluid
|
||||
css={{
|
||||
justifyContent: 'flex-end',
|
||||
alignItems: 'center',
|
||||
mb: '$3',
|
||||
pr: '1px'
|
||||
}}
|
||||
>
|
||||
<Text muted css={{ mr: '$3', textAlign: 'end' }}>
|
||||
Destination account:{' '}
|
||||
</Text>
|
||||
<Select
|
||||
instanceId="to-account"
|
||||
placeholder="Select the destination account"
|
||||
css={{ width: '70%' }}
|
||||
options={destAccountOptions}
|
||||
value={selectedDestAccount}
|
||||
isClearable
|
||||
onChange={(acc: any) => setState({ selectedDestAccount: acc })}
|
||||
/>
|
||||
</Flex>
|
||||
)}
|
||||
</TxField>
|
||||
<TxField label="Sequence">
|
||||
<AccountSequence address={selectedAccount?.value} />
|
||||
</TxField>
|
||||
{richFields.includes('Flags') && (
|
||||
<Flex
|
||||
row
|
||||
fluid
|
||||
css={{
|
||||
justifyContent: 'flex-end',
|
||||
alignItems: 'center',
|
||||
mb: '$3',
|
||||
pr: '1px'
|
||||
}}
|
||||
>
|
||||
<Text muted css={{ mr: '$3' }}>
|
||||
Flags:{' '}
|
||||
</Text>
|
||||
<TxField label="Flags">
|
||||
<Select
|
||||
isClearable
|
||||
css={{ width: '70%' }}
|
||||
instanceId="flags"
|
||||
placeholder="Select flags to apply"
|
||||
menuPosition="fixed"
|
||||
@@ -229,41 +184,139 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
selectedFlags ? selectedFlags.length >= flagsOptions.length - 1 : false
|
||||
}
|
||||
/>
|
||||
</Flex>
|
||||
</TxField>
|
||||
)}
|
||||
{otherFields.map(field => {
|
||||
let _value = txFields[field]
|
||||
|
||||
let value: string | undefined
|
||||
if (typeof _value === 'object') {
|
||||
if (_value.$type === 'json' && typeof _value.$value === 'object') {
|
||||
if (typeIs(_value, 'object')) {
|
||||
if (_value.$type === 'json' && typeIs(_value.$value, ['object', 'array'])) {
|
||||
value = JSON.stringify(_value.$value, null, 2)
|
||||
} else {
|
||||
value = _value.$value.toString()
|
||||
value = _value.$value?.toString()
|
||||
}
|
||||
} else {
|
||||
value = _value?.toString()
|
||||
}
|
||||
|
||||
const isXrp = typeof _value === 'object' && _value.$type === 'xrp'
|
||||
const isAccount = typeIs(_value, 'object') && _value.$type === 'account'
|
||||
const isXrpAmount = typeIs(_value, 'object') && _value.$type === 'amount.xrp'
|
||||
const isTokenAmount = typeIs(_value, 'object') && _value.$type === 'amount.token'
|
||||
const isJson = typeof _value === 'object' && _value.$type === 'json'
|
||||
const isFee = field === 'Fee'
|
||||
let rows = isJson ? (value?.match(/\n/gm)?.length || 0) + 1 : undefined
|
||||
if (rows && rows > 5) rows = 5
|
||||
let tokenAmount = defaultTokenAmount
|
||||
if (isTokenAmount && typeIs(_value, 'object') && typeIs(_value.$value, 'object')) {
|
||||
tokenAmount = {
|
||||
value: _value.$value.value,
|
||||
currency: _value.$value.currency,
|
||||
issuer: _value.$value.issuer
|
||||
}
|
||||
}
|
||||
|
||||
if (isXrpAmount || isTokenAmount) {
|
||||
return (
|
||||
<Flex column key={field} css={{ mb: '$2', pr: '1px' }}>
|
||||
<TxField key={field} label={field}>
|
||||
<Flex fluid css={{ alignItems: 'center' }}>
|
||||
{isTokenAmount ? (
|
||||
<Flex
|
||||
row
|
||||
fluid
|
||||
row
|
||||
align="center"
|
||||
justify="space-between"
|
||||
css={{ position: 'relative' }}
|
||||
>
|
||||
{/* <Input
|
||||
type="text"
|
||||
placeholder="Issuer"
|
||||
value={tokenAmount.issuer}
|
||||
onChange={e =>
|
||||
setRawField(field, 'amount.token', {
|
||||
...tokenAmount,
|
||||
issuer: e.target.value
|
||||
})
|
||||
}
|
||||
/> */}
|
||||
<Input
|
||||
type="text"
|
||||
value={tokenAmount.currency}
|
||||
placeholder="Currency"
|
||||
onChange={e => {
|
||||
setRawField(field, 'amount.token', {
|
||||
...tokenAmount,
|
||||
currency: e.target.value
|
||||
})
|
||||
}}
|
||||
/>
|
||||
<Input
|
||||
css={{ mx: '$1' }}
|
||||
type="number"
|
||||
value={tokenAmount.value}
|
||||
placeholder="Value"
|
||||
onChange={e => {
|
||||
setRawField(field, 'amount.token', {
|
||||
...tokenAmount,
|
||||
value: e.target.value
|
||||
})
|
||||
}}
|
||||
/>
|
||||
<Box css={{ width: '50%' }}>
|
||||
<CreatableAccount
|
||||
value={tokenAmount.issuer}
|
||||
field={'Issuer' as any}
|
||||
placeholder="Issuer"
|
||||
setField={(_, value = '') => {
|
||||
setRawField(field, 'amount.token', {
|
||||
...tokenAmount,
|
||||
issuer: value
|
||||
})
|
||||
}}
|
||||
/>
|
||||
</Box>
|
||||
</Flex>
|
||||
) : (
|
||||
<Input
|
||||
css={{ flex: 'inherit' }}
|
||||
type="number"
|
||||
value={value}
|
||||
onChange={e => handleSetField(field, e.target.value)}
|
||||
/>
|
||||
)}
|
||||
<Box
|
||||
css={{
|
||||
justifyContent: 'flex-end',
|
||||
alignItems: 'center',
|
||||
position: 'relative'
|
||||
ml: '$2',
|
||||
width: '150px'
|
||||
}}
|
||||
>
|
||||
<Text muted css={{ mr: '$3' }}>
|
||||
{field + (isXrp ? ' (XRP)' : '')}:{' '}
|
||||
</Text>
|
||||
<Select
|
||||
instanceId="currency-type"
|
||||
options={amountOptions}
|
||||
value={isXrpAmount ? amountOptions['0'] : amountOptions['1']}
|
||||
onChange={(e: any) => {
|
||||
const opt = e as typeof amountOptions[number]
|
||||
if (opt.value === 'xrp') {
|
||||
setRawField(field, 'amount.xrp', '0')
|
||||
} else {
|
||||
setRawField(field, 'amount.token', defaultTokenAmount)
|
||||
}
|
||||
}}
|
||||
/>
|
||||
</Box>
|
||||
</Flex>
|
||||
</TxField>
|
||||
)
|
||||
}
|
||||
if (isAccount) {
|
||||
return (
|
||||
<TxField key={field} label={field}>
|
||||
<CreatableAccount value={value} field={field} setField={handleSetField} />
|
||||
</TxField>
|
||||
)
|
||||
}
|
||||
return (
|
||||
<TxField key={field} label={field}>
|
||||
{isJson ? (
|
||||
<Textarea
|
||||
rows={rows}
|
||||
@@ -271,7 +324,6 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
spellCheck={false}
|
||||
onChange={switchToJson}
|
||||
css={{
|
||||
width: '70%',
|
||||
flex: 'inherit',
|
||||
resize: 'vertical'
|
||||
}}
|
||||
@@ -298,7 +350,6 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
: undefined
|
||||
}
|
||||
css={{
|
||||
width: '70%',
|
||||
flex: 'inherit',
|
||||
'-moz-appearance': 'textfield',
|
||||
'&::-webkit-outer-spin-button': {
|
||||
@@ -333,11 +384,200 @@ export const TxUI: FC<UIProps> = ({ state: txState, setState, resetState, estima
|
||||
Suggest
|
||||
</Button>
|
||||
)}
|
||||
</Flex>
|
||||
</Flex>
|
||||
</TxField>
|
||||
)
|
||||
})}
|
||||
<TxField multiLine label="Hook parameters">
|
||||
<Flex column fluid>
|
||||
{Object.entries(hookParameters).map(([id, { label, value }]) => (
|
||||
<Flex column key={id} css={{ mb: '$2' }}>
|
||||
<Flex row>
|
||||
<Input
|
||||
placeholder="Parameter name"
|
||||
value={label}
|
||||
onChange={e => {
|
||||
setState({
|
||||
hookParameters: {
|
||||
...hookParameters,
|
||||
[id]: { label: e.target.value, value }
|
||||
}
|
||||
})
|
||||
}}
|
||||
/>
|
||||
<Input
|
||||
css={{ mx: '$2' }}
|
||||
placeholder="Value (hex-quoted)"
|
||||
value={value}
|
||||
onChange={e => {
|
||||
setState({
|
||||
hookParameters: {
|
||||
...hookParameters,
|
||||
[id]: { label, value: e.target.value }
|
||||
}
|
||||
})
|
||||
}}
|
||||
/>
|
||||
<Button
|
||||
onClick={() => {
|
||||
const { [id]: _, ...rest } = hookParameters
|
||||
setState({ hookParameters: rest })
|
||||
}}
|
||||
variant="destroy"
|
||||
>
|
||||
<Trash weight="regular" size="16px" />
|
||||
</Button>
|
||||
</Flex>
|
||||
</Flex>
|
||||
))}
|
||||
<Button
|
||||
outline
|
||||
fullWidth
|
||||
type="button"
|
||||
onClick={() => {
|
||||
const id = Object.keys(hookParameters).length
|
||||
setState({
|
||||
hookParameters: { ...hookParameters, [id]: { label: '', value: '' } }
|
||||
})
|
||||
}}
|
||||
>
|
||||
<Plus size="16px" />
|
||||
Add Hook Parameter
|
||||
</Button>
|
||||
</Flex>
|
||||
</TxField>
|
||||
<TxField multiLine label="Memos">
|
||||
<Flex column fluid>
|
||||
{Object.entries(memos).map(([id, memo]) => (
|
||||
<Flex column key={id} css={{ mb: '$2' }}>
|
||||
<Flex
|
||||
row
|
||||
css={{
|
||||
flexWrap: 'wrap',
|
||||
width: '100%'
|
||||
}}
|
||||
>
|
||||
<Input
|
||||
placeholder="Memo type"
|
||||
value={memo.type}
|
||||
onChange={e => {
|
||||
setState({
|
||||
memos: {
|
||||
...memos,
|
||||
[id]: { ...memo, type: e.target.value }
|
||||
}
|
||||
})
|
||||
}}
|
||||
/>
|
||||
<Input
|
||||
placeholder="Data (hex-quoted)"
|
||||
css={{ mx: '$2' }}
|
||||
value={memo.data}
|
||||
onChange={e => {
|
||||
setState({
|
||||
memos: {
|
||||
...memos,
|
||||
[id]: { ...memo, data: e.target.value }
|
||||
}
|
||||
})
|
||||
}}
|
||||
/>
|
||||
<Input
|
||||
placeholder="Format"
|
||||
value={memo.format}
|
||||
onChange={e => {
|
||||
setState({
|
||||
memos: {
|
||||
...memos,
|
||||
[id]: { ...memo, format: e.target.value }
|
||||
}
|
||||
})
|
||||
}}
|
||||
/>
|
||||
<Button
|
||||
css={{ ml: '$2' }}
|
||||
onClick={() => {
|
||||
const { [id]: _, ...rest } = memos
|
||||
setState({ memos: rest })
|
||||
}}
|
||||
variant="destroy"
|
||||
>
|
||||
<Trash weight="regular" size="16px" />
|
||||
</Button>
|
||||
</Flex>
|
||||
</Flex>
|
||||
))}
|
||||
<Button
|
||||
outline
|
||||
fullWidth
|
||||
type="button"
|
||||
onClick={() => {
|
||||
const id = Object.keys(memos).length
|
||||
setState({
|
||||
memos: { ...memos, [id]: { data: '', format: '', type: '' } }
|
||||
})
|
||||
}}
|
||||
>
|
||||
<Plus size="16px" />
|
||||
Add Memo
|
||||
</Button>
|
||||
</Flex>
|
||||
</TxField>
|
||||
</Flex>
|
||||
</Container>
|
||||
)
|
||||
}
|
||||
|
||||
export const CreatableAccount: FC<{
|
||||
value: string | undefined
|
||||
field: keyof TxFields
|
||||
placeholder?: string
|
||||
setField: (field: keyof TxFields, value: string, opFields?: TxFields) => void
|
||||
}> = ({ value, field, setField, placeholder }) => {
|
||||
const { accounts } = useSnapshot(state)
|
||||
const accountOptions: SelectOption[] = accounts.map(acc => ({
|
||||
label: acc.name,
|
||||
value: acc.address
|
||||
}))
|
||||
const label = accountOptions.find(a => a.value === value)?.label || value
|
||||
const val = {
|
||||
value,
|
||||
label
|
||||
}
|
||||
placeholder = placeholder || `${capitalize(field)} account`
|
||||
return (
|
||||
<CreatableSelect
|
||||
isClearable
|
||||
instanceId={field}
|
||||
placeholder={placeholder}
|
||||
options={accountOptions}
|
||||
value={value ? val : undefined}
|
||||
onChange={(acc: any) => setField(field, acc?.value)}
|
||||
/>
|
||||
)
|
||||
}
|
||||
|
||||
export const TxField: FC<{ label: string; children: ReactNode; multiLine?: boolean }> = ({
|
||||
label,
|
||||
children,
|
||||
multiLine = false
|
||||
}) => {
|
||||
return (
|
||||
<Flex
|
||||
row
|
||||
fluid
|
||||
css={{
|
||||
justifyContent: 'flex-end',
|
||||
alignItems: multiLine ? 'flex-start' : 'center',
|
||||
position: 'relative',
|
||||
mb: '$2',
|
||||
mt: '1px',
|
||||
pr: '1px'
|
||||
}}
|
||||
>
|
||||
<Text muted css={{ mr: '$3', mt: multiLine ? '$2' : 0 }}>
|
||||
{label}:{' '}
|
||||
</Text>
|
||||
<Flex css={{ width: '70%', alignItems: 'center' }}>{children}</Flex>
|
||||
</Flex>
|
||||
)
|
||||
}
|
||||
|
||||
@@ -2,7 +2,10 @@
|
||||
{
|
||||
"TransactionType": "AccountDelete",
|
||||
"Account": "rWYkbWkCeg8dP6rXALnjgZSjjLyih5NXm",
|
||||
"Destination": "rPT1Sjq2YGrBMTttX4GZHjKu9dyfzbpAYe",
|
||||
"Destination": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"DestinationTag": 13,
|
||||
"Fee": "2000000",
|
||||
"Sequence": 2470665,
|
||||
@@ -28,7 +31,11 @@
|
||||
"TransactionType": "CheckCash",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"DeliverMin": {
|
||||
"$value": "",
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"CheckID": "838766BA2B995C00744175F69A1B11E32C3DBC40E64801A4056FCBD657F57334",
|
||||
"Fee": "12"
|
||||
@@ -36,7 +43,10 @@
|
||||
{
|
||||
"TransactionType": "CheckCreate",
|
||||
"Account": "rUn84CUYbNjRoTQ6mSW7BVJPSVJNLb1QLo",
|
||||
"Destination": "rfkE1aSy9G8Upk4JssnwBxhEv5p4mn2KTy",
|
||||
"Destination": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"SendMax": "100000000",
|
||||
"Expiration": 570113521,
|
||||
"InvoiceID": "6F1DFD1D0FE8A32E40E1F2C05CF1C15545BAB56B617F9C6C2D63A6B704BEF59B",
|
||||
@@ -54,7 +64,10 @@
|
||||
{
|
||||
"Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||
"TransactionType": "EscrowCancel",
|
||||
"Owner": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||
"Owner": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"OfferSequence": 7,
|
||||
"Fee": "10"
|
||||
},
|
||||
@@ -63,9 +76,12 @@
|
||||
"TransactionType": "EscrowCreate",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"Destination": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"Destination": "rsA2LpzuawewSBQXkiju3YQTMzW13pAAdW",
|
||||
"CancelAfter": 533257958,
|
||||
"FinishAfter": 533171558,
|
||||
"Condition": "A0258020E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855810100",
|
||||
@@ -76,56 +92,15 @@
|
||||
{
|
||||
"Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||
"TransactionType": "EscrowFinish",
|
||||
"Owner": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||
"Owner": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"OfferSequence": 7,
|
||||
"Condition": "A0258020E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855810100",
|
||||
"Fulfillment": "A0028000",
|
||||
"Fee": "10"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenMint",
|
||||
"Account": "rvYAfWj5gh67oV6fW32ZzP3Aw4Eubs59B",
|
||||
"Fee": "10",
|
||||
"NFTokenTaxon": 0,
|
||||
"URI": "697066733A2F2F516D614374444B5A4656767666756676626479346573745A626851483744586831364354707631686F776D424779"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenBurn",
|
||||
"Account": "rvYAfWj5gh67oV6fW32ZzP3Aw4Eubs59B",
|
||||
"Fee": "10",
|
||||
"NFTokenID": "000B013A95F14B0044F78A264E41713C64B5F89242540EE208C3098E00000D65"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenAcceptOffer",
|
||||
"Account": "rvYAfWj5gh67oV6fW32ZzP3Aw4Eubs59B",
|
||||
"Fee": "10",
|
||||
"NFTokenSellOffer": "A2FA1A9911FE2AEF83DAB05F437768E26A301EF899BD31EB85E704B3D528FF18",
|
||||
"NFTokenBuyOffer": "4AAAEEA76E3C8148473CB3840CE637676E561FB02BD4CA22CA59729EA815B862",
|
||||
"NFTokenBrokerFee": "10"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenCancelOffer",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"Fee": "10",
|
||||
"NFTokenOffers": {
|
||||
"$type": "json",
|
||||
"$value": [
|
||||
"4AAAEEA76E3C8148473CB3840CE637676E561FB02BD4CA22CA59729EA815B862"
|
||||
]
|
||||
}
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenCreateOffer",
|
||||
"Account": "rs8jBmmfpwgmrSPgwMsh7CvKRmRt1JTVSX",
|
||||
"NFTokenID": "000100001E962F495F07A990F4ED55ACCFEEF365DBAA76B6A048C0A200000007",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
},
|
||||
"Flags": "1",
|
||||
"Destination": "",
|
||||
"Fee": "10"
|
||||
},
|
||||
{
|
||||
"TransactionType": "OfferCancel",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
@@ -142,19 +117,29 @@
|
||||
"Flags": "0",
|
||||
"LastLedgerSequence": 7108682,
|
||||
"Sequence": 8,
|
||||
"TakerGets": "6000000",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
"TakerGets": {
|
||||
"$type": "amount.xrp",
|
||||
"$value": "6000000"
|
||||
},
|
||||
"TakerPays": {
|
||||
"$type": "amount.token",
|
||||
"$value": {
|
||||
"currency": "GKO",
|
||||
"issuer": "ruazs5h1qEsqpke88pcqnaseXdm6od2xc",
|
||||
"value": "2"
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"TransactionType": "Payment",
|
||||
"Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||
"Destination": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"Destination": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"Fee": "12",
|
||||
"Flags": "2147483648",
|
||||
@@ -165,9 +150,12 @@
|
||||
"TransactionType": "PaymentChannelCreate",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"Destination": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"Destination": "rsA2LpzuawewSBQXkiju3YQTMzW13pAAdW",
|
||||
"SettleDelay": 86400,
|
||||
"PublicKey": "32D2471DB72B27E3310F355BB33E339BF26F8392D5A93D3BC0FC3B566612DA0F0A",
|
||||
"CancelAfter": 533171558,
|
||||
@@ -181,7 +169,7 @@
|
||||
"Channel": "C1AE6DDDEEC05CF2978C0BAD6FE302948E9533691DC749DCDD3B9E5992CA6198",
|
||||
"Amount": {
|
||||
"$value": "200",
|
||||
"$type": "xrp"
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"Expiration": 543171558,
|
||||
"Fee": "10"
|
||||
@@ -237,7 +225,7 @@
|
||||
"Flags": "262144",
|
||||
"LastLedgerSequence": 8007750,
|
||||
"LimitAmount": {
|
||||
"$type": "json",
|
||||
"$type": "amount.token",
|
||||
"$value": {
|
||||
"currency": "USD",
|
||||
"issuer": "rsP3mgGb2tcYUrxiLFiHJiQXhsziegtwBc",
|
||||
@@ -248,11 +236,54 @@
|
||||
},
|
||||
{
|
||||
"TransactionType": "Invoke",
|
||||
"Destination": {
|
||||
"$type": "account",
|
||||
"$value": ""
|
||||
},
|
||||
"Fee": "12"
|
||||
},
|
||||
{
|
||||
"TransactionType": "UriToken",
|
||||
"Fee": "12",
|
||||
"URI": "697066733A2F2F516D614374444B5A4656767666756676626479346573745A626851483744586831364354707631686F776D424779"
|
||||
"TransactionType": "URITokenMint",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"URI": "697066733A2F2F434944",
|
||||
"Fee": "10",
|
||||
"Sequence": 1
|
||||
},
|
||||
{
|
||||
"TransactionType": "URITokenBurn",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"URITokenID": "B792B56B558C89C4E942E41B5DB66074E005CB198DB70C26C707AAC2FF5F74CB",
|
||||
"Fee": "10",
|
||||
"Sequence": 1
|
||||
},
|
||||
{
|
||||
"TransactionType": "URITokenCreateSellOffer",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"URITokenID": "B792B56B558C89C4E942E41B5DB66074E005CB198DB70C26C707AAC2FF5F74CB",
|
||||
"Destination": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"Fee": "10",
|
||||
"Sequence": 1
|
||||
},
|
||||
{
|
||||
"TransactionType": "URITokenCancelSellOffer",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"URITokenID": "B792B56B558C89C4E942E41B5DB66074E005CB198DB70C26C707AAC2FF5F74CB",
|
||||
"Fee": "10",
|
||||
"Sequence": 1
|
||||
},
|
||||
{
|
||||
"TransactionType": "URITokenBuy",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"URITokenID": "B792B56B558C89C4E942E41B5DB66074E005CB198DB70C26C707AAC2FF5F74CB",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "amount.xrp"
|
||||
},
|
||||
"Fee": "10",
|
||||
"Sequence": 1
|
||||
}
|
||||
]
|
||||
@@ -83,6 +83,6 @@
|
||||
"typescript": "4.4.4"
|
||||
},
|
||||
"resolutions": {
|
||||
"ripple-binary-codec": "=1.4.2"
|
||||
"ripple-binary-codec": "=1.6.0"
|
||||
}
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
346
patches/ripple-binary-codec+1.6.0.patch
Normal file
346
patches/ripple-binary-codec+1.6.0.patch
Normal file
@@ -0,0 +1,346 @@
|
||||
diff --git a/node_modules/ripple-binary-codec/dist/enums/definitions.json b/node_modules/ripple-binary-codec/dist/enums/definitions.json
|
||||
index e623376..7e1e4d5 100644
|
||||
--- a/node_modules/ripple-binary-codec/dist/enums/definitions.json
|
||||
+++ b/node_modules/ripple-binary-codec/dist/enums/definitions.json
|
||||
@@ -44,11 +44,16 @@
|
||||
"NegativeUNL": 78,
|
||||
"NFTokenPage": 80,
|
||||
"NFTokenOffer": 55,
|
||||
+ "URIToken": 85,
|
||||
"Any": -3,
|
||||
"Child": -2,
|
||||
"Nickname": 110,
|
||||
"Contract": 99,
|
||||
- "GeneratorMap": 103
|
||||
+ "GeneratorMap": 103,
|
||||
+ "Hook": 72,
|
||||
+ "HookState": 118,
|
||||
+ "HookDefinition": 68,
|
||||
+ "EmittedTxn": 69
|
||||
},
|
||||
"FIELDS": [
|
||||
[
|
||||
@@ -321,6 +326,16 @@
|
||||
"type": "UInt16"
|
||||
}
|
||||
],
|
||||
+ [
|
||||
+ "NetworkID",
|
||||
+ {
|
||||
+ "nth": 1,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "UInt32"
|
||||
+ }
|
||||
+ ],
|
||||
[
|
||||
"Flags",
|
||||
{
|
||||
@@ -761,6 +776,36 @@
|
||||
"type": "UInt32"
|
||||
}
|
||||
],
|
||||
+ [
|
||||
+ "LockCount",
|
||||
+ {
|
||||
+ "nth": 49,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "UInt32"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "FirstNFTokenSequence",
|
||||
+ {
|
||||
+ "nth": 50,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "UInt32"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "ImportSequence",
|
||||
+ {
|
||||
+ "nth": 97,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "UInt32"
|
||||
+ }
|
||||
+ ],
|
||||
[
|
||||
"IndexNext",
|
||||
{
|
||||
@@ -891,16 +936,6 @@
|
||||
"type": "UInt64"
|
||||
}
|
||||
],
|
||||
- [
|
||||
- "HookOn",
|
||||
- {
|
||||
- "nth": 16,
|
||||
- "isVLEncoded": false,
|
||||
- "isSerialized": true,
|
||||
- "isSigningField": true,
|
||||
- "type": "UInt64"
|
||||
- }
|
||||
- ],
|
||||
[
|
||||
"HookInstructionCount",
|
||||
{
|
||||
@@ -1151,6 +1186,16 @@
|
||||
"type": "Hash256"
|
||||
}
|
||||
],
|
||||
+ [
|
||||
+ "HookOn",
|
||||
+ {
|
||||
+ "nth": 20,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Hash256"
|
||||
+ }
|
||||
+ ],
|
||||
[
|
||||
"Digest",
|
||||
{
|
||||
@@ -1281,6 +1326,36 @@
|
||||
"type": "Hash256"
|
||||
}
|
||||
],
|
||||
+ [
|
||||
+ "OfferID",
|
||||
+ {
|
||||
+ "nth": 34,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Hash256"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "EscrowID",
|
||||
+ {
|
||||
+ "nth": 35,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Hash256"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "URITokenID",
|
||||
+ {
|
||||
+ "nth": 36,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Hash256"
|
||||
+ }
|
||||
+ ],
|
||||
[
|
||||
"Amount",
|
||||
{
|
||||
@@ -1421,6 +1496,56 @@
|
||||
"type": "Amount"
|
||||
}
|
||||
],
|
||||
+ [
|
||||
+ "HookCallbackFee",
|
||||
+ {
|
||||
+ "nth": 20,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Amount"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "LockedBalance",
|
||||
+ {
|
||||
+ "nth": 21,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Amount"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "BaseFeeDrops",
|
||||
+ {
|
||||
+ "nth": 22,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Amount"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "ReserveBaseDrops",
|
||||
+ {
|
||||
+ "nth": 23,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Amount"
|
||||
+ }
|
||||
+ ],
|
||||
+ [
|
||||
+ "ReserveIncrementDrops",
|
||||
+ {
|
||||
+ "nth": 24,
|
||||
+ "isVLEncoded": false,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Amount"
|
||||
+ }
|
||||
+ ],
|
||||
[
|
||||
"PublicKey",
|
||||
{
|
||||
@@ -1661,6 +1786,16 @@
|
||||
"type": "Blob"
|
||||
}
|
||||
],
|
||||
+ [
|
||||
+ "Blob",
|
||||
+ {
|
||||
+ "nth": 26,
|
||||
+ "isVLEncoded": true,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Blob"
|
||||
+ }
|
||||
+ ],
|
||||
[
|
||||
"Account",
|
||||
{
|
||||
@@ -1801,6 +1936,16 @@
|
||||
"type": "Vector256"
|
||||
}
|
||||
],
|
||||
+ [
|
||||
+ "HookNamespaces",
|
||||
+ {
|
||||
+ "nth": 5,
|
||||
+ "isVLEncoded": true,
|
||||
+ "isSerialized": true,
|
||||
+ "isSigningField": true,
|
||||
+ "type": "Vector256"
|
||||
+ }
|
||||
+ ],
|
||||
[
|
||||
"Paths",
|
||||
{
|
||||
@@ -2176,6 +2321,12 @@
|
||||
"telCAN_NOT_QUEUE_BLOCKED": -389,
|
||||
"telCAN_NOT_QUEUE_FEE": -388,
|
||||
"telCAN_NOT_QUEUE_FULL": -387,
|
||||
+ "telWRONG_NETWORK": -386,
|
||||
+ "telREQUIRES_NETWORK_ID": -385,
|
||||
+ "telNETWORK_ID_MAKES_TX_NON_CANONICAL": -384,
|
||||
+ "telNON_LOCAL_EMITTED_TXN": -383,
|
||||
+ "telIMPORT_VL_KEY_NOT_RECOGNISED": -382,
|
||||
+ "telCAN_NOT_QUEUE_IMPORT": -381,
|
||||
"temMALFORMED": -299,
|
||||
"temBAD_AMOUNT": -298,
|
||||
"temBAD_CURRENCY": -297,
|
||||
@@ -2214,6 +2365,16 @@
|
||||
"temUNKNOWN": -264,
|
||||
"temSEQ_AND_TICKET": -263,
|
||||
"temBAD_NFTOKEN_TRANSFER_FEE": -262,
|
||||
+ "temAMM_BAD_TOKENS": -261,
|
||||
+ "temXCHAIN_EQUAL_DOOR_ACCOUNTS": -260,
|
||||
+ "temXCHAIN_BAD_PROOF": -259,
|
||||
+ "temXCHAIN_BRIDGE_BAD_ISSUES": -258,
|
||||
+ "temXCHAIN_BRIDGE_NONDOOR_OWNER": -257,
|
||||
+ "temXCHAIN_BRIDGE_BAD_MIN_ACCOUNT_CREATE_AMOUNT": -256,
|
||||
+ "temXCHAIN_BRIDGE_BAD_REWARD_AMOUNT": -255,
|
||||
+ "temXCHAIN_TOO_MANY_ATTESTATIONS": -254,
|
||||
+ "temHOOK_DATA_TOO_LARGE": -253,
|
||||
+ "temHOOK_REJECTED": -252,
|
||||
"tefFAILURE": -199,
|
||||
"tefALREADY": -198,
|
||||
"tefBAD_ADD_AUTH": -197,
|
||||
@@ -2235,6 +2396,7 @@
|
||||
"tefTOO_BIG": -181,
|
||||
"tefNO_TICKET": -180,
|
||||
"tefNFTOKEN_IS_NOT_TRANSFERABLE": -179,
|
||||
+ "tefPAST_IMPORT_SEQ": -178,
|
||||
"terRETRY": -99,
|
||||
"terFUNDS_SPENT": -98,
|
||||
"terINSUF_FEE_B": -97,
|
||||
@@ -2247,6 +2409,8 @@
|
||||
"terNO_RIPPLE": -90,
|
||||
"terQUEUED": -89,
|
||||
"terPRE_TICKET": -88,
|
||||
+ "terNO_AMM": -87,
|
||||
+ "terNO_HOOK": -86,
|
||||
"tesSUCCESS": 0,
|
||||
"tecCLAIM": 100,
|
||||
"tecPATH_PARTIAL": 101,
|
||||
@@ -2286,6 +2450,7 @@
|
||||
"tecKILLED": 150,
|
||||
"tecHAS_OBLIGATIONS": 151,
|
||||
"tecTOO_SOON": 152,
|
||||
+ "tecHOOK_REJECTED": 153,
|
||||
"tecMAX_SEQUENCE_REACHED": 154,
|
||||
"tecNO_SUITABLE_NFTOKEN_PAGE": 155,
|
||||
"tecNFTOKEN_BUY_SELL_MISMATCH": 156,
|
||||
@@ -2293,7 +2458,33 @@
|
||||
"tecCANT_ACCEPT_OWN_NFTOKEN_OFFER": 158,
|
||||
"tecINSUFFICIENT_FUNDS": 159,
|
||||
"tecOBJECT_NOT_FOUND": 160,
|
||||
- "tecINSUFFICIENT_PAYMENT": 161
|
||||
+ "tecINSUFFICIENT_PAYMENT": 161,
|
||||
+ "tecAMM_UNFUNDED": 162,
|
||||
+ "tecAMM_BALANCE": 163,
|
||||
+ "tecAMM_FAILED_DEPOSIT": 164,
|
||||
+ "tecAMM_FAILED_WITHDRAW": 165,
|
||||
+ "tecAMM_INVALID_TOKENS": 166,
|
||||
+ "tecAMM_FAILED_BID": 167,
|
||||
+ "tecAMM_FAILED_VOTE": 168,
|
||||
+ "tecREQUIRES_FLAG": 169,
|
||||
+ "tecPRECISION_LOSS": 170,
|
||||
+ "tecBAD_XCHAIN_TRANSFER_ISSUE": 171,
|
||||
+ "tecXCHAIN_NO_CLAIM_ID": 172,
|
||||
+ "tecXCHAIN_BAD_CLAIM_ID": 173,
|
||||
+ "tecXCHAIN_CLAIM_NO_QUORUM": 174,
|
||||
+ "tecXCHAIN_PROOF_UNKNOWN_KEY": 175,
|
||||
+ "tecXCHAIN_CREATE_ACCOUNT_NONXRP_ISSUE": 176,
|
||||
+ "tecXCHAIN_WRONG_CHAIN": 177,
|
||||
+ "tecXCHAIN_REWARD_MISMATCH": 178,
|
||||
+ "tecXCHAIN_NO_SIGNERS_LIST": 179,
|
||||
+ "tecXCHAIN_SENDING_ACCOUNT_MISMATCH": 180,
|
||||
+ "tecXCHAIN_INSUFF_CREATE_AMOUNT": 181,
|
||||
+ "tecXCHAIN_ACCOUNT_CREATE_PAST": 182,
|
||||
+ "tecXCHAIN_ACCOUNT_CREATE_TOO_MANY": 183,
|
||||
+ "tecXCHAIN_PAYMENT_FAILED": 184,
|
||||
+ "tecXCHAIN_SELF_COMMIT": 185,
|
||||
+ "tecXCHAIN_BAD_PUBLIC_KEY_ACCOUNT_PAIR": 186,
|
||||
+ "tecLAST_POSSIBLE_ENTRY": 255
|
||||
},
|
||||
"TRANSACTION_TYPES": {
|
||||
"Invalid": -1,
|
||||
@@ -2325,8 +2516,16 @@
|
||||
"NFTokenCreateOffer": 27,
|
||||
"NFTokenCancelOffer": 28,
|
||||
"NFTokenAcceptOffer": 29,
|
||||
+ "URITokenMint": 45,
|
||||
+ "URITokenBurn": 46,
|
||||
+ "URITokenBuy": 47,
|
||||
+ "URITokenCreateSellOffer": 48,
|
||||
+ "URITokenCancelSellOffer": 49,
|
||||
+ "Import": 97,
|
||||
+ "Invoke": 99,
|
||||
"EnableAmendment": 100,
|
||||
"SetFee": 101,
|
||||
- "UNLModify": 102
|
||||
+ "UNLModify": 102,
|
||||
+ "EmitFailure": 103
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
import toast from 'react-hot-toast'
|
||||
import state, { FaucetAccountRes } from '../index'
|
||||
import fetchAccountInfo from '../../utils/accountInfo';
|
||||
|
||||
export const names = [
|
||||
'Alice',
|
||||
@@ -35,26 +36,23 @@ export const addFaucetAccount = async (name?: string, showToast: boolean = false
|
||||
})
|
||||
const json: FaucetAccountRes | { error: string } = await res.json()
|
||||
if ('error' in json) {
|
||||
if (showToast) {
|
||||
if (!showToast) return;
|
||||
return toast.error(json.error, { id: toastId })
|
||||
} else {
|
||||
return
|
||||
}
|
||||
} else {
|
||||
if (showToast) {
|
||||
toast.success('New account created', { id: toastId })
|
||||
}
|
||||
const currNames = state.accounts.map(acc => acc.name)
|
||||
const info = await fetchAccountInfo(json.address, { silent: true })
|
||||
state.accounts.push({
|
||||
name: name || names.filter(name => !currNames.includes(name))[0],
|
||||
xrp: (json.xrp || 0 * 1000000).toString(),
|
||||
address: json.address,
|
||||
secret: json.secret,
|
||||
sequence: 1,
|
||||
sequence: info?.Sequence || 1,
|
||||
hooks: [],
|
||||
isLoading: false,
|
||||
version: '2'
|
||||
})
|
||||
if (showToast) {
|
||||
toast.success('New account created', { id: toastId })
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -14,11 +14,4 @@ export const deleteAccount = (addr?: string) => {
|
||||
if (!acc) return
|
||||
acc.label = acc.value
|
||||
})
|
||||
transactionsState.transactions
|
||||
.filter(t => t.state.selectedDestAccount?.value === addr)
|
||||
.forEach(t => {
|
||||
const acc = t.state.selectedDestAccount
|
||||
if (!acc) return
|
||||
acc.label = acc.value
|
||||
})
|
||||
}
|
||||
|
||||
@@ -6,7 +6,7 @@ import calculateHookOn, { TTS } from '../../utils/hookOnCalculator'
|
||||
import { Link } from '../../components'
|
||||
import { ref } from 'valtio'
|
||||
import estimateFee from '../../utils/estimateFee'
|
||||
import { SetHookData } from '../../utils/setHook'
|
||||
import { SetHookData, toHex } from '../../utils/setHook'
|
||||
import ResultLink from '../../components/ResultLink'
|
||||
import { xrplSend } from './xrpl-client'
|
||||
|
||||
@@ -18,13 +18,6 @@ export const sha256 = async (string: string) => {
|
||||
return hashHex
|
||||
}
|
||||
|
||||
function toHex(str: string) {
|
||||
var result = ''
|
||||
for (var i = 0; i < str.length; i++) {
|
||||
result += str.charCodeAt(i).toString(16)
|
||||
}
|
||||
return result.toUpperCase()
|
||||
}
|
||||
|
||||
function arrayBufferToHex(arrayBuffer?: ArrayBuffer | null) {
|
||||
if (!arrayBuffer) {
|
||||
|
||||
@@ -10,7 +10,6 @@ interface TransactionOptions {
|
||||
TransactionType: string
|
||||
Account?: string
|
||||
Fee?: string
|
||||
Destination?: string
|
||||
[index: string]: any
|
||||
}
|
||||
interface OtherOptions {
|
||||
@@ -31,6 +30,10 @@ export const sendTransaction = async (
|
||||
...opts
|
||||
}
|
||||
const { logPrefix = '' } = options || {}
|
||||
state.transactionLogs.push({
|
||||
type: 'log',
|
||||
message: `${logPrefix}${JSON.stringify(tx, null, 2)}`
|
||||
})
|
||||
try {
|
||||
const signedAccount = derive.familySeed(account.secret)
|
||||
const { signedTransaction } = sign(tx, signedAccount)
|
||||
|
||||
@@ -47,6 +47,9 @@ export const transactionFlags: { [key: /* TransactionType */ string]: Flags } =
|
||||
tfSetFreeze: '0x00100000',
|
||||
tfClearFreeze: '0x00200000',
|
||||
},
|
||||
URITokenMint: {
|
||||
tfBurnable: '0x00000001',
|
||||
},
|
||||
}
|
||||
|
||||
export const getFlags = (tt?: string) => {
|
||||
|
||||
@@ -5,35 +5,55 @@ import state from '.'
|
||||
import { showAlert } from '../state/actions/showAlert'
|
||||
import { parseJSON } from '../utils/json'
|
||||
import { extractFlags, getFlags } from './constants/flags'
|
||||
import { fromHex } from '../utils/setHook'
|
||||
import { typeIs } from '../utils/helpers'
|
||||
|
||||
export type SelectOption = {
|
||||
value: string
|
||||
label: string
|
||||
}
|
||||
|
||||
export type HookParameters = {
|
||||
[key: string]: SelectOption
|
||||
}
|
||||
|
||||
export type Memos = {
|
||||
[key: string]: {
|
||||
type: string
|
||||
format: string
|
||||
data: string
|
||||
}
|
||||
}
|
||||
|
||||
export interface TransactionState {
|
||||
selectedTransaction: SelectOption | null
|
||||
selectedAccount: SelectOption | null
|
||||
selectedDestAccount: SelectOption | null
|
||||
selectedFlags: SelectOption[] | null
|
||||
hookParameters: HookParameters
|
||||
memos: Memos
|
||||
txIsLoading: boolean
|
||||
txIsDisabled: boolean
|
||||
txFields: TxFields
|
||||
viewType: 'json' | 'ui'
|
||||
editorValue?: string
|
||||
editorIsSaved: boolean
|
||||
estimatedFee?: string
|
||||
}
|
||||
|
||||
const commonFields = ['TransactionType', 'Account', 'Sequence', "HookParameters"] as const;
|
||||
|
||||
export type TxFields = Omit<
|
||||
Partial<typeof transactionsData[0]>,
|
||||
'Account' | 'Sequence' | 'TransactionType'
|
||||
typeof commonFields[number]
|
||||
>
|
||||
|
||||
export const defaultTransaction: TransactionState = {
|
||||
selectedTransaction: null,
|
||||
selectedAccount: null,
|
||||
selectedDestAccount: null,
|
||||
selectedFlags: null,
|
||||
hookParameters: {},
|
||||
memos: {},
|
||||
editorIsSaved: true,
|
||||
txIsLoading: false,
|
||||
txIsDisabled: false,
|
||||
txFields: {},
|
||||
@@ -109,46 +129,51 @@ export const modifyTxState = (
|
||||
return tx.state
|
||||
}
|
||||
|
||||
// state to tx options
|
||||
export const prepareTransaction = (data: any) => {
|
||||
let options = { ...data }
|
||||
|
||||
Object.keys(options).forEach(field => {
|
||||
let _value = options[field]
|
||||
// convert xrp
|
||||
if (_value && typeof _value === 'object' && _value.$type === 'xrp') {
|
||||
if (+_value.$value) {
|
||||
options[field] = (+_value.$value * 1000000 + '') as any
|
||||
if (!typeIs(_value, 'object')) return
|
||||
// amount.xrp
|
||||
if (_value.$type === 'amount.xrp') {
|
||||
if (_value.$value) {
|
||||
options[field] = (+(_value as any).$value * 1000000 + '')
|
||||
} else {
|
||||
options[field] = undefined // 👇 💀
|
||||
options[field] = ""
|
||||
}
|
||||
}
|
||||
// handle type: `json`
|
||||
if (_value && typeof _value === 'object' && _value.$type === 'json') {
|
||||
if (typeof _value.$value === 'object') {
|
||||
// amount.token
|
||||
if (_value.$type === 'amount.token') {
|
||||
if (typeIs(_value.$value, 'string')) {
|
||||
options[field] = parseJSON(_value.$value)
|
||||
} else if (typeIs(_value.$value, 'object')) {
|
||||
options[field] = _value.$value
|
||||
} else {
|
||||
try {
|
||||
options[field] = JSON.parse(_value.$value)
|
||||
} catch (error) {
|
||||
const message = `Input error for json field '${field}': ${error instanceof Error ? error.message : ''
|
||||
}`
|
||||
console.error(message)
|
||||
options[field] = _value.$value
|
||||
options[field] = undefined
|
||||
}
|
||||
}
|
||||
// account
|
||||
if (_value.$type === 'account') {
|
||||
options[field] = (_value.$value as any)?.toString() || ""
|
||||
}
|
||||
// json
|
||||
if (_value.$type === 'json') {
|
||||
const val = _value.$value;
|
||||
let res: any = val;
|
||||
if (typeIs(val, ["object", "array"])) {
|
||||
options[field] = res
|
||||
} else if (typeIs(val, "string") && (res = parseJSON(val))) {
|
||||
options[field] = res;
|
||||
} else {
|
||||
options[field] = res;
|
||||
}
|
||||
|
||||
// delete unnecessary fields
|
||||
if (!options[field]) {
|
||||
delete options[field]
|
||||
}
|
||||
})
|
||||
|
||||
return options
|
||||
}
|
||||
|
||||
// editor value to state
|
||||
export const prepareState = (value: string, transactionType?: string) => {
|
||||
const options = parseJSON(value)
|
||||
if (!options) {
|
||||
@@ -158,7 +183,7 @@ export const prepareState = (value: string, transactionType?: string) => {
|
||||
return
|
||||
}
|
||||
|
||||
const { Account, TransactionType, Destination, ...rest } = options
|
||||
const { Account, TransactionType, HookParameters, Memos, ...rest } = options
|
||||
let tx: Partial<TransactionState> = {}
|
||||
const schema = getTxFields(transactionType)
|
||||
|
||||
@@ -188,25 +213,23 @@ export const prepareState = (value: string, transactionType?: string) => {
|
||||
tx.selectedTransaction = null
|
||||
}
|
||||
|
||||
if (schema.Destination !== undefined) {
|
||||
const dest = state.accounts.find(acc => acc.address === Destination)
|
||||
if (dest) {
|
||||
tx.selectedDestAccount = {
|
||||
label: dest.name,
|
||||
value: dest.address
|
||||
if (HookParameters && HookParameters instanceof Array) {
|
||||
tx.hookParameters = HookParameters.reduce<TransactionState["hookParameters"]>((acc, cur, idx) => {
|
||||
const param = { label: fromHex(cur.HookParameter?.HookParameterName || ""), value: cur.HookParameter?.HookParameterValue || "" }
|
||||
acc[idx] = param;
|
||||
return acc;
|
||||
}, {})
|
||||
}
|
||||
} else if (Destination) {
|
||||
tx.selectedDestAccount = {
|
||||
label: Destination,
|
||||
value: Destination
|
||||
}
|
||||
} else {
|
||||
tx.selectedDestAccount = null
|
||||
}
|
||||
} else if (Destination) {
|
||||
rest.Destination = Destination
|
||||
|
||||
if (Memos && Memos instanceof Array) {
|
||||
tx.memos = Memos.reduce<TransactionState["memos"]>((acc, cur, idx) => {
|
||||
const memo = { data: cur.Memo?.MemoData || "", type: fromHex(cur.Memo?.MemoType || ""), format: fromHex(cur.Memo?.MemoFormat || "") }
|
||||
acc[idx] = memo;
|
||||
return acc;
|
||||
}, {})
|
||||
}
|
||||
|
||||
|
||||
if (getFlags(TransactionType) && rest.Flags) {
|
||||
const flags = extractFlags(TransactionType, rest.Flags)
|
||||
|
||||
@@ -217,17 +240,31 @@ export const prepareState = (value: string, transactionType?: string) => {
|
||||
Object.keys(rest).forEach(field => {
|
||||
const value = rest[field]
|
||||
const schemaVal = schema[field as keyof TxFields]
|
||||
const isXrp =
|
||||
typeof value !== 'object' &&
|
||||
schemaVal &&
|
||||
typeof schemaVal === 'object' &&
|
||||
schemaVal.$type === 'xrp'
|
||||
if (isXrp) {
|
||||
|
||||
const isAmount = schemaVal &&
|
||||
typeIs(schemaVal, "object") &&
|
||||
schemaVal.$type.startsWith('amount.');
|
||||
const isAccount = schemaVal &&
|
||||
typeIs(schemaVal, "object") &&
|
||||
schemaVal.$type.startsWith("account");
|
||||
|
||||
if (isAmount && ["number", "string"].includes(typeof value)) {
|
||||
rest[field] = {
|
||||
$type: 'xrp',
|
||||
$type: 'amount.xrp', // TODO narrow typed $type.
|
||||
$value: +value / 1000000 // ! maybe use bigint?
|
||||
}
|
||||
} else if (typeof value === 'object') {
|
||||
} else if (isAmount && typeof value === 'object') {
|
||||
rest[field] = {
|
||||
$type: 'amount.token',
|
||||
$value: value
|
||||
}
|
||||
} else if (isAccount) {
|
||||
rest[field] = {
|
||||
$type: "account",
|
||||
$value: value?.toString() || ""
|
||||
}
|
||||
}
|
||||
else if (typeof value === 'object') {
|
||||
rest[field] = {
|
||||
$type: 'json',
|
||||
$value: value
|
||||
@@ -236,6 +273,7 @@ export const prepareState = (value: string, transactionType?: string) => {
|
||||
})
|
||||
|
||||
tx.txFields = rest
|
||||
tx.editorIsSaved = true;
|
||||
|
||||
return tx
|
||||
}
|
||||
@@ -246,12 +284,12 @@ export const getTxFields = (tt?: string) => {
|
||||
if (!txFields) return {}
|
||||
|
||||
let _txFields = Object.keys(txFields)
|
||||
.filter(key => !['TransactionType', 'Account', 'Sequence'].includes(key))
|
||||
.filter(key => !commonFields.includes(key as any))
|
||||
.reduce<TxFields>((tf, key) => ((tf[key as keyof TxFields] = (txFields as any)[key]), tf), {})
|
||||
return _txFields
|
||||
}
|
||||
|
||||
export { transactionsData }
|
||||
export { transactionsData, commonFields }
|
||||
|
||||
export const transactionsOptions = transactionsData.map(tx => ({
|
||||
value: tx.TransactionType,
|
||||
|
||||
31
utils/accountInfo.ts
Normal file
31
utils/accountInfo.ts
Normal file
@@ -0,0 +1,31 @@
|
||||
import toast from 'react-hot-toast'
|
||||
|
||||
import { xrplSend } from '../state/actions/xrpl-client'
|
||||
|
||||
interface AccountInfo {
|
||||
Account: string,
|
||||
Sequence: number,
|
||||
Flags: number,
|
||||
Balance?: string,
|
||||
}
|
||||
|
||||
const fetchAccountInfo = async (
|
||||
address: string,
|
||||
opts: { silent?: boolean } = {}
|
||||
): Promise<AccountInfo | undefined> => {
|
||||
try {
|
||||
const res = await xrplSend({
|
||||
id: `hooks-builder-req-info-${address}`,
|
||||
command: 'account_info',
|
||||
account: address
|
||||
})
|
||||
return res.account_data;
|
||||
} catch (err) {
|
||||
if (!opts.silent) {
|
||||
console.error(err)
|
||||
toast.error('Could not fetch account info!')
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export default fetchAccountInfo
|
||||
@@ -24,6 +24,9 @@ const estimateFee = async (
|
||||
const { signedTransaction } = sign(copyTx, keypair)
|
||||
|
||||
const res = await xrplSend({ command: 'fee', tx_blob: signedTransaction })
|
||||
if (res.error) {
|
||||
throw new Error(`[${res.error}] ${res.error_exception}.`);
|
||||
}
|
||||
if (res && res.drops) {
|
||||
return res.drops
|
||||
}
|
||||
@@ -31,7 +34,8 @@ const estimateFee = async (
|
||||
} catch (err) {
|
||||
if (!opts.silent) {
|
||||
console.error(err)
|
||||
toast.error('Cannot estimate fee.') // ? Some better msg
|
||||
const msg = err instanceof Error ? err.message : 'Error estimating fee!';
|
||||
toast.error(msg);
|
||||
}
|
||||
return null
|
||||
}
|
||||
|
||||
@@ -19,3 +19,17 @@ export const getFileExtention = (filename?: string): string | undefined => {
|
||||
const ext = (filename.includes('.') && filename.split('.').pop()) || undefined
|
||||
return ext
|
||||
}
|
||||
|
||||
type Type = "array" | "undefined" | "object" | "string" | "number" | "bigint" | "boolean" | "symbol" | "function"
|
||||
type obj = Record<string | number | symbol, unknown>
|
||||
type arr = unknown[]
|
||||
|
||||
export const typeIs = <T extends Type>(arg: any, t: T | T[]): arg is unknown & (T extends "array" ? arr : T extends "undefined" ? undefined | null : T extends "object" ? obj : T extends "string" ? string : T extends "number" ? number : T extends "bigint" ? bigint : T extends "boolean" ? boolean : T extends "symbol" ? symbol : T extends "function" ? Function : never) => {
|
||||
const types = Array.isArray(t) ? t : [t]
|
||||
return types.includes(typeOf(arg) as T);
|
||||
}
|
||||
|
||||
export const typeOf = (arg: any): Type => {
|
||||
const type = arg instanceof Array ? 'array' : arg === null ? 'undefined' : typeof arg
|
||||
return type;
|
||||
}
|
||||
@@ -8,6 +8,7 @@ export const tts = {
|
||||
ttOFFER_CREATE: 7,
|
||||
ttOFFER_CANCEL: 8,
|
||||
ttTICKET_CREATE: 10,
|
||||
ttTICKET_CANCEL: 11,
|
||||
ttSIGNER_LIST_SET: 12,
|
||||
ttPAYCHAN_CREATE: 13,
|
||||
ttPAYCHAN_FUND: 14,
|
||||
@@ -18,18 +19,20 @@ export const tts = {
|
||||
ttDEPOSIT_PREAUTH: 19,
|
||||
ttTRUST_SET: 20,
|
||||
ttACCOUNT_DELETE: 21,
|
||||
ttHOOK_SET: 22,
|
||||
ttNFTOKEN_MINT: 25,
|
||||
ttNFTOKEN_BURN: 26,
|
||||
ttNFTOKEN_CREATE_OFFER: 27,
|
||||
ttNFTOKEN_CANCEL_OFFER: 28,
|
||||
ttNFTOKEN_ACCEPT_OFFER: 29
|
||||
ttSET_HOOK: 22,
|
||||
ttURI_TOKEN_MINT: 45,
|
||||
ttURI_TOKEN_BURN: 46,
|
||||
ttURI_TOKEN_BUY: 47,
|
||||
ttURI_TOKEN_CREATE_SELL_OFFER: 48,
|
||||
ttURI_TOKEN_CANCEL_SELL_OFFER: 49,
|
||||
ttIMPORT: 97,
|
||||
ttINVOKE: 99
|
||||
}
|
||||
|
||||
export type TTS = typeof tts
|
||||
|
||||
const calculateHookOn = (arr: (keyof TTS)[]) => {
|
||||
let s = '0x3e3ff5bf'
|
||||
let s = '0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffbfffff'
|
||||
arr.forEach(n => {
|
||||
let v = BigInt(s)
|
||||
v ^= BigInt(1) << BigInt(tts[n])
|
||||
|
||||
@@ -1,9 +1,8 @@
|
||||
import { typeIs, typeOf } from './helpers'
|
||||
|
||||
export const extractSchemaProps = <O extends object>(obj: O) =>
|
||||
Object.entries(obj).reduce((prev, [key, val]) => {
|
||||
const typeOf = <T>(arg: T) =>
|
||||
arg instanceof Array ? 'array' : arg === null ? 'undefined' : typeof arg
|
||||
|
||||
const value = typeOf(val) === 'object' && '$type' in val && '$value' in val ? val?.$value : val
|
||||
const value = typeIs(val, "object") && '$type' in val && '$value' in val ? val?.$value : val
|
||||
const type = typeOf(value)
|
||||
|
||||
let schema: any = {
|
||||
@@ -12,19 +11,19 @@ export const extractSchemaProps = <O extends object>(obj: O) =>
|
||||
default: value
|
||||
}
|
||||
|
||||
if (typeOf(value) === 'array') {
|
||||
if (typeIs(value, "array")) {
|
||||
const item = value[0] // TODO merge other item schema's into one
|
||||
if (typeOf(item) !== 'object') {
|
||||
if (typeIs(item, "object")) {
|
||||
schema.items = {
|
||||
type: 'object',
|
||||
properties: extractSchemaProps(item),
|
||||
default: item
|
||||
}
|
||||
}
|
||||
// TODO support primitive-value arrays
|
||||
// TODO primitive-value arrays
|
||||
}
|
||||
|
||||
if (typeOf(value) === 'object') {
|
||||
if (typeIs(value, "object")) {
|
||||
schema.properties = extractSchemaProps(value)
|
||||
}
|
||||
return {
|
||||
|
||||
@@ -20,6 +20,13 @@ export type SetHookData = {
|
||||
}
|
||||
$metaData?: any
|
||||
}[]
|
||||
Memos?: {
|
||||
Memo: {
|
||||
MemoType: string,
|
||||
MemoData: string
|
||||
MemoFormat: string
|
||||
}
|
||||
}[]
|
||||
// HookGrants: {
|
||||
// HookGrant: {
|
||||
// Authorize: string;
|
||||
@@ -65,12 +72,22 @@ export const getInvokeOptions = (content?: string) => {
|
||||
label: opt,
|
||||
value: opt
|
||||
}))
|
||||
|
||||
// default
|
||||
if (!invokeOptions.length) {
|
||||
const payment = transactionOptions.find(tx => tx.value === 'ttPAYMENT')
|
||||
if (payment) return [payment]
|
||||
}
|
||||
|
||||
return invokeOptions
|
||||
}
|
||||
|
||||
export function toHex(str: string) {
|
||||
var result = ''
|
||||
for (var i = 0; i < str.length; i++) {
|
||||
const hex = str.charCodeAt(i).toString(16)
|
||||
result += hex.padStart(2, '0')
|
||||
}
|
||||
return result.toUpperCase()
|
||||
}
|
||||
|
||||
export function fromHex(hex: string) {
|
||||
var str = ''
|
||||
for (var i = 0; i < hex.length; i += 2) {
|
||||
str += String.fromCharCode(parseInt(hex.substring(i, i + 2), 16))
|
||||
}
|
||||
return str
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-account-buf-len
|
||||
|
||||
Function [hook_account](https://xrpl-hooks.readme.io/v2.0/reference/hook_account) has fixed-size account ID output.
|
||||
Function [hook_account](https://xrpl-hooks.readme.io/reference/hook_account) has fixed-size account ID output.
|
||||
|
||||
This check warns about too-small size of its output buffer (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-account-conv-buf-len
|
||||
|
||||
Function [util_raddr](https://xrpl-hooks.readme.io/v2.0/reference/util_raddr) has fixed-size account ID input.
|
||||
Function [util_raddr](https://xrpl-hooks.readme.io/reference/util_raddr) has fixed-size account ID input.
|
||||
|
||||
This check warns unless the correct size is passed in the input size parameter (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-account-conv-pure
|
||||
|
||||
Hooks identify accounts by the 20 byte account ID, which can be converted to an raddr using the [util_raddr](https://xrpl-hooks.readme.io/v2.0/reference/util_raddr) function. If the account ID never changes, a more efficient way to do this is precompute the raddr from the account ID.
|
||||
Hooks identify accounts by the 20 byte account ID, which can be converted to an raddr using the [util_raddr](https://xrpl-hooks.readme.io/reference/util_raddr) function. If the account ID never changes, a more efficient way to do this is precompute the raddr from the account ID.
|
||||
|
||||
This check warns about calls of `util_raddr` with constant input and proposes to add a tracing statement showing the computed value (so that the user can use it to replace the call).
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-array-buf-len
|
||||
|
||||
Hook API [sto_subarray](https://xrpl-hooks.readme.io/v2.0/reference/sto_subarray) requires non-empty input buffer and takes a parameter specifying the array index, whose value is limited - the sought object cannot be found if the limit is exceeded.
|
||||
Hook API [sto_subarray](https://xrpl-hooks.readme.io/reference/sto_subarray) requires non-empty input buffer and takes a parameter specifying the array index, whose value is limited - the sought object cannot be found if the limit is exceeded.
|
||||
|
||||
This check warns about empty input as well as too-large values of the index specified in calls to `sto_subarray` (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/serialized-objects)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/serialized-objects)
|
||||
|
||||
@@ -1,3 +1,3 @@
|
||||
# hooks-burden-prereq
|
||||
|
||||
Hook API [etxn_burden](https://xrpl-hooks.readme.io/v2.0/reference/etxn_burden) computes transaction burden, based on (i.a.) the number of reserved transactions, so a call to it must be preceded by a call to [etxn_reserve](https://xrpl-hooks.readme.io/v2.0/reference/etxn_reserve).
|
||||
Hook API [etxn_burden](https://xrpl-hooks.readme.io/reference/etxn_burden) computes transaction burden, based on (i.a.) the number of reserved transactions, so a call to it must be preceded by a call to [etxn_reserve](https://xrpl-hooks.readme.io/reference/etxn_reserve).
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-control-string-arg
|
||||
|
||||
Functions [accept](https://xrpl-hooks.readme.io/v2.0/reference/accept) and [rollback](https://xrpl-hooks.readme.io/v2.0/reference/rollback) take an optional string buffer stored outside the hook as its result message. This is useful for debugging but takes up space.
|
||||
Functions [accept](https://xrpl-hooks.readme.io/reference/accept) and [rollback](https://xrpl-hooks.readme.io/reference/rollback) take an optional string buffer stored outside the hook as its result message. This is useful for debugging but takes up space.
|
||||
|
||||
For a release version, this check warns about constant strings passed to `accept` and `rollback`.
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-detail-buf-len
|
||||
|
||||
Function [etxn_details](https://xrpl-hooks.readme.io/v2.0/reference/etxn_details) has fixed-size sfEmitDetails output.
|
||||
Function [etxn_details](https://xrpl-hooks.readme.io/reference/etxn_details) has fixed-size sfEmitDetails output.
|
||||
|
||||
This check warns about too-small size of its output buffer (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/emitted-transactions)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/emitted-transactions)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-detail-prereq
|
||||
|
||||
Hook API [etxn_details](https://xrpl-hooks.readme.io/v2.0/reference/etxn_details) serializes emit details, based on (i.a.) the number of reserved transactions, so a call to it must be preceded by a call to [etxn_reserve](https://xrpl-hooks.readme.io/v2.0/reference/etxn_reserve).
|
||||
Hook API [etxn_details](https://xrpl-hooks.readme.io/reference/etxn_details) serializes emit details, based on (i.a.) the number of reserved transactions, so a call to it must be preceded by a call to [etxn_reserve](https://xrpl-hooks.readme.io/reference/etxn_reserve).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/emitted-transactions)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/emitted-transactions)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-emit-buf-len
|
||||
|
||||
Function [emit](https://xrpl-hooks.readme.io/v2.0/reference/emit) has fixed-size transaction hash output.
|
||||
Function [emit](https://xrpl-hooks.readme.io/reference/emit) has fixed-size transaction hash output.
|
||||
|
||||
This check warns about too-small size of its output buffer (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/emitted-transactions)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/emitted-transactions)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-emit-prereq
|
||||
|
||||
Before emitting a transaction using [emit](https://xrpl-hooks.readme.io/v2.0/reference/emit) Hook API, a hook must set a maximal count of transactions it plans to emit, by calling [etxn_reserve](https://xrpl-hooks.readme.io/v2.0/reference/etxn_reserve).
|
||||
Before emitting a transaction using [emit](https://xrpl-hooks.readme.io/reference/emit) Hook API, a hook must set a maximal count of transactions it plans to emit, by calling [etxn_reserve](https://xrpl-hooks.readme.io/reference/etxn_reserve).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/emitted-transactions)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/emitted-transactions)
|
||||
|
||||
@@ -2,4 +2,4 @@
|
||||
|
||||
Recursive calls are disallowed in the implementation of hook entry points.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/loops-and-guarding#no-recursion)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/loops-and-guarding#no-recursion)
|
||||
|
||||
@@ -2,4 +2,4 @@
|
||||
|
||||
Shows error on function definitions with unexpected (that is, neither `hook` nor `cbak`) names.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/compiling-hooks#constraints)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/compiling-hooks#constraints)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-entry-points
|
||||
|
||||
A Hook always implements and exports a [hook](https://xrpl-hooks.readme.io/v2.0/reference/hook) function.
|
||||
A Hook always implements and exports a [hook](https://xrpl-hooks.readme.io/reference/hook) function.
|
||||
|
||||
This check shows error on translation units that do not have it.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/compiling-hooks)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/compiling-hooks)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-fee-prereq
|
||||
|
||||
Hook API [etxn_fee_base](https://xrpl-hooks.readme.io/v2.0/reference/etxn_fee_base) estimates a transaction fee, based on (i.a.) the number of reserved transactions, so a call to it must be preceded by a call to [etxn_reserve](https://xrpl-hooks.readme.io/v2.0/reference/etxn_reserve).
|
||||
Hook API [etxn_fee_base](https://xrpl-hooks.readme.io/reference/etxn_fee_base) estimates a transaction fee, based on (i.a.) the number of reserved transactions, so a call to it must be preceded by a call to [etxn_reserve](https://xrpl-hooks.readme.io/reference/etxn_reserve).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/hook-fees)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/hook-fees)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-field-add-buf-len
|
||||
|
||||
Emplacing a new field into STObject by calling [sto_emplace](https://xrpl-hooks.readme.io/v2.0/reference/sto_emplace) requires enough space to serialize the new STObject into; the API also limits sizes of the old object and field.
|
||||
Emplacing a new field into STObject by calling [sto_emplace](https://xrpl-hooks.readme.io/reference/sto_emplace) requires enough space to serialize the new STObject into; the API also limits sizes of the old object and field.
|
||||
|
||||
This check warns about insufficient output buffer space as well as too-large values of the inputs in calls to `sto_emplace` (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/serialized-objects)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/serialized-objects)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-field-buf-len
|
||||
|
||||
Hook API [sto_subfield](https://xrpl-hooks.readme.io/v2.0/reference/sto_subfield) requires non-empty input buffer.
|
||||
Hook API [sto_subfield](https://xrpl-hooks.readme.io/reference/sto_subfield) requires non-empty input buffer.
|
||||
|
||||
This check warns about empty input in calls to `sto_subfield` (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/serialized-objects)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/serialized-objects)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-field-del-buf-len
|
||||
|
||||
Erasing a field from STObject by calling [sto_erase](https://xrpl-hooks.readme.io/v2.0/reference/sto_erase) requires enough space to serialize the new STObject into; the API also limits size of the old object.
|
||||
Erasing a field from STObject by calling [sto_erase](https://xrpl-hooks.readme.io/reference/sto_erase) requires enough space to serialize the new STObject into; the API also limits size of the old object.
|
||||
|
||||
This check warns about insufficient output buffer space as well as too-large value of the input STObject in calls to `sto_erase` (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/serialized-objects)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/serialized-objects)
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
# hooks-float-arith-pure
|
||||
|
||||
Hooks can compute floating-point values in XFL format by calling functions [float_multiply](https://xrpl-hooks.readme.io/v2.0/reference/float_multiply), [float_mulratio](https://xrpl-hooks.readme.io/v2.0/reference/float_mulratio), [float_negate](https://xrpl-hooks.readme.io/v2.0/reference/float_negate), [float_sum](https://xrpl-hooks.readme.io/v2.0/reference/float_sum), [float_invert](https://xrpl-hooks.readme.io/v2.0/reference/float_invert) and [float_divide](https://xrpl-hooks.readme.io/v2.0/reference/float_divide) and access their constituent parts by calling [float_exponent](https://xrpl-hooks.readme.io/v2.0/reference/float_exponent), [float_mantissa](https://xrpl-hooks.readme.io/v2.0/reference/float_mantissa) and [float_sign](https://xrpl-hooks.readme.io/v2.0/reference/float_sign). If the inputs of the computation never change, a more efficient way to do this is to precompute it.
|
||||
Hooks can compute floating-point values in XFL format by calling functions [float_multiply](https://xrpl-hooks.readme.io/reference/float_multiply), [float_mulratio](https://xrpl-hooks.readme.io/reference/float_mulratio), [float_negate](https://xrpl-hooks.readme.io/reference/float_negate), [float_sum](https://xrpl-hooks.readme.io/reference/float_sum), [float_invert](https://xrpl-hooks.readme.io/reference/float_invert) and [float_divide](https://xrpl-hooks.readme.io/reference/float_divide) and access their constituent parts by calling [float_exponent](https://xrpl-hooks.readme.io/reference/float_exponent), [float_mantissa](https://xrpl-hooks.readme.io/reference/float_mantissa) and [float_sign](https://xrpl-hooks.readme.io/reference/float_sign). If the inputs of the computation never change, a more efficient way to do this is to precompute it.
|
||||
|
||||
This check warns about calls of the aforementioned functions with constant inputs and in simple cases proposes to add a tracing statement showing the computed value (so that the user can use it to replace the call). It also checks that the divisor passed to `float_divide`, `float_mulratio` and `float_invert` is not 0 (if it's specified by a constant - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/floating-point-numbers-xfl)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/floating-point-numbers-xfl)
|
||||
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-float-compare-pure
|
||||
|
||||
Hooks can compare floating-point values in XFL format by calling the [float_compare](https://xrpl-hooks.readme.io/v2.0/reference/float_compare) function. If the inputs of the comparison never change, its result is fixed and the function need not be called.
|
||||
Hooks can compare floating-point values in XFL format by calling the [float_compare](https://xrpl-hooks.readme.io/reference/float_compare) function. If the inputs of the comparison never change, its result is fixed and the function need not be called.
|
||||
|
||||
This check warns about calls of `float_compare` with constant inputs as well as invalid values of the comparison mode parameter (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/floating-point-numbers-xfl)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/floating-point-numbers-xfl)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-float-int-pure
|
||||
|
||||
Hooks can convert floating-point values in XFL format to integers by calling the [float_int](https://xrpl-hooks.readme.io/v2.0/reference/float_int) function. If the inputs of this function never change, a more efficient way to do this is to precompute the integer value.
|
||||
Hooks can convert floating-point values in XFL format to integers by calling the [float_int](https://xrpl-hooks.readme.io/reference/float_int) function. If the inputs of this function never change, a more efficient way to do this is to precompute the integer value.
|
||||
|
||||
This check warns about calls of `float_int` with constant inputs as well as invalid values of the decimal places parameter (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/floating-point-numbers-xfl)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/floating-point-numbers-xfl)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-float-manip-pure
|
||||
|
||||
Hooks can directly manipulate floating-point values in XFL format by calling functions [float_exponent_set](https://xrpl-hooks.readme.io/v2.0/reference/float_exponent_set), [float_mantissa_set](https://xrpl-hooks.readme.io/v2.0/reference/float_mantissa_set) and [float_sign_set](https://xrpl-hooks.readme.io/v2.0/reference/float_sign_set). If the inputs of the update never change, a more efficient way to do this is to precompute it.
|
||||
Hooks can directly manipulate floating-point values in XFL format by calling functions [float_exponent_set](https://xrpl-hooks.readme.io/reference/float_exponent_set), [float_mantissa_set](https://xrpl-hooks.readme.io/reference/float_mantissa_set) and [float_sign_set](https://xrpl-hooks.readme.io/reference/float_sign_set). If the inputs of the update never change, a more efficient way to do this is to precompute it.
|
||||
|
||||
This check warns about calls of the aforementioned functions with constant inputs and in simple cases proposes to add a tracing statement showing the computed value (so that the user can use it to replace the call). It also checks documented bounds of the second parameter of these functions (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/floating-point-numbers-xfl)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/floating-point-numbers-xfl)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-float-one-pure
|
||||
|
||||
Hooks can obtain XFL enclosing number 1 by calling the [float_one](https://xrpl-hooks.readme.io/v2.0/reference/float_one) function. Since the number never changes, a more efficient way is to use its precomputed value.
|
||||
Hooks can obtain XFL enclosing number 1 by calling the [float_one](https://xrpl-hooks.readme.io/reference/float_one) function. Since the number never changes, a more efficient way is to use its precomputed value.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/floating-point-numbers-xfl)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/floating-point-numbers-xfl)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-float-pure
|
||||
|
||||
Hooks can use floating-point values in XFL format, creating them from mantissa and exponent by calling the [float_set](https://xrpl-hooks.readme.io/v2.0/reference/float_set) function. If the mantissa and exponent never change, a more efficient way to do this is to precompute the floating-point value.
|
||||
Hooks can use floating-point values in XFL format, creating them from mantissa and exponent by calling the [float_set](https://xrpl-hooks.readme.io/reference/float_set) function. If the mantissa and exponent never change, a more efficient way to do this is to precompute the floating-point value.
|
||||
|
||||
This check warns about calls of `float_set` with constant inputs and proposes to add a tracing statement showing the computed value (so that the user can use it to replace the call). In the special case of 0 mantissa and 0 exponent ("canonical 0"), a replacement value of 0 is proposed directly, with no need to trace it.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/floating-point-numbers-xfl)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/floating-point-numbers-xfl)
|
||||
|
||||
@@ -3,4 +3,4 @@
|
||||
Only compile-time constants can be used as an argument in loop GUARD call. This check warns if a non compile-time constant is used.
|
||||
It also checks whether a compile-time constant is used as a first argument of `_g()` call and whether it is a unique value. If not - it warns.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/loops-and-guarding)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/loops-and-guarding)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-guard-called
|
||||
|
||||
Every hook needs to import the guard function [_g](https://xrpl-hooks.readme.io/v2.0/docs/loops-and-guarding#the-guard-function) and use it at least once.
|
||||
Every hook needs to import the guard function [_g](https://xrpl-hooks.readme.io/docs/loops-and-guarding#the-guard-function) and use it at least once.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/loops-and-guarding)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/loops-and-guarding)
|
||||
|
||||
@@ -32,4 +32,4 @@ for(int i = 0; GUARD(3), i < 3; ++i) {
|
||||
This check will warn if the GUARD call is missing and also it will propose a GUARD value based on the for loop initial value,
|
||||
the increment and loop condition.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/loops-and-guarding)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/loops-and-guarding)
|
||||
|
||||
@@ -11,4 +11,4 @@ Like for loops, while loops must have a guard in their condition:
|
||||
```
|
||||
|
||||
<BR/>
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/loops-and-guarding)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/loops-and-guarding)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-hash-buf-len
|
||||
|
||||
Functions [util_sha512h](https://xrpl-hooks.readme.io/v2.0/reference/util_sha512h), [hook_hash](https://xrpl-hooks.readme.io/v2.0/reference/hook_hash), [ledger_last_hash](https://xrpl-hooks.readme.io/v2.0/reference/ledger_last_hash), [etxn_nonce](https://xrpl-hooks.readme.io/v2.0/reference/etxn_nonce) and [ledger_nonce](https://xrpl-hooks.readme.io/v2.0/reference/ledger_nonce) have fixed-size hash output.
|
||||
Functions [util_sha512h](https://xrpl-hooks.readme.io/reference/util_sha512h), [hook_hash](https://xrpl-hooks.readme.io/reference/hook_hash), [ledger_last_hash](https://xrpl-hooks.readme.io/reference/ledger_last_hash), [etxn_nonce](https://xrpl-hooks.readme.io/reference/etxn_nonce) and [ledger_nonce](https://xrpl-hooks.readme.io/reference/ledger_nonce) have fixed-size hash output.
|
||||
|
||||
This check warns about too-small size of their output buffer (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-keylet-buf-len
|
||||
|
||||
Computing a ripple keylet by calling [util_keylet](https://xrpl-hooks.readme.io/v2.0/reference/util_keylet) requires valid parameters dependent on the keylet type.
|
||||
Computing a ripple keylet by calling [util_keylet](https://xrpl-hooks.readme.io/reference/util_keylet) requires valid parameters dependent on the keylet type.
|
||||
|
||||
This check does not fully parse these parameters, but warns about invalid keylet type as well as buffer sizes that cannot be valid (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/slots-and-keylets)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/slots-and-keylets)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-param-buf-len
|
||||
|
||||
Function [hook_param](https://xrpl-hooks.readme.io/v2.0/reference/hook_param) expects a limited-length name input and produces fixed-size value output.
|
||||
Function [hook_param](https://xrpl-hooks.readme.io/reference/hook_param) expects a limited-length name input and produces fixed-size value output.
|
||||
|
||||
This check warns about invalid sizes of input and output buffers (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/parameters)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/parameters)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-param-set-buf-len
|
||||
|
||||
Function [hook_param_set](https://xrpl-hooks.readme.io/v2.0/reference/hook_param_set) expects limited-length name, fixed-length hash and limited-length value inputs.
|
||||
Function [hook_param_set](https://xrpl-hooks.readme.io/reference/hook_param_set) expects limited-length name, fixed-length hash and limited-length value inputs.
|
||||
|
||||
This check warns about invalid sizes of input buffers (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/parameters)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/parameters)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-raddr-conv-buf-len
|
||||
|
||||
Hook API [util_accid](https://xrpl-hooks.readme.io/v2.0/reference/util_accid) has upper limit on the length of its input (because it expects it to be a raddr) and fixed-size account ID output.
|
||||
Hook API [util_accid](https://xrpl-hooks.readme.io/reference/util_accid) has upper limit on the length of its input (because it expects it to be a raddr) and fixed-size account ID output.
|
||||
|
||||
This check warns about invalid sizes of input and output parameters (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-raddr-conv-pure
|
||||
|
||||
Hooks identify accounts by the 20 byte account ID, which can be converted from a raddr using the [util_accid](https://xrpl-hooks.readme.io/v2.0/reference/util_accid) function. If the raddr never changes, a more efficient way to do this is precompute the account-id from the raddr.
|
||||
Hooks identify accounts by the 20 byte account ID, which can be converted from a raddr using the [util_accid](https://xrpl-hooks.readme.io/reference/util_accid) function. If the raddr never changes, a more efficient way to do this is precompute the account-id from the raddr.
|
||||
|
||||
This check warns about calls of `util_accid` with constant input and proposes to add a tracing statement showing the computed value (so that the user can use it to replace the call).
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-reserve-limit
|
||||
|
||||
Hook API [etxn_reserve](https://xrpl-hooks.readme.io/v2.0/reference/etxn_reserve) takes a parameter specifying the number of transactions intended to emit from the calling hook. Value of this parameter is limited, and the function fails if the limit is exceeded.
|
||||
Hook API [etxn_reserve](https://xrpl-hooks.readme.io/reference/etxn_reserve) takes a parameter specifying the number of transactions intended to emit from the calling hook. Value of this parameter is limited, and the function fails if the limit is exceeded.
|
||||
|
||||
This check warns about too-large values of the number of reserved transactions (if they're specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/emitted-transactions)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/emitted-transactions)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-skip-hash-buf-len
|
||||
|
||||
Function [hook_skip](https://xrpl-hooks.readme.io/v2.0/reference/hook_skip) has fixed-size canonical hash input.
|
||||
Function [hook_skip](https://xrpl-hooks.readme.io/reference/hook_skip) has fixed-size canonical hash input.
|
||||
|
||||
This check warns about invalid size of its input buffer (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-slot-hash-buf-len
|
||||
|
||||
Function [slot_id](https://xrpl-hooks.readme.io/v2.0/reference/slot_id) has fixed-size canonical hash output.
|
||||
Function [slot_id](https://xrpl-hooks.readme.io/reference/slot_id) has fixed-size canonical hash output.
|
||||
|
||||
This check warns about too-small size of its output buffer as well as invalid values of the slot number parameter (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/slots-and-keylets)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/slots-and-keylets)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-slot-keylet-buf-len
|
||||
|
||||
Function [slot_set](https://xrpl-hooks.readme.io/v2.0/reference/slot_set) has structured keylet input.
|
||||
Function [slot_set](https://xrpl-hooks.readme.io/reference/slot_set) has structured keylet input.
|
||||
|
||||
This check does not parse the input, but warns about its sizes that cannot be valid as well as invalid values of the slot number parameter (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/slots-and-keylets)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/slots-and-keylets)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-slot-limit
|
||||
|
||||
Hook APIs [slot](https://xrpl-hooks.readme.io/v2.0/reference/slot), [slot_count](https://xrpl-hooks.readme.io/v2.0/reference/slot_count), [slot_clear](https://xrpl-hooks.readme.io/v2.0/reference/slot_clear), [slot_size](https://xrpl-hooks.readme.io/v2.0/reference/slot_size), [slot_float](https://xrpl-hooks.readme.io/v2.0/reference/slot_float) and [trace_slot](https://xrpl-hooks.readme.io/v2.0/reference/trace_slot) take a parameter specifying the accessed slot number. Value of this parameter is limited, and the functions fail if the limit is exceeded.
|
||||
Hook APIs [slot](https://xrpl-hooks.readme.io/reference/slot), [slot_count](https://xrpl-hooks.readme.io/reference/slot_count), [slot_clear](https://xrpl-hooks.readme.io/reference/slot_clear), [slot_size](https://xrpl-hooks.readme.io/reference/slot_size), [slot_float](https://xrpl-hooks.readme.io/reference/slot_float) and [trace_slot](https://xrpl-hooks.readme.io/reference/trace_slot) take a parameter specifying the accessed slot number. Value of this parameter is limited, and the functions fail if the limit is exceeded.
|
||||
|
||||
This check warns about too-large values of the slot number (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/slots-and-keylets)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/slots-and-keylets)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-slot-sub-limit
|
||||
|
||||
Hook APIs [slot_subarray](https://xrpl-hooks.readme.io/v2.0/reference/slot_subarray) and [slot_subfield](https://xrpl-hooks.readme.io/v2.0/reference/slot_subfield) take parameters specifying parent and child slot numbers. Values of these parameters are limited, and the functions fail if the limit is exceeded.
|
||||
Hook APIs [slot_subarray](https://xrpl-hooks.readme.io/reference/slot_subarray) and [slot_subfield](https://xrpl-hooks.readme.io/reference/slot_subfield) take parameters specifying parent and child slot numbers. Values of these parameters are limited, and the functions fail if the limit is exceeded.
|
||||
|
||||
This check warns about too-large values of the slot numbers (if they're specified by a constant - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/slots-and-keylets)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/slots-and-keylets)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-slot-type-limit
|
||||
|
||||
Hook API [slot_type](https://xrpl-hooks.readme.io/v2.0/reference/slot_type) takes a parameter specifying the accessed slot number. Value of this parameter is limited, and the function fails if the limit is exceeded.
|
||||
Hook API [slot_type](https://xrpl-hooks.readme.io/reference/slot_type) takes a parameter specifying the accessed slot number. Value of this parameter is limited, and the function fails if the limit is exceeded.
|
||||
|
||||
This check warns about too-large values of the slot number as well as invalid values of the flags parameter (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/slots-and-keylets)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/slots-and-keylets)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-state-buf-len
|
||||
|
||||
Functions [state](https://xrpl-hooks.readme.io/v2.0/reference/state) and [state_set](https://xrpl-hooks.readme.io/v2.0/reference/state_set) accept fixed-size Hook State key.
|
||||
Functions [state](https://xrpl-hooks.readme.io/reference/state) and [state_set](https://xrpl-hooks.readme.io/reference/state_set) accept fixed-size Hook State key.
|
||||
|
||||
This check warns about invalid size of its input buffer (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/state-management)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/state-management)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-transaction-hash-buf-len
|
||||
|
||||
Function [otxn_id](https://xrpl-hooks.readme.io/v2.0/reference/otxn_id) has fixed-size canonical hash output.
|
||||
Function [otxn_id](https://xrpl-hooks.readme.io/reference/otxn_id) has fixed-size canonical hash output.
|
||||
|
||||
This check warns about too-small size of its output buffer (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-transaction-slot-limit
|
||||
|
||||
Function [otxn_slot](https://xrpl-hooks.readme.io/v2.0/reference/otxn_slot) takes a parameter specifying the accessed slot number. Value of this parameter is limited, and the function fails if the limit is exceeded.
|
||||
Function [otxn_slot](https://xrpl-hooks.readme.io/reference/otxn_slot) takes a parameter specifying the accessed slot number. Value of this parameter is limited, and the function fails if the limit is exceeded.
|
||||
|
||||
This check warns about too-large values of the slot number (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/slots-and-keylets)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/slots-and-keylets)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-trivial-cbak
|
||||
|
||||
A Hook may implement and export a [cbak](https://xrpl-hooks.readme.io/v2.0/reference/cbak) function.
|
||||
A Hook may implement and export a [cbak](https://xrpl-hooks.readme.io/reference/cbak) function.
|
||||
|
||||
But the function is optional, and defining it so that it doesn't do anything besides returning a constant value is unnecessary (except for some debugging scenarios) and just increases the hook size. This check warns about such implementations.
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/compiling-hooks)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/compiling-hooks)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# hooks-validate-buf-len
|
||||
|
||||
Hook API [sto_validate](https://xrpl-hooks.readme.io/v2.0/reference/sto_validate) requires non-empty input buffer.
|
||||
Hook API [sto_validate](https://xrpl-hooks.readme.io/reference/sto_validate) requires non-empty input buffer.
|
||||
|
||||
This check warns about empty input in calls to `sto_validate` (if it's specified by a constant - variable parameter is ignored).
|
||||
|
||||
[Read more](https://xrpl-hooks.readme.io/v2.0/docs/serialized-objects)
|
||||
[Read more](https://xrpl-hooks.readme.io/docs/serialized-objects)
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# hooks-verify-buf-len
|
||||
|
||||
Verifying a cryptographic signature by calling [util_verify](https://xrpl-hooks.readme.io/v2.0/reference/util_verify) requires valid public key & data signature.
|
||||
Verifying a cryptographic signature by calling [util_verify](https://xrpl-hooks.readme.io/reference/util_verify) requires valid public key & data signature.
|
||||
|
||||
This check does not fully parse these parameters, but warns about their sizes that cannot be valid (if they're specified by constants - variable parameters are ignored).
|
||||
|
||||
44
yarn.lock
44
yarn.lock
@@ -1383,16 +1383,16 @@ balanced-match@^1.0.0:
|
||||
resolved "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz"
|
||||
integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==
|
||||
|
||||
base-x@3.0.9, base-x@^3.0.2, base-x@^3.0.3:
|
||||
base-x@3.0.9, base-x@^3.0.2, base-x@^3.0.3, base-x@^3.0.9:
|
||||
version "3.0.9"
|
||||
resolved "https://registry.npmjs.org/base-x/-/base-x-3.0.9.tgz"
|
||||
resolved "https://registry.yarnpkg.com/base-x/-/base-x-3.0.9.tgz#6349aaabb58526332de9f60995e548a53fe21320"
|
||||
integrity sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ==
|
||||
dependencies:
|
||||
safe-buffer "^5.0.1"
|
||||
|
||||
base64-js@^1.0.2, base64-js@^1.5.1:
|
||||
base64-js@^1.0.2, base64-js@^1.3.1, base64-js@^1.5.1:
|
||||
version "1.5.1"
|
||||
resolved "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz"
|
||||
resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a"
|
||||
integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==
|
||||
|
||||
before-after-hook@^2.2.0:
|
||||
@@ -1660,13 +1660,13 @@ buffer-xor@^1.0.3:
|
||||
resolved "https://registry.yarnpkg.com/buffer-xor/-/buffer-xor-1.0.3.tgz#26e61ed1422fb70dd42e6e36729ed51d855fe8d9"
|
||||
integrity sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==
|
||||
|
||||
buffer@5.6.0:
|
||||
version "5.6.0"
|
||||
resolved "https://registry.npmjs.org/buffer/-/buffer-5.6.0.tgz"
|
||||
integrity sha512-/gDYp/UtU0eA1ys8bOs9J6a+E/KWIY+DZ+Q2WESNUA0jFRsJOc0SNUO6xJ5SGA1xueg3NL65W6s+NY5l9cunuw==
|
||||
buffer@6.0.3:
|
||||
version "6.0.3"
|
||||
resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6"
|
||||
integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==
|
||||
dependencies:
|
||||
base64-js "^1.0.2"
|
||||
ieee754 "^1.1.4"
|
||||
base64-js "^1.3.1"
|
||||
ieee754 "^1.2.1"
|
||||
|
||||
buffer@~5.2.1:
|
||||
version "5.2.1"
|
||||
@@ -2813,9 +2813,9 @@ https-proxy-agent@^5.0.0:
|
||||
agent-base "6"
|
||||
debug "4"
|
||||
|
||||
ieee754@^1.1.4:
|
||||
ieee754@^1.1.4, ieee754@^1.2.1:
|
||||
version "1.2.1"
|
||||
resolved "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz"
|
||||
resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352"
|
||||
integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==
|
||||
|
||||
ignore@^4.0.6:
|
||||
@@ -4610,17 +4610,25 @@ ripple-address-codec@^4.2.4:
|
||||
base-x "3.0.9"
|
||||
create-hash "^1.1.2"
|
||||
|
||||
ripple-binary-codec@=1.4.2, ripple-binary-codec@^1.1.3, ripple-binary-codec@^1.4.2, ripple-binary-codec@^1.5.0-beta.2:
|
||||
version "1.4.2"
|
||||
resolved "https://registry.yarnpkg.com/ripple-binary-codec/-/ripple-binary-codec-1.4.2.tgz#cdc35353e4bc7c3a704719247c82b4c4d0b57dd3"
|
||||
integrity sha512-EDKIyZMa/6Ay/oNgCwjD9b9CJv0zmBreeHVQeG4BYwy+9GPnIQjNeT5e/aB6OjAnhcmpgbPeBmzwmNVwzxlt0w==
|
||||
ripple-address-codec@^4.3.0:
|
||||
version "4.3.0"
|
||||
resolved "https://registry.yarnpkg.com/ripple-address-codec/-/ripple-address-codec-4.3.0.tgz#45edeb0312b4fe4607b37b7c4cff467802ad571d"
|
||||
integrity sha512-Tvd81i7hpDmNqHvkj6iYlj8Tv3I1Romw5gfjni9eacewJvGV2xe+p2y0FAw39z72qfciRMhQyHvpnviBcWVBNw==
|
||||
dependencies:
|
||||
base-x "^3.0.9"
|
||||
create-hash "^1.1.2"
|
||||
|
||||
ripple-binary-codec@=1.6.0, ripple-binary-codec@^1.1.3, ripple-binary-codec@^1.4.2, ripple-binary-codec@^1.5.0-beta.2:
|
||||
version "1.6.0"
|
||||
resolved "https://registry.yarnpkg.com/ripple-binary-codec/-/ripple-binary-codec-1.6.0.tgz#848f93a10363a521f2a158751a873a9e89c25d15"
|
||||
integrity sha512-fa0aMSbh1VOGEHIWCF/VuIvoMoQ/1HLJoBxm+oPNPIDyZJG1uRpLYph1pcvAlDuMutHM3ZHMzWjJpe3AaiMIUA==
|
||||
dependencies:
|
||||
assert "^2.0.0"
|
||||
big-integer "^1.6.48"
|
||||
buffer "5.6.0"
|
||||
buffer "6.0.3"
|
||||
create-hash "^1.2.0"
|
||||
decimal.js "^10.2.0"
|
||||
ripple-address-codec "^4.2.4"
|
||||
ripple-address-codec "^4.3.0"
|
||||
|
||||
ripple-keypairs@^1.0.3:
|
||||
version "1.1.3"
|
||||
|
||||
Reference in New Issue
Block a user