diff --git a/cashtab/src/components/Send/Send.js b/cashtab/src/components/Send/Send.js
index 36dddfe07..67fffe1df 100644
--- a/cashtab/src/components/Send/Send.js
+++ b/cashtab/src/components/Send/Send.js
@@ -1,1010 +1,1009 @@
 import React, { useState, useEffect } from 'react';
 import { useLocation } from 'react-router-dom';
 import PropTypes from 'prop-types';
 import { WalletContext } from 'utils/context';
 import {
     AntdFormWrapper,
     SendBchInput,
     DestinationAddressSingle,
     DestinationAddressMulti,
 } from 'components/Common/EnhancedInputs';
 import { ThemedMailOutlined } from 'components/Common/CustomIcons';
 import { CustomCollapseCtn } from 'components/Common/StyledCollapse';
 import { Form, message, Modal, Alert, Input } from 'antd';
 import { Row, Col, Switch } from 'antd';
 import PrimaryButton, { DisabledButton } from 'components/Common/PrimaryButton';
 import useWindowDimensions from 'hooks/useWindowDimensions';
 import {
     sendXecNotification,
     errorNotification,
 } from 'components/Common/Notifications';
 import { isMobile, isIOS, isSafari } from 'react-device-detect';
 import {
     parseAddressForParams,
     sumOneToManyXec,
     toSatoshis,
 } from 'utils/cashMethods';
 import { Event } from 'utils/GoogleAnalytics';
 import {
     fiatToCrypto,
     shouldRejectAmountInput,
     isValidXecAddress,
     isValidEtokenAddress,
     isAliasFormat,
     isValidMultiSendUserInput,
 } from 'utils/validation';
 import BalanceHeader from 'components/Common/BalanceHeader';
 import BalanceHeaderFiat from 'components/Common/BalanceHeaderFiat';
 import {
     ZeroBalanceHeader,
     ConvertAmount,
     AlertMsg,
     WalletInfoCtn,
     SidePaddingCtn,
     FormLabel,
     TxLink,
     MsgBytesizeError,
 } from 'components/Common/Atoms';
-import {
-    getWalletState,
-    fromSatoshisToXec,
-    calcFee,
-    getMessageByteSize,
-} from 'utils/cashMethods';
+import { getWalletState, fromSatoshisToXec, calcFee } from 'utils/cashMethods';
 import { sendXec, getMultisendTargetOutputs } from 'transactions';
-import { getCashtabMsgTargetOutput, getAirdropTargetOutput } from 'opreturn';
+import {
+    getCashtabMsgTargetOutput,
+    getAirdropTargetOutput,
+    getCashtabMsgByteCount,
+} from 'opreturn';
 import ApiError from 'components/Common/ApiError';
 import { formatFiatBalance, formatBalance } from 'utils/formatting';
 import styled from 'styled-components';
 import WalletLabel from 'components/Common/WalletLabel.js';
 import { opReturn as opreturnConfig } from 'config/opreturn';
 import { explorer } from 'config/explorer';
 import { queryAliasServer } from 'utils/aliasUtils';
 import { supportedFiatCurrencies } from 'config/cashtabSettings';
 import appConfig from 'config/app';
 import aliasSettings from 'config/alias';
 
 const { TextArea } = Input;
 
 const TextAreaLabel = styled.div`
     text-align: left;
     color: ${props => props.theme.forms.text};
     padding-left: 1px;
     white-space: nowrap;
 `;
 
 const AliasAddressPreviewLabel = styled.div`
     text-align: center;
     color: ${props => props.theme.forms.text};
     padding-left: 1px;
     white-space: nowrap;
 `;
 
 const AmountPreviewCtn = styled.div`
     display: flex;
     flex-direction: column;
     justify-content: top;
     max-height: 1rem;
 `;
 
 const SendInputCtn = styled.div`
     .ant-form-item-explain-error {
         @media (max-width: 300px) {
             font-size: 12px;
         }
     }
 `;
 
 const LocaleFormattedValue = styled.h3`
     color: ${props => props.theme.contrast};
     font-weight: bold;
     margin-bottom: 0;
 `;
 
 const SendAddressHeader = styled.div`
     display: flex;
     align-items: center;
 `;
 const DestinationAddressSingleCtn = styled.div``;
 const DestinationAddressMultiCtn = styled.div``;
 
 const ExpandingAddressInputCtn = styled.div`
     min-height: 14rem;
     ${DestinationAddressSingleCtn} {
         overflow: hidden;
         transition: ${props =>
             props.open
                 ? 'max-height 200ms ease-in, opacity 200ms ease-out'
                 : 'max-height 200ms cubic-bezier(0, 1, 0, 1), opacity 200ms ease-in'};
         max-height: ${props => (props.open ? '0rem' : '12rem')};
         opacity: ${props => (props.open ? 0 : 1)};
     }
     ${DestinationAddressMultiCtn} {
         overflow: hidden;
         transition: ${props =>
             props.open
                 ? 'max-height 200ms ease-in, transform 200ms ease-out, opacity 200ms ease-in'
                 : 'max-height 200ms cubic-bezier(0, 1, 0, 1), transform 200ms ease-out'};
         max-height: ${props => (props.open ? '13rem' : '0rem')};
         transform: ${props =>
             props.open ? 'translateY(0%)' : 'translateY(100%)'};
         opacity: ${props => (props.open ? 1 : 0)};
     }
 `;
 
 const PanelHeaderCtn = styled.div`
     display: flex;
     justify-content: center;
     align-items: center;
     gap: 1rem;
 `;
 
 const SendBCH = ({ passLoadingStatus }) => {
     // use balance parameters from wallet.state object and not legacy balances parameter from walletState, if user has migrated wallet
     // this handles edge case of user with old wallet who has not opened latest Cashtab version yet
 
     // If the wallet object from ContextValue has a `state key`, then check which keys are in the wallet object
     // Else set it as blank
     const ContextValue = React.useContext(WalletContext);
     const location = useLocation();
     const {
         wallet,
         fiatPrice,
         apiError,
         cashtabSettings,
         changeCashtabSettings,
         chronik,
         cashtabCache,
     } = ContextValue;
     const walletState = getWalletState(wallet);
     const { balances, nonSlpUtxos } = walletState;
     // Modal settings
     const [isOneToManyXECSend, setIsOneToManyXECSend] = useState(false);
     const [opReturnMsg, setOpReturnMsg] = useState(false);
 
     // Get device window width
     // If this is less than 769, the page will open with QR scanner open
     const { width } = useWindowDimensions();
     // Load with QR code open if device is mobile and NOT iOS + anything but safari
     const scannerSupported =
         cashtabSettings &&
         cashtabSettings.autoCameraOn === true &&
         width < 769 &&
         isMobile &&
         !(isIOS && !isSafari);
 
     const [formData, setFormData] = useState({
         value: '',
         address: '',
         airdropTokenId: '',
     });
     const [queryStringText, setQueryStringText] = useState(null);
     const [sendBchAddressError, setSendBchAddressError] = useState(false);
     const [sendBchAmountError, setSendBchAmountError] = useState(false);
     const [isMsgError, setIsMsgError] = useState(false);
     const [aliasInputAddress, setAliasInputAddress] = useState(false);
     const [selectedCurrency, setSelectedCurrency] = useState(appConfig.ticker);
 
     // Support cashtab button from web pages
     const [txInfoFromUrl, setTxInfoFromUrl] = useState(false);
 
     // Show a confirmation modal on transactions created by populating form from web page button
     const [isModalVisible, setIsModalVisible] = useState(false);
 
     // Airdrop transactions embed the additional tokenId (32 bytes), along with prefix (4 bytes) and two pushdata (2 bytes)
     // hence setting airdrop tx message limit to 38 bytes less than opreturnConfig.cashtabMsgByteLimit
     const pushDataByteCount = 1;
     const prefixByteCount = 4;
     const tokenIdByteCount = 32;
     const localAirdropTxAddedBytes =
         pushDataByteCount +
         tokenIdByteCount +
         pushDataByteCount +
         prefixByteCount; // 38
 
     const [airdropFlag, setAirdropFlag] = useState(false);
 
     const userLocale = navigator.language;
     const clearInputForms = () => {
         setFormData({
             value: '',
             address: '',
         });
         setOpReturnMsg(''); // OP_RETURN message has its own state field
         setAliasInputAddress(false); // clear alias address preview
     };
 
     const checkForConfirmationBeforeSendXec = () => {
         if (txInfoFromUrl || queryStringText !== null) {
             setIsModalVisible(true);
         } else if (cashtabSettings.sendModal) {
             setIsModalVisible(cashtabSettings.sendModal);
         } else {
             // if the user does not have the send confirmation enabled in settings then send directly
             send();
         }
     };
 
     const handleOk = () => {
         setIsModalVisible(false);
         send();
     };
 
     const handleCancel = () => {
         setIsModalVisible(false);
     };
 
     // If the balance has changed, unlock the UI
     // This is redundant, if backend has refreshed in 1.75s timeout below, UI will already be unlocked
     useEffect(() => {
         passLoadingStatus(false);
     }, [balances.totalBalance]);
 
     useEffect(() => {
         // only run this useEffect block if cashtabCache is defined
         if (!cashtabCache || typeof cashtabCache === 'undefined') {
             return;
         }
 
         // Manually parse for txInfo object on page load when Send.js is loaded with a query string
 
         // if this was routed from Wallet screen's Reply to message link then prepopulate the address and value field
         if (location && location.state && location.state.replyAddress) {
             setFormData({
                 address: location.state.replyAddress,
                 value: `${fromSatoshisToXec(appConfig.dustSats).toString()}`,
             });
         }
 
         // if this was routed from the Contact List
         if (location && location.state && location.state.contactSend) {
             setFormData({
                 address: location.state.contactSend,
             });
 
             // explicitly trigger the address validation upon navigation from contact list
             handleAddressChange({
                 target: {
                     name: 'address',
                     value: location.state.contactSend,
                 },
             });
         }
 
         // if this was routed from the Airdrop screen's Airdrop Calculator then
         // switch to multiple recipient mode and prepopulate the recipients field
         if (
             location &&
             location.state &&
             location.state.airdropRecipients &&
             location.state.airdropTokenId
         ) {
             setIsOneToManyXECSend(true);
             setFormData({
                 address: location.state.airdropRecipients,
                 airdropTokenId: location.state.airdropTokenId,
             });
 
             // validate the airdrop outputs from the calculator
             handleMultiAddressChange({
                 target: {
                     value: location.state.airdropRecipients,
                 },
             });
 
             setAirdropFlag(true);
         }
 
         // Do not set txInfo in state if query strings are not present
         if (
             !window.location ||
             !window.location.hash ||
             window.location.hash === '#/send'
         ) {
             return;
         }
 
         const txInfoArr = window.location.hash.split('?')[1].split('&');
 
         // Iterate over this to create object
         const txInfo = {};
         for (let i = 0; i < txInfoArr.length; i += 1) {
             let txInfoKeyValue = txInfoArr[i].split('=');
             let key = txInfoKeyValue[0];
             let value = txInfoKeyValue[1];
             txInfo[key] = value;
         }
         console.log(`txInfo from page params`, txInfo);
         setTxInfoFromUrl(txInfo);
         populateFormsFromUrl(txInfo);
     }, [cashtabCache]);
 
     function populateFormsFromUrl(txInfo) {
         if (txInfo && txInfo.address && txInfo.value) {
             setFormData({
                 address: txInfo.address,
                 value: txInfo.value,
             });
         }
     }
 
     function handleSendXecError(errorObj, oneToManyFlag) {
         // Set loading to false here as well, as balance may not change depending on where error occured in try loop
         passLoadingStatus(false);
         let message;
         if (
             errorObj.error &&
             errorObj.error.includes(
                 'too-long-mempool-chain, too many unconfirmed ancestors [limit: 50] (code 64)',
             )
         ) {
             message = `The ${appConfig.ticker} you are trying to send has too many unconfirmed ancestors to send (limit 50). Sending will be possible after a block confirmation. Try again in about 10 minutes.`;
         } else {
             message =
                 errorObj.message || errorObj.error || JSON.stringify(errorObj);
         }
 
         if (oneToManyFlag) {
             errorNotification(errorObj, message, 'Sending XEC one to many');
         } else {
             errorNotification(errorObj, message, 'Sending XEC');
         }
     }
 
     async function send() {
         setFormData({
             ...formData,
         });
 
         // Initialize targetOutputs for this tx
         let targetOutputs = [];
 
         // If you have an OP_RETURN output, add it at index 0
         // Aesthetic choice, easier to see when checking on block explorer
 
         if (airdropFlag) {
             // Airdrop txs require special OP_RETURN handling
             targetOutputs.push(
                 getAirdropTargetOutput(
                     formData.airdropTokenId,
                     opReturnMsg ? opReturnMsg : '',
                 ),
             );
         } else if (opReturnMsg !== false && opReturnMsg !== '') {
             // If not an airdrop msg, add opReturnMsg as a Cashtab Msg, if it exists
             targetOutputs.push(getCashtabMsgTargetOutput(opReturnMsg));
         }
 
         if (isOneToManyXECSend) {
             // Handle XEC send to multiple addresses
             targetOutputs = targetOutputs.concat(
                 getMultisendTargetOutputs(formData.address),
             );
 
             Event('Send.js', 'SendToMany', selectedCurrency);
         } else {
             // Handle XEC send to one address
             let cleanAddress;
             // check state on whether this is an alias or ecash address
             if (aliasInputAddress) {
                 cleanAddress = aliasInputAddress;
             } else {
                 // Get the non-alias param-free address
                 cleanAddress = formData.address.split('?')[0];
             }
             // Calculate the amount in XEC
             let xecSendValue = formData.value;
 
             if (selectedCurrency !== 'XEC') {
                 // If fiat send is selected, calculate send amount in XEC
                 xecSendValue = fiatToCrypto(xecSendValue, fiatPrice);
             }
 
             const satoshisToSend = toSatoshis(xecSendValue);
 
             targetOutputs.push({
                 address: cleanAddress,
                 value: satoshisToSend,
             });
 
             Event('Send.js', 'Send', selectedCurrency);
         }
 
         passLoadingStatus(true);
 
         // Send and notify
         try {
             const txObj = await sendXec(
                 chronik,
                 wallet,
                 targetOutputs,
                 appConfig.defaultFee,
             );
             sendXecNotification(
                 `${explorer.blockExplorerUrl}/tx/${txObj.response.txid}`,
             );
             clearInputForms();
             setAirdropFlag(false);
         } catch (err) {
             handleSendXecError(err, isOneToManyXECSend);
         }
     }
 
     const handleAddressChange = async e => {
         setAliasInputAddress(false); // clear alias address preview
         const { value, name } = e.target;
         let error = false;
         let addressString = value;
         // parse address for parameters
         const addressInfo = parseAddressForParams(addressString);
         const { address, queryString, amount } = addressInfo;
 
         // validate address
         const isValid = isValidXecAddress(addressInfo.address);
 
         // Store query string in state (disable currency selection if loaded from query string)
         setQueryStringText(queryString);
 
         // Is this valid address?
         if (!isValid) {
             error = `Invalid address`;
             // If valid address but token format
             if (isValidEtokenAddress(address)) {
                 error = `eToken addresses are not supported for ${appConfig.ticker} sends`;
             }
         }
 
         // if input is invalid as an ecash address, check if it's a valid alias
         // otherwise the invalid address error above will be displayed
         const isAliasInput = isAliasFormat(address);
         if (isAliasInput) {
             // reset the invalid address check from above
             error = false;
 
             // extract alias without the `.xec`
             const aliasName = address.slice(0, address.length - 4);
 
             // retrieve the alias details for `aliasName` from alias-server
             let aliasDetails;
             try {
                 aliasDetails = await queryAliasServer('alias', aliasName);
                 if (!aliasDetails.address) {
                     error =
                         'eCash Alias does not exist or yet to receive 1 confirmation';
                     setAliasInputAddress(false);
                 } else {
                     // Valid address response returned
                     setAliasInputAddress(aliasDetails.address);
                 }
             } catch (err) {
                 console.log(`handleAddressChange(): error retrieving alias`);
                 setAliasInputAddress(false);
                 errorNotification(null, 'Error retrieving alias info');
             }
         }
 
         setSendBchAddressError(error);
 
         // Set amount if it's in the query string
         if (amount !== null) {
             // Set currency to BCHA
             setSelectedCurrency(appConfig.ticker);
 
             // Use this object to mimic user input and get validation for the value
             let amountObj = {
                 target: {
                     name: 'value',
                     value: amount,
                 },
             };
             handleBchAmountChange(amountObj);
             setFormData({
                 ...formData,
                 value: amount,
             });
         }
 
         // Set address field to user input
         setFormData(p => ({
             ...p,
             [name]: value,
         }));
     };
 
     const handleMultiAddressChange = e => {
         const { value, name } = e.target;
         let errorOrIsValid = isValidMultiSendUserInput(value);
 
         // If you get an error msg, set it. If validation is good, clear error msg.
         setSendBchAddressError(
             typeof errorOrIsValid === 'string' ? errorOrIsValid : false,
         );
 
         // Set address field to user input
         setFormData(p => ({
             ...p,
             [name]: value,
         }));
     };
 
     const handleSelectedCurrencyChange = e => {
         setSelectedCurrency(e);
         // Clear input field to prevent accidentally sending 1 BCH instead of 1 USD
         setFormData(p => ({
             ...p,
             value: '',
         }));
     };
 
     const handleBchAmountChange = e => {
         const { value, name } = e.target;
         let bchValue = value;
         const error = shouldRejectAmountInput(
             bchValue,
             selectedCurrency,
             fiatPrice,
             balances.totalBalance,
         );
         setSendBchAmountError(error);
 
         setFormData(p => ({
             ...p,
             [name]: value,
         }));
     };
 
     const handleMsgChange = e => {
         const { value } = e.target;
         let msgError = false;
-        const msgByteSize = getMessageByteSize(value);
+        const msgByteSize = getCashtabMsgByteCount(value);
 
         const maxSize =
             location && location.state && location.state.airdropTokenId
                 ? opreturnConfig.cashtabMsgByteLimit - localAirdropTxAddedBytes
                 : opreturnConfig.cashtabMsgByteLimit;
         if (msgByteSize > maxSize) {
             msgError = `Message can not exceed ${maxSize} bytes`;
         }
         setIsMsgError(msgError);
         setOpReturnMsg(e.target.value);
     };
 
     const onMax = async () => {
         // Clear amt error
         setSendBchAmountError(false);
         // Set currency to BCH
         setSelectedCurrency(appConfig.ticker);
         try {
             const txFeeSats = calcFee(nonSlpUtxos);
 
             const txFeeBch = txFeeSats / 10 ** appConfig.cashDecimals;
             let value =
                 balances.totalBalance - txFeeBch >= 0
                     ? (balances.totalBalance - txFeeBch).toFixed(
                           appConfig.cashDecimals,
                       )
                     : 0;
 
             setFormData({
                 ...formData,
                 value,
             });
         } catch (err) {
             console.log(`Error in onMax:`);
             console.log(err);
             message.error(
                 'Unable to calculate the max value due to network errors',
             );
         }
     };
     // Display price in USD below input field for send amount, if it can be calculated
     let fiatPriceString = '';
     if (fiatPrice !== null && !isNaN(formData.value)) {
         if (selectedCurrency === appConfig.ticker) {
             // calculate conversion to fiatPrice
             fiatPriceString = `${(fiatPrice * Number(formData.value)).toFixed(
                 2,
             )}`;
 
             // formats to fiat locale style
             fiatPriceString = formatFiatBalance(
                 Number(fiatPriceString),
                 userLocale,
             );
 
             // insert symbol and currency before/after the locale formatted fiat balance
             fiatPriceString = `${
                 cashtabSettings
                     ? `${
                           supportedFiatCurrencies[cashtabSettings.fiatCurrency]
                               .symbol
                       } `
                     : '$ '
             } ${fiatPriceString} ${
                 cashtabSettings && cashtabSettings.fiatCurrency
                     ? cashtabSettings.fiatCurrency.toUpperCase()
                     : 'USD'
             }`;
         } else {
             fiatPriceString = `${
                 formData.value
                     ? formatFiatBalance(
                           Number(fiatToCrypto(formData.value, fiatPrice)),
                           userLocale,
                       )
                     : formatFiatBalance(0, userLocale)
             } ${appConfig.ticker}`;
         }
     }
 
     const priceApiError = fiatPrice === null && selectedCurrency !== 'XEC';
 
     return (
         <>
             <Modal
                 title="Confirm Send"
                 open={isModalVisible}
                 onOk={handleOk}
                 onCancel={handleCancel}
             >
                 <p>
                     {isOneToManyXECSend ? (
                         <>
                             Are you sure you want to send{' '}
                             {sumOneToManyXec(
                                 formData.address.split('\n'),
                             ).toLocaleString(userLocale, {
                                 maximumFractionDigits: 2,
                             })}{' '}
                             XEC in the following transaction?
                             <br />
                             <br />
                             {formData.address}
                         </>
                     ) : (
                         `Are you sure you want to send ${formData.value}${' '}
                   ${selectedCurrency} to ${
                             queryStringText === null
                                 ? formData.address
                                 : formData.address.slice(
                                       0,
                                       formData.address.indexOf('?'),
                                   )
                         }?`
                     )}
                 </p>
             </Modal>
             <WalletInfoCtn>
                 <WalletLabel
                     name={wallet.name}
                     cashtabSettings={cashtabSettings}
                     changeCashtabSettings={changeCashtabSettings}
                 ></WalletLabel>
                 {!balances.totalBalance ? (
                     <ZeroBalanceHeader>
                         You currently have 0 {appConfig.ticker}
                         <br />
                         Deposit some funds to use this feature
                     </ZeroBalanceHeader>
                 ) : (
                     <>
                         <BalanceHeader
                             balance={balances.totalBalance}
                             ticker={appConfig.ticker}
                             cashtabSettings={cashtabSettings}
                         />
 
                         <BalanceHeaderFiat
                             balance={balances.totalBalance}
                             settings={cashtabSettings}
                             fiatPrice={fiatPrice}
                         />
                     </>
                 )}
             </WalletInfoCtn>
             <SidePaddingCtn>
                 <Row type="flex">
                     <Col span={24}>
                         <Form
                             style={{
                                 width: 'auto',
                                 marginTop: '40px',
                             }}
                         >
                             <SendAddressHeader>
                                 {' '}
                                 <FormLabel>Send to</FormLabel>
                                 <TextAreaLabel>
                                     Multiple Recipients:&nbsp;&nbsp;
                                     <Switch
                                         defaultunchecked="true"
                                         checked={isOneToManyXECSend}
                                         onChange={() => {
                                             setIsOneToManyXECSend(
                                                 !isOneToManyXECSend,
                                             );
                                             // Do not persist multisend input to single send and vice versa
                                             clearInputForms();
                                         }}
                                         style={{
                                             marginBottom: '7px',
                                         }}
                                     />
                                 </TextAreaLabel>
                             </SendAddressHeader>
                             <ExpandingAddressInputCtn open={isOneToManyXECSend}>
                                 <SendInputCtn>
                                     <DestinationAddressSingleCtn>
                                         <DestinationAddressSingle
                                             style={{ marginBottom: '0px' }}
                                             loadWithCameraOpen={
                                                 location &&
                                                 location.state &&
                                                 location.state.replyAddress
                                                     ? false
                                                     : scannerSupported
                                             }
                                             validateStatus={
                                                 sendBchAddressError
                                                     ? 'error'
                                                     : ''
                                             }
                                             help={
                                                 sendBchAddressError
                                                     ? sendBchAddressError
                                                     : ''
                                             }
                                             onScan={result =>
                                                 handleAddressChange({
                                                     target: {
                                                         name: 'address',
                                                         value: result,
                                                     },
                                                 })
                                             }
                                             inputProps={{
                                                 placeholder:
                                                     aliasSettings.aliasEnabled
                                                         ? `Address or Alias`
                                                         : `Address`,
                                                 name: 'address',
                                                 onChange: e =>
                                                     handleAddressChange(e),
                                                 required: true,
                                                 value: formData.address,
                                             }}
                                         ></DestinationAddressSingle>
                                         <AliasAddressPreviewLabel>
                                             <TxLink
                                                 key={aliasInputAddress}
                                                 href={`${explorer.blockExplorerUrl}/address/${aliasInputAddress}`}
                                                 target="_blank"
                                                 rel="noreferrer"
                                             >
                                                 {aliasInputAddress &&
                                                     `${aliasInputAddress.slice(
                                                         0,
                                                         10,
                                                     )}...${aliasInputAddress.slice(
                                                         -5,
                                                     )}`}
                                             </TxLink>
                                         </AliasAddressPreviewLabel>
                                         <FormLabel>Amount</FormLabel>
                                         <SendBchInput
                                             activeFiatCode={
                                                 cashtabSettings &&
                                                 cashtabSettings.fiatCurrency
                                                     ? cashtabSettings.fiatCurrency.toUpperCase()
                                                     : 'USD'
                                             }
                                             validateStatus={
                                                 sendBchAmountError
                                                     ? 'error'
                                                     : ''
                                             }
                                             help={
                                                 sendBchAmountError
                                                     ? sendBchAmountError
                                                     : ''
                                             }
                                             onMax={onMax}
                                             inputProps={{
                                                 name: 'value',
                                                 dollar:
                                                     selectedCurrency === 'USD'
                                                         ? 1
                                                         : 0,
                                                 placeholder: 'Amount',
                                                 onChange: e =>
                                                     handleBchAmountChange(e),
                                                 required: true,
                                                 value: formData.value,
                                                 disabled: priceApiError,
                                             }}
                                             selectProps={{
                                                 value: selectedCurrency,
                                                 disabled:
                                                     queryStringText !== null,
                                                 onChange: e =>
                                                     handleSelectedCurrencyChange(
                                                         e,
                                                     ),
                                             }}
                                         ></SendBchInput>
                                     </DestinationAddressSingleCtn>
                                     {priceApiError && (
                                         <AlertMsg>
                                             Error fetching fiat price. Setting
                                             send by{' '}
                                             {supportedFiatCurrencies[
                                                 cashtabSettings.fiatCurrency
                                             ].slug.toUpperCase()}{' '}
                                             disabled
                                         </AlertMsg>
                                     )}
                                 </SendInputCtn>
 
                                 <>
                                     <DestinationAddressMultiCtn>
                                         <DestinationAddressMulti
                                             validateStatus={
                                                 sendBchAddressError
                                                     ? 'error'
                                                     : ''
                                             }
                                             help={
                                                 sendBchAddressError
                                                     ? sendBchAddressError
                                                     : ''
                                             }
                                             inputProps={{
                                                 placeholder: `One address & value per line, separated by comma \ne.g. \necash:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gkpyrlwu8,500 \necash:qzvydd4n3lm3xv62cx078nu9rg0e3srmqq0knykfed,700`,
                                                 name: 'address',
                                                 onChange: e =>
                                                     handleMultiAddressChange(e),
                                                 required: true,
                                                 value: formData.address,
                                             }}
                                         ></DestinationAddressMulti>
                                     </DestinationAddressMultiCtn>
                                 </>
                                 <AmountPreviewCtn>
                                     {!priceApiError && !isOneToManyXECSend && (
                                         <>
                                             <LocaleFormattedValue>
                                                 {!isNaN(formData.value)
                                                     ? formatBalance(
                                                           formData.value,
                                                           userLocale,
                                                       ) +
                                                       ' ' +
                                                       selectedCurrency
                                                     : ''}
                                             </LocaleFormattedValue>
                                             <ConvertAmount>
                                                 {fiatPriceString !== '' && '='}{' '}
                                                 {fiatPriceString}
                                             </ConvertAmount>
                                         </>
                                     )}
                                 </AmountPreviewCtn>
                             </ExpandingAddressInputCtn>
 
                             <div
                                 style={{
                                     paddingTop: '1rem',
                                 }}
                             >
                                 {!balances.totalBalance ||
                                 apiError ||
                                 sendBchAmountError ||
                                 sendBchAddressError ||
                                 isMsgError ||
                                 priceApiError ||
                                 (!isOneToManyXECSend &&
                                     isNaN(formData.value)) ? (
                                     <DisabledButton>Send</DisabledButton>
                                 ) : (
                                     <>
                                         {txInfoFromUrl ? (
                                             <PrimaryButton
                                                 onClick={() =>
                                                     checkForConfirmationBeforeSendXec()
                                                 }
                                             >
                                                 Send
                                             </PrimaryButton>
                                         ) : (
                                             <PrimaryButton
                                                 onClick={() => {
                                                     checkForConfirmationBeforeSendXec();
                                                 }}
                                             >
                                                 Send
                                             </PrimaryButton>
                                         )}
                                     </>
                                 )}
                             </div>
 
                             <CustomCollapseCtn
                                 panelHeader={
                                     <PanelHeaderCtn>
                                         <ThemedMailOutlined /> Message
                                     </PanelHeaderCtn>
                                 }
                                 optionalDefaultActiveKey={
                                     location &&
                                     location.state &&
                                     location.state.replyAddress
                                         ? ['1']
                                         : ['0']
                                 }
                                 optionalKey="1"
                             >
                                 <AntdFormWrapper
                                     style={{
                                         marginBottom: '20px',
                                     }}
                                 >
                                     <Alert
                                         style={{
                                             marginBottom: '10px',
                                         }}
                                         description="Please note this message will be public."
                                         type="warning"
                                         showIcon
                                     />
                                     <TextArea
                                         name="opReturnMsg"
                                         placeholder={
                                             location &&
                                             location.state &&
                                             location.state.airdropTokenId
                                                 ? `(max ${
                                                       opreturnConfig.cashtabMsgByteLimit -
                                                       localAirdropTxAddedBytes
                                                   } bytes)`
                                                 : `(max ${opreturnConfig.cashtabMsgByteLimit} bytes)`
                                         }
                                         value={opReturnMsg ? opReturnMsg : ''}
                                         onChange={e => handleMsgChange(e)}
                                         onKeyDown={e =>
                                             e.keyCode == 13
                                                 ? e.preventDefault()
                                                 : ''
                                         }
                                     />
                                     <MsgBytesizeError>
                                         {isMsgError ? isMsgError : ''}
                                     </MsgBytesizeError>
                                 </AntdFormWrapper>
                             </CustomCollapseCtn>
                             {apiError && <ApiError />}
                         </Form>
                     </Col>
                 </Row>
             </SidePaddingCtn>
         </>
     );
 };
 
 /*
 passLoadingStatus must receive a default prop that is a function
 in order to pass the rendering unit test in Send.test.js
 
 status => {console.log(status)} is an arbitrary stub function
 */
 
 SendBCH.defaultProps = {
     passLoadingStatus: status => {
         console.log(status);
     },
 };
 
 SendBCH.propTypes = {
     passLoadingStatus: PropTypes.func,
 };
 
 export default SendBCH;
diff --git a/cashtab/src/opreturn/__tests__/index.test.js b/cashtab/src/opreturn/__tests__/index.test.js
index f8a9c3b66..4d5766a7a 100644
--- a/cashtab/src/opreturn/__tests__/index.test.js
+++ b/cashtab/src/opreturn/__tests__/index.test.js
@@ -1,112 +1,133 @@
 // Copyright (c) 2023 The Bitcoin developers
 // Distributed under the MIT software license, see the accompanying
 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
 import {
     getCashtabMsgTargetOutput,
     getAirdropTargetOutput,
     getAliasTargetOutput,
     getAliasByteCount,
+    getCashtabMsgByteCount,
 } from 'opreturn';
 import { opReturnVectors } from '../fixtures/vectors';
 
 describe('Cashtab Msg building functions', () => {
     const { expectedReturns, expectedErrors } = opReturnVectors.cashtabMsgs;
 
     // Successfully created targetOutputs
     expectedReturns.forEach(expectedReturn => {
         const { description, cashtabMsg, outputScriptHex } = expectedReturn;
         it(`getCashtabMsgTargetOutput: ${description}`, () => {
             const targetOutput = getCashtabMsgTargetOutput(cashtabMsg);
             // Output value should be zero for OP_RETURN
             expect(targetOutput.value).toStrictEqual(0);
             // Test vs hex string as cannot store buffer type in vectors
             expect(targetOutput.script.toString('hex')).toStrictEqual(
                 outputScriptHex,
             );
         });
     });
     // Error cases
     expectedErrors.forEach(expectedError => {
         const { description, cashtabMsg, errorMsg } = expectedError;
         it(`getCashtabMsgTargetOutput throws error for: ${description}`, () => {
             expect(() => getCashtabMsgTargetOutput(cashtabMsg)).toThrow(
                 errorMsg,
             );
         });
     });
 });
 
 describe('getAirdropTargetOutput', () => {
     const { expectedReturns, expectedErrors } = opReturnVectors.airdrops;
 
     // Successfully created targetOutputs
     expectedReturns.forEach(expectedReturn => {
         const { description, tokenId, airdropMsg, outputScriptHex } =
             expectedReturn;
         it(`${description}`, () => {
             const targetOutput = getAirdropTargetOutput(tokenId, airdropMsg);
             // Output value should be zero for OP_RETURN
             expect(targetOutput.value).toStrictEqual(0);
             // Test vs hex string as cannot store buffer type in vectors
             expect(targetOutput.script.toString('hex')).toStrictEqual(
                 outputScriptHex,
             );
         });
     });
     // Error cases
     expectedErrors.forEach(expectedError => {
         const { description, tokenId, airdropMsg, errorMsg } = expectedError;
         it(`getAirdropTargetOutput throws error for: ${description}`, () => {
             expect(() => getAirdropTargetOutput(tokenId, airdropMsg)).toThrow(
                 errorMsg,
             );
         });
     });
 });
 
 describe('Alias registration target output building functions', () => {
     const { expectedReturns, expectedErrors } =
         opReturnVectors.aliasRegistrations;
 
     // Successfully created targetOutputs
     expectedReturns.forEach(expectedReturn => {
         const { description, alias, address, outputScriptHex } = expectedReturn;
         it(`getAliasTargetOutput: ${description}`, () => {
             const targetOutput = getAliasTargetOutput(alias, address);
             // Output value should be zero for OP_RETURN
             expect(targetOutput.value).toStrictEqual(0);
             // Test vs hex string as cannot store buffer type in vectors
             expect(targetOutput.script.toString('hex')).toStrictEqual(
                 outputScriptHex,
             );
         });
     });
     // Error cases
     expectedErrors.forEach(expectedError => {
         const { description, alias, address, errorMsg } = expectedError;
         it(`getAliasTargetOutput throws error for: ${description}`, () => {
             expect(() => getAliasTargetOutput(alias, address)).toThrow(
                 errorMsg,
             );
         });
     });
 });
 
 describe('Determines byte count of user input alias registrations', () => {
     const { expectedReturns, expectedErrors } = opReturnVectors.aliasByteCounts;
 
     // Successfully calculates alias bytecounts
     expectedReturns.forEach(expectedReturn => {
         const { description, alias, byteCount } = expectedReturn;
         it(`getAliasByteCount: ${description}`, () => {
             expect(getAliasByteCount(alias)).toBe(byteCount);
         });
     });
     // Error cases
     expectedErrors.forEach(expectedError => {
         const { description, alias, errorMsg } = expectedError;
         it(`getAliasByteCount throws error for: ${description}`, () => {
             expect(() => getAliasByteCount(alias)).toThrow(errorMsg);
         });
     });
 });
+
+describe('Determines bytecount of user input Cashtab Msg', () => {
+    const { expectedReturns, expectedErrors } =
+        opReturnVectors.cashtabMsgByteCounts;
+
+    // Successfully calculates Cashtab Msg bytecounts
+    expectedReturns.forEach(expectedReturn => {
+        const { description, cashtabMsg, byteCount } = expectedReturn;
+        it(`getCashtabMsgByteCount: ${description}`, () => {
+            expect(getCashtabMsgByteCount(cashtabMsg)).toBe(byteCount);
+        });
+    });
+    // Error cases
+    expectedErrors.forEach(expectedError => {
+        const { description, cashtabMsg, errorMsg } = expectedError;
+        it(`getCashtabMsgByteCount throws error for: ${description}`, () => {
+            expect(() => getCashtabMsgByteCount(cashtabMsg)).toThrow(errorMsg);
+        });
+    });
+});
diff --git a/cashtab/src/opreturn/fixtures/vectors.js b/cashtab/src/opreturn/fixtures/vectors.js
index b0df53555..6d2f1a48a 100644
--- a/cashtab/src/opreturn/fixtures/vectors.js
+++ b/cashtab/src/opreturn/fixtures/vectors.js
@@ -1,206 +1,254 @@
 // Copyright (c) 2023 The Bitcoin developers
 // Distributed under the MIT software license, see the accompanying
 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
 import { opReturn } from 'config/opreturn';
 
 // Test vectors for opreturn output generating functions
 
 export const opReturnVectors = {
     cashtabMsgs: {
         expectedReturns: [
             {
                 description: 'Alphanumeric string',
                 cashtabMsg: 'This is a Cashtab Msg',
                 outputScriptHex:
                     '6a0400746162155468697320697320612043617368746162204d7367',
             },
             {
                 description: 'String with emojis',
                 cashtabMsg: '๐Ÿ™๐Ÿ“ฌ๐Ÿซก๐Ÿ‘€๐Ÿ•ต๏ธ๐Ÿ‘‘๐ŸŽƒ๐Ÿช–๐Ÿ‹๐ŸŽฏ',
                 outputScriptHex:
                     '6a04007461622bf09f998ff09f93acf09faba1f09f9180f09f95b5efb88ff09f9191f09f8e83f09faa96f09f908bf09f8eaf',
             },
             {
                 description: 'String of max length for Cashtab Msg',
                 cashtabMsg:
                     '00000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000112345',
                 outputScriptHex:
                     '6a04007461624cd73030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313132333435',
             },
         ],
         expectedErrors: [
             {
                 description:
                     'String exceeding max length for Cashtab Msg by 1 byte',
                 cashtabMsg:
                     '000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001123456',
                 errorMsg: `Cashtab msg is 216 bytes. Exceeds ${opReturn.cashtabMsgByteLimit} byte limit.`,
             },
             {
                 description: 'non-string input',
                 cashtabMsg: { cashtabMsg: 'good to go' },
                 errorMsg: 'getCashtabMsgTargetOutput requires string input',
             },
             {
                 description: 'Empty string',
                 cashtabMsg: '',
                 errorMsg: 'Cashtab Msg cannot be an empty string',
             },
         ],
     },
     airdrops: {
         expectedReturns: [
             {
                 description: 'Airdrop with no optional msg',
                 tokenId:
                     '50d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg: '',
                 outputScriptHex:
                     '6a0464726f702050d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
             },
             {
                 description:
                     'Airdrop with many spaces for optional msg treated as no optional msg',
                 tokenId:
                     '50d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg: '          ',
                 outputScriptHex:
                     '6a0464726f702050d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
             },
             {
                 description: 'Airdrop with optional alphanumeric msg',
                 tokenId:
                     '50d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg: 'Test airdrop msg',
                 outputScriptHex:
                     '6a0464726f702050d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e10546573742061697264726f70206d7367',
             },
             {
                 description:
                     'Airdrop with optional emoji and special characters msg',
                 tokenId:
                     '50d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg:
                     '30~40 ํ”„๋กœ ์ƒ์Šน์œผ๋กœ ๋งŒ์กฑ๋ชปํ•˜๊ฒ ์œผ๋‹ˆ 300~400ํ”„๋กœ ํŽŒํ•‘ ํ•จ ๊ฐ€์ฆˆ์•„~ ์‹œ์ฒด๋ฐญ๋„˜๊ณ ~๐Ÿค”',
                 outputScriptHex:
                     '6a0464726f702050d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e4c6833307e343020ed9484eba19c20ec8381ec8ab9ec9cbceba19c20eba78ceca1b1ebaabbed9598eab2a0ec9cbceb8b88203330307e343030ed9484eba19c20ed8e8ced959120ed95a820eab080eca688ec95847e20ec8b9cecb2b4ebb0adeb8498eab3a07ef09fa494',
             },
             {
                 description:
                     'Airdrop with optional msg of max allowable length',
                 tokenId:
                     '50d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg:
                     '00000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000112',
                 outputScriptHex:
                     '6a0464726f702050d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e4cb63030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313030303030303030303130303030303030303031303030303030303030313132',
             },
         ],
         expectedErrors: [
             {
                 description: 'Invalid tokenId provided',
                 tokenId:
                     '0d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg:
                     '000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001123',
                 errorMsg: `Invalid tokenId: 0d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e`,
             },
             {
                 description:
                     'Airdrop msg exceeding max length for airdrop msg by 1 byte',
                 tokenId:
                     '50d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg:
                     '000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001123',
                 errorMsg: `Airdrop msg is 183 bytes. Exceeds ${opReturn.airdropMsgByteLimit} byte limit.`,
             },
             {
                 description: 'non-string input for airdrop msg',
                 tokenId:
                     '50d8292c6255cda7afc6c8566fed3cf42a2794e9619740fe8f4c95431271410e',
                 airdropMsg: { airdropMsg: 'good to go' },
                 errorMsg:
                     'getAirdropTargetOutput requires string input for tokenId and airdropMsg',
             },
         ],
     },
     aliasRegistrations: {
         expectedReturns: [
             {
                 description: 'Valid alias to p2pkh address',
                 alias: 'test',
                 address: 'ecash:qz2708636snqhsxu8wnlka78h6fdp77ar59jrf5035',
                 outputScriptHex:
                     '6a042e786563000474657374150095e79f51d4260bc0dc3ba7fb77c7be92d0fbdd1d',
             },
             {
                 description: 'Valid alias to p2sh address',
                 alias: 'testtwo',
                 address: 'ecash:prfhcnyqnl5cgrnmlfmms675w93ld7mvvqd0y8lz07',
                 outputScriptHex:
                     '6a042e78656300077465737474776f1508d37c4c809fe9840e7bfa77b86bd47163f6fb6c60',
             },
         ],
         expectedErrors: [
             {
                 description: 'Invalid alias',
                 alias: 'test_WITH_badchars',
                 address: 'ecash:qz2708636snqhsxu8wnlka78h6fdp77ar59jrf5035',
                 errorMsg: 'Invalid alias "test_WITH_badchars"',
             },
             {
                 description: 'Invalid address',
                 alias: 'test',
                 address: 'not an address',
                 errorMsg: 'Invalid address "not an address"',
             },
         ],
     },
     aliasByteCounts: {
         expectedReturns: [
             { description: 'Alias with emoji', alias: '๐Ÿ™ˆ', byteCount: 4 },
             {
                 description: 'Alias with emoji and text',
                 alias: 'monkey๐Ÿ™ˆ',
                 byteCount: 10,
             },
             {
                 description: 'Alias with special characters',
                 alias: 'monkeyยฉยฎส•โ€ขฬแดฅโ€ขฬ€ส”ใฃโ™ก',
                 byteCount: 33,
             },
             {
                 description: 'Alias with Korean text',
                 alias: '์†Œ์ฃผ',
                 byteCount: 6,
             },
             {
                 description: 'Alias with Arabic text',
                 alias: 'ู…ุญูŠุท',
                 byteCount: 8,
             },
             {
                 description: 'Alias with Chinese text',
                 alias: 'ๅ†ฐๆท‡ๆท‹',
                 byteCount: 9,
             },
             {
                 description: 'Alias with mixed foreign alphabets and emoji',
                 alias: '๐Ÿ™ˆยฉๅ†ฐ์†Œ์ฃผ',
                 byteCount: 15,
             },
             {
                 description: 'Alphanumeric valid v0 alias',
                 alias: 'justanormalalias',
                 byteCount: 16,
             },
         ],
         expectedErrors: [
             {
                 description: 'non-text input',
                 alias: null,
                 errorMsg: 'alias input must be a string',
             },
         ],
     },
+    cashtabMsgByteCounts: {
+        expectedReturns: [
+            {
+                description: 'a single emoji',
+                cashtabMsg: '๐Ÿ™ˆ',
+                byteCount: 4,
+            },
+            {
+                description: 'msg input with characters and emojis',
+                cashtabMsg: 'monkey๐Ÿ™ˆ',
+                byteCount: 10,
+            },
+            {
+                description: 'msg input with special characters',
+                cashtabMsg: 'monkeyยฉยฎส•โ€ขฬแดฅโ€ขฬ€ส”ใฃโ™ก',
+                byteCount: 33,
+            },
+            {
+                description:
+                    'msg input with a mixture of symbols, multilingual characters and emojis',
+                cashtabMsg: '๐Ÿ™ˆยฉๅ†ฐ์†Œ์ฃผ',
+                byteCount: 15,
+            },
+            {
+                description: 'Alphanumeric string',
+                cashtabMsg: 'This is a Cashtab Msg',
+                byteCount: 21,
+            },
+            {
+                description: 'String with emojis',
+                cashtabMsg: '๐Ÿ™๐Ÿ“ฌ๐Ÿซก๐Ÿ‘€๐Ÿ•ต๏ธ๐Ÿ‘‘๐ŸŽƒ๐Ÿช–๐Ÿ‹๐ŸŽฏ',
+                byteCount: 43,
+            },
+            {
+                description: 'String of max length for Cashtab Msg',
+                cashtabMsg:
+                    '00000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000100000000010000000001000000000112345',
+                byteCount: 215,
+            },
+        ],
+        expectedErrors: [
+            {
+                description: 'non-text input',
+                cashtabMsg: null,
+                errorMsg: 'cashtabMsg must be a string',
+            },
+        ],
+    },
 };
diff --git a/cashtab/src/opreturn/index.js b/cashtab/src/opreturn/index.js
index a20dea6b0..70f41ddb3 100644
--- a/cashtab/src/opreturn/index.js
+++ b/cashtab/src/opreturn/index.js
@@ -1,183 +1,200 @@
 // Copyright (c) 2023 The Bitcoin developers
 // Distributed under the MIT software license, see the accompanying
 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
 
 import * as utxolib from '@bitgo/utxo-lib';
 import cashaddr from 'ecashaddrjs';
 import { opReturn } from 'config/opreturn';
 import { isValidTokenId, isValidAlias } from 'utils/validation';
 
 /**
  * Initialize an OP_RETURN script element in a way that utxolib.script.compile(script) accepts
  * utxolib.script.compile(script) will add pushdata bytes for each buffer
  * utxolib.script.compile(script) will not add pushdata bytes for raw data
  * Initialize script array with OP_RETURN byte (6a) as rawdata (i.e. you want compiled result of 6a, not 016a)
  */
 const initializeScript = () => {
     return [opReturn.opReturnPrefixDec];
 };
 
 /**
  * Get targetOutput for a Cashtab Msg from user input string
  * @param {string} cashtabMsg string
  * @throws {error} if msg exceeds opReturnByteLimit of 223 or invalid input
  * @returns {object} targetOutput, e.g. {value: 0, script: <encoded cashtab msg>}
  */
 export const getCashtabMsgTargetOutput = cashtabMsg => {
     if (typeof cashtabMsg !== 'string') {
         throw new Error('getCashtabMsgTargetOutput requires string input');
     }
     if (cashtabMsg === '') {
         throw new Error('Cashtab Msg cannot be an empty string');
     }
 
     let script = initializeScript();
 
     // Push Cashtab Msg protocol identifier
     script.push(Buffer.from(opReturn.appPrefixesHex.cashtab, 'hex'));
 
     // Cashtab msgs are utf8 encoded
     const cashtabMsgScript = Buffer.from(cashtabMsg, 'utf8');
     const cashtabMsgByteCount = cashtabMsgScript.length;
 
     if (cashtabMsgByteCount > opReturn.cashtabMsgByteLimit) {
         throw new Error(
             `Error: Cashtab msg is ${cashtabMsgByteCount} bytes. Exceeds ${opReturn.cashtabMsgByteLimit} byte limit.`,
         );
     }
 
     script.push(cashtabMsgScript);
 
     script = utxolib.script.compile(script);
 
     // Create output
     return { value: 0, script };
 };
 
+/**
+ * Calculates the bytecount of a Cashtab Msg as part of an OP_RETURN output
+ * Used to validate user input in Send.js
+ *
+ * @param {string} cashtabMsg alias input from a text input field
+ * @returns {number} aliasInputByteSize the byte size of the alias input
+ */
+export const getCashtabMsgByteCount = cashtabMsg => {
+    if (typeof cashtabMsg !== 'string') {
+        throw new Error('cashtabMsg must be a string');
+    }
+
+    // Cashtab msgs are utf8 encoded
+    const cashtabMsgScript = Buffer.from(cashtabMsg, 'utf8');
+    return cashtabMsgScript.length;
+};
+
 /**
  * Get targetOutput for an Airdrop tx OP_RETURN from token id and optional user message
  * Airdrop tx spec: <Airdrop Protocol Identifier> <tokenId> <optionalMsg>
  * @param {string} tokenId tokenId of the token receiving this airdrop tx
  * @param {string} airdropMsg optional brief msg accompanying the airdrop
  * @throws {error} if msg exceeds opReturnByteLimit of 223 or invalid input
  * @returns {object} targetOutput, e.g. {value: 0, script: <encoded airdrop msg>}
  */
 export const getAirdropTargetOutput = (tokenId, airdropMsg = '') => {
     if (!isValidTokenId(tokenId)) {
         throw new Error(`Invalid tokenId: ${tokenId}`);
     }
     if (typeof airdropMsg !== 'string') {
         throw new Error(
             'getAirdropTargetOutput requires string input for tokenId and airdropMsg',
         );
     }
 
     let script = initializeScript();
 
     // Push Airdrop protocol identifier
     script.push(Buffer.from(opReturn.appPrefixesHex.airdrop, 'hex'));
 
     // add the airdrop token ID to script
     script.push(Buffer.from(tokenId, 'hex'));
 
     if (airdropMsg.trim() !== '') {
         // Cashtab msgs are utf8 encoded
         const airdropMsgScript = Buffer.from(airdropMsg, 'utf8');
         const airdropMsgByteCount = airdropMsgScript.length;
 
         if (airdropMsgByteCount > opReturn.airdropMsgByteLimit) {
             throw new Error(
                 `Error: Airdrop msg is ${airdropMsgByteCount} bytes. Exceeds ${opReturn.airdropMsgByteLimit} byte limit.`,
             );
         }
 
         script.push(airdropMsgScript);
     }
 
     script = utxolib.script.compile(script);
 
     // Create output
     return { value: 0, script };
 };
 
 /**
  * Generate an OP_RETURN targetOutput for use in broadcasting a v0 alias registration
  *
  * @param {string} alias
  * @param {string} address
  * @throws {error} validation errors on alias or address
  * @returns {object} targetOutput ready for transaction building, see sendXec function at src/transactions
  */
 export const getAliasTargetOutput = (alias, address) => {
     if (!isValidAlias(alias)) {
         throw new Error(`Invalid alias "${alias}"`);
     }
 
     let script = initializeScript();
 
     // Push alias protocol identifier
     script.push(
         Buffer.from(opReturn.appPrefixesHex.aliasRegistration, 'hex'), // '.xec'
     );
 
     // Push alias protocol tx version to stack
     // Per spec, push this as OP_0
     script.push(0);
 
     // Push alias to the stack
     script.push(Buffer.from(alias, 'utf8'));
 
     // Get the type and hash of the address in string format
     let decoded;
     try {
         decoded = cashaddr.decode(address, true);
     } catch (err) {
         throw new Error(`Invalid address "${address}"`);
     }
     const { type, hash } = decoded;
 
     // Determine address type and corresponding address version byte
     let addressVersionByte;
     // Version bytes per cashaddr spec,https://github.com/bitcoincashorg/bitcoincash.org/blob/master/spec/cashaddr.md
     if (type === 'p2pkh') {
         addressVersionByte = '00'; // one byte 0 in hex
     } else if (type === 'p2sh') {
         addressVersionByte = '08'; // one byte 8 in hex
     } else {
         throw new Error(
             `Unsupported address type ${type}. Only p2pkh and p2sh addresses are supported.`,
         );
     }
 
     // Push <addressVersionByte> and <addressPayload>
     script.push(Buffer.from(`${addressVersionByte}${hash}`, 'hex'));
 
     script = utxolib.script.compile(script);
 
     // Create output
     return { value: 0, script };
 };
 
 /**
  * Calculates the bytecount of the alias input
  *
  * @param {string} alias alias input from a text input field
  * @returns {number} aliasInputByteSize the byte size of the alias input
  */
 export const getAliasByteCount = alias => {
     if (typeof alias !== 'string') {
         // Make sure .trim() is available
         throw new Error('alias input must be a string');
     }
     // Do not validate the specific alias as the user may type in invalid aliases
     // We still want to return a size
     if (alias.trim() === '') {
         return 0;
     }
 
     // Get alias as utf8
     const aliasUtf8Hex = Buffer.from(alias, 'utf8');
 
     // Return bytecount
     return aliasUtf8Hex.length;
 };
diff --git a/cashtab/src/utils/__tests__/cashMethods.test.js b/cashtab/src/utils/__tests__/cashMethods.test.js
index d0edf60a5..1f92bdef5 100644
--- a/cashtab/src/utils/__tests__/cashMethods.test.js
+++ b/cashtab/src/utils/__tests__/cashMethods.test.js
@@ -1,1886 +1,1644 @@
 import BigNumber from 'bignumber.js';
 import * as utxolib from '@bitgo/utxo-lib';
 import cashaddr from 'ecashaddrjs';
 import {
     fromSatoshisToXec,
     flattenContactList,
     loadStoredWallet,
     isValidStoredWallet,
     fromLegacyDecimals,
     convertToEcashPrefix,
     isLegacyMigrationRequired,
     convertEtokenToEcashAddr,
     parseOpReturn,
     convertEcashtoEtokenAddr,
     getHashArrayFromWallet,
     isActiveWebsocket,
     getChangeAddressFromInputUtxos,
     generateAliasOpReturnScript,
-    generateOpReturnScript,
     generateTxInput,
     generateTxOutput,
     generateTokenTxInput,
     signAndBuildTx,
     fromXecToSatoshis,
     getWalletBalanceFromUtxos,
     signUtxosByAddress,
     generateTokenTxOutput,
     getCashtabByteCount,
     calcFee,
     toHash160,
     generateGenesisOpReturn,
     generateSendOpReturn,
     generateBurnOpReturn,
     hash160ToAddress,
     outputScriptToAddress,
-    getMessageByteSize,
     parseAddressForParams,
     sumOneToManyXec,
     toSatoshis,
 } from 'utils/cashMethods';
 import { validAddressArrayInput } from '../__mocks__/mockAddressArray';
 import {
     mockGenesisOpReturnScript,
     mockSendOpReturnScript,
     mockSendOpReturnTokenUtxos,
     mockBurnOpReturnScript,
     mockBurnOpReturnTokenUtxos,
 } from '../__mocks__/mockOpReturnScript';
 import {
     cachedUtxos,
     utxosLoadedFromCache,
 } from '../__mocks__/mockCachedUtxos';
 import {
     validStoredWallet,
     invalidStoredWallet,
     invalidpreChronikStoredWallet,
     validStoredWalletAfter20221123Streamline,
     invalidStoredWalletMissingPath1899AndMnemonic,
 } from '../__mocks__/mockStoredWallets';
 
 import {
     missingPath1899Wallet,
     missingPublicKeyInPath1899Wallet,
     missingPublicKeyInPath145Wallet,
     missingPublicKeyInPath245Wallet,
     notLegacyWallet,
     missingHash160,
 } from '../__mocks__/mockLegacyWalletsUtils';
 
 import {
     shortCashtabMessageInputHex,
     longCashtabMessageInputHex,
     shortExternalMessageInputHex,
     longExternalMessageInputHex,
     shortSegmentedExternalMessageInputHex,
     longSegmentedExternalMessageInputHex,
     mixedSegmentedExternalMessageInputHex,
     mockParsedShortCashtabMessageArray,
     mockParsedLongCashtabMessageArray,
     mockParsedShortExternalMessageArray,
     mockParsedLongExternalMessageArray,
     mockParsedShortSegmentedExternalMessageArray,
     mockParsedLongSegmentedExternalMessageArray,
     mockParsedMixedSegmentedExternalMessageArray,
     eTokenInputHex,
     mockParsedETokenOutputArray,
     mockAirdropHexOutput,
     mockParsedAirdropMessageArray,
 } from '../__mocks__/mockOpReturnParsedArray';
 
 import mockLegacyWallets from 'hooks/__mocks__/mockLegacyWallets';
 import sendBCHMock from '../__mocks__/sendBCH';
 import {
     activeWebsocketAlpha,
     disconnectedWebsocketAlpha,
     unsubscribedWebsocket,
 } from '../__mocks__/chronikWs';
 import mockNonSlpUtxos from '../../hooks/__mocks__/mockNonSlpUtxos';
 import mockSlpUtxos from '../../hooks/__mocks__/mockSlpUtxos';
 import {
     mockOneToOneSendXecTxBuilderObj,
     mockOneToManySendXecTxBuilderObj,
     mockCreateTokenOutputsTxBuilderObj,
     mockSendTokenOutputsTxBuilderObj,
     mockBurnTokenOutputsTxBuilderObj,
     mockCreateTokenTxBuilderObj,
     mockSendTokenTxBuilderObj,
     mockBurnTokenTxBuilderObj,
 } from '../__mocks__/mockTxBuilderObj';
 import {
     mockSingleInputUtxo,
     mockMultipleInputUtxos,
     mockSingleOutput,
     mockMultipleOutputs,
 } from '../__mocks__/mockTxBuilderData';
 import createTokenMock from '../__mocks__/createToken';
 import { opReturn as opreturnConfig } from 'config/opreturn';
 import appConfig from 'config/app';
 
 test('parseAddressForParams() returns valid info for query string based input', () => {
     const inputString =
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx?amount=500000';
     const expectedObject = {
         address: 'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx',
         amount: '500000',
         queryString: 'amount=500000',
     };
     const addressInfo = parseAddressForParams(inputString);
     expect(addressInfo).toStrictEqual(expectedObject);
 });
 
 test('parseAddressForParams() returns no amount for a malformed query string input', () => {
     const inputString =
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx?*&@^&%@amount=-500000';
     const expectedObject = {
         address: 'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx',
         amount: null,
         queryString: '*&@^&%@amount=-500000',
     };
     const addressInfo = parseAddressForParams(inputString);
     expect(addressInfo).toStrictEqual(expectedObject);
 });
 
 test('parseAddressForParams() returns valid address info for a non-query string based input', () => {
     const inputString = 'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx';
     const expectedObject = {
         address: 'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx',
         amount: null,
         queryString: null,
     };
     const addressInfo = parseAddressForParams(inputString);
     expect(addressInfo).toStrictEqual(expectedObject);
 });
 
 test('parseAddressForParams() returns valid address info for a valid prefix-less eCash address', () => {
     const inputString = 'qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx';
     const expectedObject = {
         address: 'qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx',
         amount: null,
         queryString: null,
     };
     const addressInfo = parseAddressForParams(inputString);
     expect(addressInfo).toStrictEqual(expectedObject);
 });
 
-it(`OP_RETURN msg byte length matches for an encrypted msg input with a single emoji`, () => {
-    const msgInput = '๐Ÿ™ˆ';
-    const encryptedEjMock = {
-        type: 'Buffer',
-        data: [
-            2, 241, 30, 211, 127, 184, 181, 145, 219, 158, 127, 99, 178, 221,
-            90, 234, 194, 108, 152, 147, 60, 77, 74, 176, 112, 249, 23, 170,
-            186, 204, 20, 209, 135, 98, 156, 215, 47, 144, 123, 71, 111, 123,
-            199, 26, 89, 67, 76, 135, 250, 112, 226, 74, 182, 186, 79, 52, 15,
-            88, 214, 142, 141, 145, 103, 89, 66, 158, 107, 191, 144, 255, 139,
-            65, 21, 141, 128, 61, 33, 172, 31, 246, 145, 72, 62, 161, 173, 23,
-            249, 4, 79, 245, 183, 202, 115, 140, 0, 83, 42,
-        ],
-    };
-    const opReturnMsgByteLength = getMessageByteSize(
-        msgInput,
-        true,
-        encryptedEjMock,
-    );
-    expect(opReturnMsgByteLength).toStrictEqual(97);
-});
-
-it(`OP_RETURN msg byte length matches for an encrypted msg input with characters and emojis`, () => {
-    const msgInput = 'monkey๐Ÿ™ˆ';
-    const encryptedEjMock = {
-        type: 'Buffer',
-        data: [
-            2, 74, 145, 240, 12, 210, 143, 66, 224, 155, 246, 106, 238, 186,
-            167, 192, 123, 39, 44, 165, 231, 97, 166, 149, 93, 121, 10, 107, 45,
-            12, 235, 45, 158, 251, 183, 245, 6, 206, 9, 153, 146, 208, 40, 156,
-            106, 3, 140, 137, 68, 126, 240, 70, 87, 131, 54, 91, 115, 164, 223,
-            109, 199, 173, 127, 106, 94, 82, 200, 83, 77, 157, 55, 195, 16, 17,
-            99, 1, 148, 226, 150, 243, 120, 133, 80, 17, 226, 109, 17, 154, 226,
-            59, 203, 36, 203, 230, 236, 12, 104,
-        ],
-    };
-    const opReturnMsgByteLength = getMessageByteSize(
-        msgInput,
-        true,
-        encryptedEjMock,
-    );
-    expect(opReturnMsgByteLength).toStrictEqual(97);
-});
-
-it(`OP_RETURN msg byte length matches for an encrypted msg input with special characters`, () => {
-    const msgInput = 'monkeyยฉยฎส•โ€ขฬแดฅโ€ขฬ€ส”ใฃโ™ก';
-    const encryptedEjMock = {
-        type: 'Buffer',
-        data: [
-            2, 137, 237, 42, 23, 72, 146, 79, 69, 190, 11, 115, 20, 173, 218,
-            99, 121, 188, 45, 14, 219, 135, 46, 91, 165, 121, 166, 149, 100,
-            140, 231, 143, 38, 1, 169, 226, 26, 136, 124, 82, 59, 223, 210, 65,
-            50, 241, 86, 155, 225, 85, 167, 213, 235, 24, 143, 118, 136, 87, 38,
-            161, 153, 18, 110, 198, 168, 196, 77, 250, 255, 2, 132, 13, 44, 44,
-            220, 93, 61, 73, 89, 160, 16, 247, 115, 174, 238, 80, 102, 26, 158,
-            44, 28, 173, 174, 3, 120, 130, 221, 220, 147, 143, 252, 137, 109,
-            143, 28, 106, 73, 253, 145, 161, 118, 109, 54, 95, 13, 137, 214,
-            253, 11, 238, 115, 89, 84, 241, 227, 103, 78, 246, 22,
-        ],
-    };
-    const opReturnMsgByteLength = getMessageByteSize(
-        msgInput,
-        true,
-        encryptedEjMock,
-    );
-    expect(opReturnMsgByteLength).toStrictEqual(129);
-});
-
-it(`OP_RETURN msg byte length matches for an encrypted msg input with a mixture of symbols, multilingual characters and emojis`, () => {
-    const msgInput = '๐Ÿ™ˆยฉๅ†ฐ์†Œ์ฃผ';
-    const encryptedEjMock = {
-        type: 'Buffer',
-        data: [
-            3, 237, 190, 133, 5, 192, 187, 247, 209, 218, 154, 239, 194, 148,
-            24, 151, 26, 150, 97, 190, 245, 27, 226, 249, 75, 203, 36, 128, 170,
-            209, 250, 181, 239, 253, 242, 53, 181, 198, 37, 123, 236, 120, 192,
-            179, 194, 103, 119, 70, 108, 242, 144, 120, 52, 205, 123, 158, 244,
-            27, 127, 232, 106, 215, 201, 88, 22, 146, 129, 6, 35, 160, 147, 198,
-            131, 236, 202, 200, 137, 39, 80, 241, 168, 158, 211, 113, 123, 76,
-            89, 81, 82, 250, 220, 162, 226, 63, 154, 76, 23,
-        ],
-    };
-    const opReturnMsgByteLength = getMessageByteSize(
-        msgInput,
-        true,
-        encryptedEjMock,
-    );
-    expect(opReturnMsgByteLength).toStrictEqual(97);
-});
-
-it(`OP_RETURN msg byte length matches for a msg input with a single emoji`, () => {
-    const msgInput = '๐Ÿ™ˆ';
-    const opReturnMsgByteLength = getMessageByteSize(msgInput);
-    expect(opReturnMsgByteLength).toStrictEqual(4);
-});
-
-it(`OP_RETURN msg byte length matches for a msg input with characters and emojis`, () => {
-    const msgInput = 'monkey๐Ÿ™ˆ';
-    const opReturnMsgByteLength = getMessageByteSize(msgInput);
-    expect(opReturnMsgByteLength).toStrictEqual(10);
-});
-
-it(`OP_RETURN msg byte length matches for a msg input with special characters`, () => {
-    const msgInput = 'monkeyยฉยฎส•โ€ขฬแดฅโ€ขฬ€ส”ใฃโ™ก';
-    const opReturnMsgByteLength = getMessageByteSize(msgInput);
-    expect(opReturnMsgByteLength).toStrictEqual(33);
-});
-
-it(`OP_RETURN msg byte length matches for a msg input with a mixture of symbols, multilingual characters and emojis`, () => {
-    const msgInput = '๐Ÿ™ˆยฉๅ†ฐ์†Œ์ฃผ';
-    const opReturnMsgByteLength = getMessageByteSize(msgInput);
-    expect(opReturnMsgByteLength).toStrictEqual(15);
-});
-
 it(`generateSendOpReturn() returns correct script object for valid tokenUtxo and send quantity`, () => {
     const tokensToSend = 50;
     const sendOpReturnScriptObj = generateSendOpReturn(
         mockSendOpReturnTokenUtxos,
         tokensToSend,
     );
 
     expect(JSON.stringify(sendOpReturnScriptObj.script)).toStrictEqual(
         JSON.stringify(mockSendOpReturnScript),
     );
 });
 
 it(`generateSendOpReturnScript() throws error on invalid input`, () => {
     const mockSendOpReturnTokenUtxos = null;
     const tokensToSend = 50;
 
     let errorThrown;
     try {
         generateSendOpReturn(mockSendOpReturnTokenUtxos, tokensToSend);
     } catch (err) {
         errorThrown = err.message;
     }
     expect(errorThrown).toStrictEqual('Invalid send token parameter');
 });
 
 it(`generateBurnOpReturn() returns correct script for valid tokenUtxo and burn quantity`, () => {
     const tokensToBurn = 7000;
     const burnOpReturnScript = generateBurnOpReturn(
         mockBurnOpReturnTokenUtxos,
         tokensToBurn,
     );
 
     expect(JSON.stringify(burnOpReturnScript)).toStrictEqual(
         JSON.stringify(mockBurnOpReturnScript),
     );
 });
 
 it(`generateBurnOpReturn() throws error on invalid input`, () => {
     const tokensToBurn = 7000;
     let errorThrown;
     try {
         generateBurnOpReturn(null, tokensToBurn);
     } catch (err) {
         errorThrown = err.message;
     }
     expect(errorThrown).toStrictEqual('Invalid burn token parameter');
 });
 
 it(`generateGenesisOpReturn() returns correct script for a valid configObj`, () => {
     const configObj = {
         name: 'ethantest',
         ticker: 'ETN',
         documentUrl: 'https://cashtab.com/',
         decimals: '3',
         initialQty: '5000',
         documentHash: '',
         mintBatonVout: null,
     };
 
     const genesisOpReturnScript = generateGenesisOpReturn(configObj);
 
     expect(JSON.stringify(genesisOpReturnScript)).toStrictEqual(
         JSON.stringify(mockGenesisOpReturnScript),
     );
 });
 
 it(`generateGenesisOpReturn() throws error on invalid configObj`, () => {
     const configObj = null;
 
     let errorThrown;
     try {
         generateGenesisOpReturn(configObj);
     } catch (err) {
         errorThrown = err.message;
     }
     expect(errorThrown).toStrictEqual('Invalid token configuration');
 });
 
 it(`signUtxosByAddress() successfully returns a txBuilder object for a one to one XEC tx`, () => {
     const isOneToMany = false;
     const { destinationAddress, wallet, utxos } = sendBCHMock;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const satoshisToSendInput = new BigNumber(2184);
     const feeInSatsPerByte = appConfig.defaultFee;
 
     // mock tx input
     const inputObj = generateTxInput(
         isOneToMany,
         utxos,
         txBuilder,
         null,
         satoshisToSendInput,
         feeInSatsPerByte,
     );
 
     // mock tx output
     const totalInputUtxoValue =
         mockOneToOneSendXecTxBuilderObj.transaction.inputs[0].value;
     const singleSendValue = new BigNumber(
         fromSatoshisToXec(
             mockOneToOneSendXecTxBuilderObj.transaction.tx.outs[0].value,
         ),
     );
     const satoshisToSendOutput = fromXecToSatoshis(
         new BigNumber(singleSendValue),
     );
     const txFee = new BigNumber(totalInputUtxoValue).minus(
         new BigNumber(satoshisToSendOutput),
     );
     const changeAddress = wallet.Path1899.cashAddress;
     const outputObj = generateTxOutput(
         isOneToMany,
         singleSendValue,
         satoshisToSendOutput,
         totalInputUtxoValue,
         destinationAddress,
         null,
         changeAddress,
         txFee,
         inputObj.txBuilder,
     );
 
     const txBuilderResponse = signUtxosByAddress(
         mockSingleInputUtxo,
         wallet,
         outputObj,
     );
     expect(txBuilderResponse.toString()).toStrictEqual(
         mockOneToOneSendXecTxBuilderObj.toString(),
     );
 });
 
 it(`signUtxosByAddress() successfully returns a txBuilder object for a one to many XEC tx`, () => {
     const isOneToMany = true;
     const { wallet, utxos } = sendBCHMock;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     let destinationAddressAndValueArray = [
         'ecash:qrmz0egsqxj35x5jmzf8szrszdeu72fx0uxgwk3r48,3000',
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx,3000',
         'ecash:qzvydd4n3lm3xv62cx078nu9rg0e3srmqq0knykfed,3000',
     ];
     const satoshisToSendInput = new BigNumber(900000);
     const feeInSatsPerByte = appConfig.defaultFee;
 
     // mock tx input
     const inputObj = generateTxInput(
         isOneToMany,
         utxos,
         txBuilder,
         destinationAddressAndValueArray,
         satoshisToSendInput,
         feeInSatsPerByte,
     );
 
     // mock tx output
     const totalInputUtxoValue =
         mockOneToManySendXecTxBuilderObj.transaction.inputs[0].value +
         mockOneToManySendXecTxBuilderObj.transaction.inputs[1].value +
         mockOneToManySendXecTxBuilderObj.transaction.inputs[2].value;
     const singleSendValue = null;
     const satoshisToSendOutput = new BigNumber(
         mockOneToManySendXecTxBuilderObj.transaction.tx.outs[0].value +
             mockOneToManySendXecTxBuilderObj.transaction.tx.outs[1].value +
             mockOneToManySendXecTxBuilderObj.transaction.tx.outs[2].value,
     );
     const txFee = new BigNumber(totalInputUtxoValue)
         .minus(satoshisToSendOutput)
         .minus(
             new BigNumber(
                 mockOneToManySendXecTxBuilderObj.transaction.tx.outs[3].value,
             ),
         ); // change value
     destinationAddressAndValueArray = validAddressArrayInput;
     const changeAddress = wallet.Path1899.cashAddress;
     const outputObj = generateTxOutput(
         isOneToMany,
         singleSendValue,
         satoshisToSendOutput,
         totalInputUtxoValue,
         null,
         destinationAddressAndValueArray,
         changeAddress,
         txFee,
         inputObj.txBuilder,
     );
 
     const txBuilderResponse = signUtxosByAddress(
         mockSingleInputUtxo,
         wallet,
         outputObj,
     );
     expect(txBuilderResponse.toString()).toStrictEqual(
         mockOneToManySendXecTxBuilderObj.toString(),
     );
 });
 
 it(`getChangeAddressFromInputUtxos() returns a correct change address from a valid inputUtxo`, () => {
     const { wallet } = sendBCHMock;
     const inputUtxo = [
         {
             height: 669639,
             tx_hash:
                 '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             tx_pos: 0,
             value: 1000,
             txid: '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             vout: 0,
             isValid: false,
             address: 'bitcoincash:qphpmfj0qn7znklqhrfn5dq7qh36l3vxavu346vqcl',
             wif: 'L58jqHoi5ynSdsskPVBJuGuVqTP8ZML1MwHQsBJY32Pv7cqDSCeH',
         },
     ];
 
     const changeAddress = getChangeAddressFromInputUtxos(inputUtxo, wallet);
     expect(changeAddress).toStrictEqual(inputUtxo[0].address);
 });
 
 it(`getChangeAddressFromInputUtxos() returns a correct change address from a valid inputUtxo and accepts ecash: format`, () => {
     const { wallet } = sendBCHMock;
     const inputUtxo = [
         {
             height: 669639,
             tx_hash:
                 '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             tx_pos: 0,
             value: 1000,
             txid: '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             vout: 0,
             isValid: false,
             address: 'ecash:qphpmfj0qn7znklqhrfn5dq7qh36l3vxav9up3h67g',
             wif: 'L58jqHoi5ynSdsskPVBJuGuVqTP8ZML1MwHQsBJY32Pv7cqDSCeH',
         },
     ];
 
     const changeAddress = getChangeAddressFromInputUtxos(inputUtxo, wallet);
     expect(changeAddress).toStrictEqual(inputUtxo[0].address);
 });
 
 it(`getChangeAddressFromInputUtxos() throws error upon a malformed input utxo`, () => {
     const { wallet } = sendBCHMock;
     const invalidInputUtxo = [
         {
             height: 669639,
             tx_hash:
                 '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             tx_pos: 0,
             value: 1000,
             txid: '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             vout: 0,
             isValid: false,
             wif: 'L58jqHoi5ynSdsskPVBJuGuVqTP8ZML1MwHQsBJY32Pv7cqDSCeH',
         },
     ];
     let thrownError;
     try {
         getChangeAddressFromInputUtxos(invalidInputUtxo, wallet);
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid input utxo');
 });
 
 it(`getChangeAddressFromInputUtxos() throws error upon a valid input utxo with invalid address param`, () => {
     const { wallet } = sendBCHMock;
     const invalidInputUtxo = [
         {
             height: 669639,
             tx_hash:
                 '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             tx_pos: 0,
             value: 1000,
             address: 'bitcoincash:1qphpmfj0qn7znklqhrfn5dq7qh36l3vxavu346vqcl', // invalid cash address
             txid: '0da6d49cf95d4603958e53360ad1e90bfccef41bfb327d6b2e8a77e242fa2d58',
             vout: 0,
             isValid: false,
             wif: 'L58jqHoi5ynSdsskPVBJuGuVqTP8ZML1MwHQsBJY32Pv7cqDSCeH',
         },
     ];
     let thrownError;
     try {
         getChangeAddressFromInputUtxos(invalidInputUtxo, wallet);
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid input utxo');
 });
 
 it(`getChangeAddressFromInputUtxos() throws an error upon a null inputUtxos param`, () => {
     const { wallet } = sendBCHMock;
     const inputUtxo = null;
 
     let thrownError;
     try {
         getChangeAddressFromInputUtxos(inputUtxo, wallet);
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual(
         'Invalid getChangeAddressFromWallet input parameter',
     );
 });
 
 it(`sumOneToManyXec() correctly parses the value for a valid one to many send XEC transaction`, () => {
     const destinationAddressAndValueArray = [
         'ecash:qrmz0egsqxj35x5jmzf8szrszdeu72fx0uxgwk3r48,1',
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx,2',
         'ecash:qzvydd4n3lm3xv62cx078nu9rg0e3srmqq0knykfed,3',
     ];
     expect(sumOneToManyXec(destinationAddressAndValueArray)).toStrictEqual(6);
 });
 it(`sumOneToManyXec() correctly parses the value for a valid one to many send XEC transaction with decimals`, () => {
     const destinationAddressAndValueArray = [
         'ecash:qrmz0egsqxj35x5jmzf8szrszdeu72fx0uxgwk3r48,1.23',
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx,2.45',
         'ecash:qzvydd4n3lm3xv62cx078nu9rg0e3srmqq0knykfed,3.67',
     ];
     expect(sumOneToManyXec(destinationAddressAndValueArray)).toStrictEqual(
         7.35,
     );
 });
 it(`sumOneToManyXec() returns NaN for an address and value array that is partially typed or has invalid format`, () => {
     const destinationAddressAndValueArray = [
         'ecash:qrmz0egsqxj35x5jmzf8szrszdeu72fx0uxgwk3r48,1',
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx,',
     ];
     expect(sumOneToManyXec(destinationAddressAndValueArray)).toStrictEqual(NaN);
 });
 it('generateAliasOpReturnScript() correctly generates OP_RETURN script for a valid alias registration for a p2pkh address', () => {
     const alias = 'test';
     const address = 'ecash:qz2708636snqhsxu8wnlka78h6fdp77ar59jrf5035';
     const { hash } = cashaddr.decode(address, true);
 
     // Manually build the expected outputScript
     const opReturn = '6a';
     // push protocol identifier
     const prefixBytesHex = '04';
     const aliasIdentifier = '2e786563';
 
     // push alias tx version
     const aliasProtocolVersionNumberHex = '00';
 
     // push the alias
     const aliasHexBytes = '04'; // alias.length in one byte of hex
     const aliasHex = Buffer.from(alias).toString('hex');
 
     // push the address
     const aliasAddressBytesHex = '15'; // (1 + 20) in one byte of hex
     const p2pkhVersionByteHex = '00';
 
     const aliasTxOpReturnOutputScript = [
         opReturn,
         prefixBytesHex,
         aliasIdentifier,
         aliasProtocolVersionNumberHex,
         aliasHexBytes,
         aliasHex,
         aliasAddressBytesHex,
         p2pkhVersionByteHex,
         hash,
     ].join('');
 
     // Calculate the expected outputScript with the tested function
     const aliasOutputScript = generateAliasOpReturnScript(alias, address);
     // aliasOutputScript.toString('hex')
     // 6a042e78656301000474657374150095e79f51d4260bc0dc3ba7fb77c7be92d0fbdd1d
 
     expect(aliasOutputScript.toString('hex')).toBe(aliasTxOpReturnOutputScript);
 });
 it('generateAliasOpReturnScript() correctly generates OP_RETURN script for a valid alias registration for a p2sh address', () => {
     const alias = 'testtwo';
     const address = 'ecash:prfhcnyqnl5cgrnmlfmms675w93ld7mvvqd0y8lz07';
     const { hash } = cashaddr.decode(address, true);
 
     // Manually build the expected outputScript
     const opReturn = '6a';
 
     // push protocol identifier
     const prefixBytesHex = '04';
     const aliasIdentifier = '2e786563';
 
     // push alias tx version
     const aliasProtocolVersionNumberHex = '00';
 
     // push the alias
     const aliasHexBytes = '07'; // alias.length in one byte of hex
     const aliasHex = Buffer.from(alias).toString('hex');
 
     // push the address
     const aliasAddressBytesHex = '15'; // (1 + 20) in one byte of hex
     const p2shVersionByteHex = '08';
 
     const aliasTxOpReturnOutputScript = [
         opReturn,
         prefixBytesHex,
         aliasIdentifier,
         aliasProtocolVersionNumberHex,
         aliasHexBytes,
         aliasHex,
         aliasAddressBytesHex,
         p2shVersionByteHex,
         hash,
     ].join('');
 
     // Calculate the expected outputScript with the tested function
     const aliasOutputScript = generateAliasOpReturnScript(alias, address);
     // aliasOutputScript.toString('hex')
     // 6a042e7865630100077465737474776f1508d37c4c809fe9840e7bfa77b86bd47163f6fb6c60
 
     expect(aliasOutputScript.toString('hex')).toBe(aliasTxOpReturnOutputScript);
 });
-it('generateOpReturnScript() correctly generates an encrypted message script', () => {
-    const optionalOpReturnMsg = 'testing generateOpReturnScript()';
-    const encryptionFlag = true;
-    const airdropFlag = false;
-    const airdropTokenId = null;
-    const mockEncryptedEj =
-        '04688f9907fe3c7c0b78a73c4ab4f75e15e7e2b79641add519617086126fe6f6b1405a14eed48e90c9c8c0fc77f0f36984a78173e76ce51f0a44af94b59e9da703c9ff82758cfdb9cc46437d662423400fb731d3bfc1df0599279356ca261213fbb40d398c041e1bac966afed1b404581ab1bcfcde1fa039d53b7c7b70e8edf26d64bea9fbeed24cc80909796e6af5863707fa021f2a2ebaa2fe894904702be19d';
-
-    const encodedScript = generateOpReturnScript(
-        optionalOpReturnMsg,
-        encryptionFlag,
-        airdropFlag,
-        airdropTokenId,
-        mockEncryptedEj,
-    );
-    expect(encodedScript.toString('hex')).toBe(
-        '6a04657461624d420130343638386639393037666533633763306237386137336334616234663735653135653765326237393634316164643531393631373038363132366665366636623134303561313465656434386539306339633863306663373766306633363938346137383137336537366365353166306134346166393462353965396461373033633966663832373538636664623963633436343337643636323432333430306662373331643362666331646630353939323739333536636132363132313366626234306433393863303431653162616339363661666564316234303435383161623162636663646531666130333964353362376337623730653865646632366436346265613966626565643234636338303930393739366536616635383633373037666130323166326132656261613266653839343930343730326265313964',
-    );
-});
-
-it('generateOpReturnScript() correctly generates an un-encrypted non-airdrop message script', () => {
-    const optionalOpReturnMsg = 'testing generateOpReturnScript()';
-    const encryptionFlag = false;
-    const airdropFlag = false;
-
-    const encodedScript = generateOpReturnScript(
-        optionalOpReturnMsg,
-        encryptionFlag,
-        airdropFlag,
-    );
-    expect(encodedScript.toString('hex')).toBe(
-        '6a04007461622074657374696e672067656e65726174654f7052657475726e5363726970742829',
-    );
-});
-
-it('generateOpReturnScript() correctly generates an un-encrypted airdrop message script', () => {
-    const optionalOpReturnMsg = 'testing generateOpReturnScript()';
-    const encryptionFlag = false;
-    const airdropFlag = true;
-    const airdropTokenId =
-        '1c6c9c64d70b285befe733f175d0f384538576876bd280b10587df81279d3f5e';
-
-    const encodedScript = generateOpReturnScript(
-        optionalOpReturnMsg,
-        encryptionFlag,
-        airdropFlag,
-        airdropTokenId,
-    );
-    expect(encodedScript.toString('hex')).toBe(
-        '6a0464726f70201c6c9c64d70b285befe733f175d0f384538576876bd280b10587df81279d3f5e04007461622074657374696e672067656e65726174654f7052657475726e5363726970742829',
-    );
-});
-
-it('generateOpReturnScript() correctly generates an un-encrypted airdrop with no message script', () => {
-    const optionalOpReturnMsg = null;
-    const encryptionFlag = false;
-    const airdropFlag = true;
-    const airdropTokenId =
-        '1c6c9c64d70b285befe733f175d0f384538576876bd280b10587df81279d3f5e';
-
-    const encodedScript = generateOpReturnScript(
-        optionalOpReturnMsg,
-        encryptionFlag,
-        airdropFlag,
-        airdropTokenId,
-    );
-    expect(encodedScript.toString('hex')).toBe(
-        '6a0464726f70201c6c9c64d70b285befe733f175d0f384538576876bd280b10587df81279d3f5e0400746162',
-    );
-});
-
-it('generateOpReturnScript() correctly throws an error on an invalid encryption input', () => {
-    const optionalOpReturnMsg = null;
-    const encryptionFlag = true;
-    const airdropFlag = false;
-    const airdropTokenId = null;
-    const mockEncryptedEj = null; // invalid given encryptionFlag is true
-    let thrownError;
-
-    try {
-        generateOpReturnScript(
-            optionalOpReturnMsg,
-            encryptionFlag,
-            airdropFlag,
-            airdropTokenId,
-            mockEncryptedEj,
-        );
-    } catch (err) {
-        thrownError = err;
-    }
-    expect(thrownError.message).toStrictEqual('Invalid OP RETURN script input');
-});
-
-it('generateOpReturnScript() correctly throws an error on an invalid airdrop input', () => {
-    const optionalOpReturnMsg = null;
-    const encryptionFlag = false;
-    const airdropFlag = true;
-    const airdropTokenId = null; // invalid given airdropFlag is true
-
-    let thrownError;
-
-    try {
-        generateOpReturnScript(
-            optionalOpReturnMsg,
-            encryptionFlag,
-            airdropFlag,
-            airdropTokenId,
-        );
-    } catch (err) {
-        thrownError = err;
-    }
-    expect(thrownError.message).toStrictEqual('Invalid OP RETURN script input');
-});
-
-it('generateOpReturnScript() correctly generates an alias registration script', () => {
-    const optionalOpReturnMsg = 'nfs'; // the alias name to be registered
-    const encodedScript = generateOpReturnScript(
-        optionalOpReturnMsg,
-        false,
-        false,
-        null,
-        null,
-        true, // alias registration flag
-    );
-    expect(encodedScript.toString('hex')).toBe('6a042e786563036e6673');
-});
 
 it(`generateTokenTxInput() returns a valid object for a valid create token tx`, async () => {
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const tokenId =
         '1c6c9c64d70b285befe733f175d0f384538576876bd280b10587df81279d3f5e';
     const tokenInputObj = generateTokenTxInput(
         'GENESIS',
         mockNonSlpUtxos,
         null, // no slpUtxos used for genesis tx
         tokenId,
         null, // no token send/burn amount for genesis tx
         appConfig.defaultFee,
         txBuilder,
     );
 
     expect(tokenInputObj.inputXecUtxos).toStrictEqual([mockNonSlpUtxos[0]]);
     expect(tokenInputObj.txBuilder.toString()).toStrictEqual(
         mockCreateTokenTxBuilderObj.toString(),
     );
     expect(tokenInputObj.remainderXecValue).toStrictEqual(
         new BigNumber(698999), // tokenInputObj.inputXecUtxos - appConfig.etokenSats 546 - txFee
     );
 });
 
 it(`generateTokenTxInput() returns a valid object for a valid send token tx`, async () => {
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const tokenId = mockSlpUtxos[0].tokenId;
 
     const tokenInputObj = generateTokenTxInput(
         'SEND',
         mockNonSlpUtxos,
         mockSlpUtxos,
         tokenId,
         new BigNumber(500), // sending 500 of these tokens
         appConfig.defaultFee,
         txBuilder,
     );
 
     expect(tokenInputObj.inputTokenUtxos).toStrictEqual(
         [mockSlpUtxos[0]].concat([mockSlpUtxos[1]]), // mockSlpUtxos[0] 400 + mockSlpUtxos[1] 6500
     );
     expect(tokenInputObj.remainderTokenValue).toStrictEqual(
         new BigNumber(6400), // token change is mockSlpUtxos[0] 400 + mockSlpUtxos[1] 6500 - [tokenAmount] 500
     );
     expect(tokenInputObj.txBuilder.toString()).toStrictEqual(
         mockSendTokenTxBuilderObj.toString(),
     );
 });
 
 it(`generateTokenTxInput() returns a valid object for a valid burn token tx`, async () => {
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const tokenId = mockSlpUtxos[0].tokenId;
 
     const tokenInputObj = generateTokenTxInput(
         'BURN',
         mockNonSlpUtxos,
         mockSlpUtxos,
         tokenId,
         new BigNumber(500), // burning 500 of these tokens
         appConfig.defaultFee,
         txBuilder,
     );
 
     expect(tokenInputObj.inputTokenUtxos).toStrictEqual(
         [mockSlpUtxos[0]].concat([mockSlpUtxos[1]]), // mockSlpUtxos[0] 400 + mockSlpUtxos[1] 6500
     );
     expect(tokenInputObj.remainderTokenValue).toStrictEqual(
         new BigNumber(6400), // token change is mockSlpUtxos[0] 400 + mockSlpUtxos[1] 6500 - [tokenAmount] 500
     );
     expect(tokenInputObj.txBuilder.toString()).toStrictEqual(
         mockBurnTokenTxBuilderObj.toString(),
     );
 });
 
 it(`generateTokenTxOutput() returns a valid object for a valid create token tx`, async () => {
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const { configObj, wallet } = createTokenMock;
     const tokenSenderCashAddress = wallet.Path1899.cashAddress;
 
     const tokenOutputObj = generateTokenTxOutput(
         txBuilder,
         'GENESIS',
         tokenSenderCashAddress,
         null, // optional, for SEND or BURN amount
         new BigNumber(500), // remainder XEC value
         configObj,
     );
 
     expect(tokenOutputObj.toString()).toStrictEqual(
         mockCreateTokenOutputsTxBuilderObj.toString(),
     );
 });
 
 it(`generateTokenTxOutput() returns a valid object for a valid send token tx`, async () => {
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const { wallet } = createTokenMock;
     const tokenSenderCashAddress = wallet.Path1899.cashAddress;
     const tokenRecipientTokenAddress = wallet.Path1899.cashAddress;
 
     const tokenOutputObj = generateTokenTxOutput(
         txBuilder,
         'SEND',
         tokenSenderCashAddress,
         mockSlpUtxos,
         new BigNumber(500), // remainder XEC value
         null, // only for genesis tx
         tokenRecipientTokenAddress, // recipient token address
         new BigNumber(50),
     );
 
     expect(tokenOutputObj.toString()).toStrictEqual(
         mockSendTokenOutputsTxBuilderObj.toString(),
     );
 });
 
 it(`generateTokenTxOutput() returns a valid object for a valid burn token tx`, async () => {
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const { wallet } = createTokenMock;
     const tokenSenderCashAddress = wallet.Path1899.cashAddress;
 
     const tokenOutputObj = generateTokenTxOutput(
         txBuilder,
         'BURN',
         tokenSenderCashAddress,
         mockSlpUtxos,
         new BigNumber(500), // remainder XEC value
         null, // only for genesis tx
         null, // no token recipients for burn tx
         new BigNumber(50),
     );
 
     expect(tokenOutputObj.toString()).toStrictEqual(
         mockBurnTokenOutputsTxBuilderObj.toString(),
     );
 });
 
 it(`generateTxInput() returns an input object for a valid one to one XEC tx`, async () => {
     const isOneToMany = false;
     const utxos = mockNonSlpUtxos;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const destinationAddressAndValueArray = null;
     const satoshisToSend = new BigNumber(2184);
     const feeInSatsPerByte = appConfig.defaultFee;
 
     const inputObj = generateTxInput(
         isOneToMany,
         utxos,
         txBuilder,
         destinationAddressAndValueArray,
         satoshisToSend,
         feeInSatsPerByte,
     );
     expect(inputObj.txBuilder).not.toStrictEqual(null);
     expect(inputObj.totalInputUtxoValue).toStrictEqual(new BigNumber(700000));
     expect(inputObj.txFee).toStrictEqual(455);
     expect(inputObj.inputUtxos.length).not.toStrictEqual(0);
 });
 
 it(`generateTxInput() returns an input object for a valid one to many XEC tx`, async () => {
     const isOneToMany = true;
     const utxos = mockNonSlpUtxos;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const destinationAddressAndValueArray = [
         'ecash:qrmz0egsqxj35x5jmzf8szrszdeu72fx0uxgwk3r48,3000',
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx,3000',
         'ecash:qzvydd4n3lm3xv62cx078nu9rg0e3srmqq0knykfed,3000',
     ];
     const satoshisToSend = new BigNumber(900000);
     const feeInSatsPerByte = appConfig.defaultFee;
 
     const inputObj = generateTxInput(
         isOneToMany,
         utxos,
         txBuilder,
         destinationAddressAndValueArray,
         satoshisToSend,
         feeInSatsPerByte,
     );
     expect(inputObj.txBuilder).not.toStrictEqual(null);
     expect(inputObj.totalInputUtxoValue).toStrictEqual(new BigNumber(1400000));
     expect(inputObj.txFee).toStrictEqual(889);
     expect(inputObj.inputUtxos.length).not.toStrictEqual(0);
 });
 
 it(`generateTxInput() throws error for a one to many XEC tx with invalid destinationAddressAndValueArray input`, async () => {
     const isOneToMany = true;
     const utxos = mockNonSlpUtxos;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const destinationAddressAndValueArray = null; // invalid since isOneToMany is true
     const satoshisToSend = new BigNumber(900000);
     const feeInSatsPerByte = appConfig.defaultFee;
 
     let thrownError;
     try {
         generateTxInput(
             isOneToMany,
             utxos,
             txBuilder,
             destinationAddressAndValueArray,
             satoshisToSend,
             feeInSatsPerByte,
         );
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid tx input parameter');
 });
 
 it(`generateTxInput() throws error for a one to many XEC tx with invalid utxos input`, async () => {
     const isOneToMany = true;
     const utxos = null;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const destinationAddressAndValueArray = [
         'ecash:qrmz0egsqxj35x5jmzf8szrszdeu72fx0uxgwk3r48,3000',
         'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx,3000',
         'ecash:qzvydd4n3lm3xv62cx078nu9rg0e3srmqq0knykfed,3000',
     ];
     const satoshisToSend = new BigNumber(900000);
     const feeInSatsPerByte = appConfig.defaultFee;
 
     let thrownError;
     try {
         generateTxInput(
             isOneToMany,
             utxos,
             txBuilder,
             destinationAddressAndValueArray,
             satoshisToSend,
             feeInSatsPerByte,
         );
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid tx input parameter');
 });
 
 it(`generateTxOutput() returns a txBuilder instance for a valid one to one XEC tx`, () => {
     // txbuilder output params
     const { destinationAddress, wallet } = sendBCHMock;
     const isOneToMany = false;
     const singleSendValue = fromSatoshisToXec(
         mockOneToOneSendXecTxBuilderObj.transaction.tx.outs[0].value,
     );
     const totalInputUtxoValue =
         mockOneToOneSendXecTxBuilderObj.transaction.inputs[0].value;
     const satoshisToSend = fromXecToSatoshis(new BigNumber(singleSendValue));
     // for unit test purposes, calculate fee by subtracting satoshisToSend from totalInputUtxoValue
     // no change output to be subtracted in this tx
     const txFee = new BigNumber(totalInputUtxoValue).minus(
         new BigNumber(satoshisToSend),
     );
 
     const destinationAddressAndValueArray = null;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const changeAddress = wallet.Path1899.cashAddress;
 
     const outputObj = generateTxOutput(
         isOneToMany,
         singleSendValue,
         satoshisToSend,
         totalInputUtxoValue,
         destinationAddress,
         destinationAddressAndValueArray,
         changeAddress,
         txFee,
         txBuilder,
     );
     expect(outputObj.toString()).toStrictEqual(
         mockOneToOneSendXecTxBuilderObj.toString(),
     );
 });
 
 it(`generateTxOutput() returns a txBuilder instance for a valid one to many XEC tx`, () => {
     // txbuilder output params
     const { destinationAddress, wallet } = sendBCHMock;
     const isOneToMany = true;
     const singleSendValue = null;
     const totalInputUtxoValue =
         mockOneToManySendXecTxBuilderObj.transaction.inputs[0].value +
         mockOneToManySendXecTxBuilderObj.transaction.inputs[1].value +
         mockOneToManySendXecTxBuilderObj.transaction.inputs[2].value;
     const satoshisToSend = new BigNumber(
         mockOneToManySendXecTxBuilderObj.transaction.tx.outs[0].value +
             mockOneToManySendXecTxBuilderObj.transaction.tx.outs[1].value +
             mockOneToManySendXecTxBuilderObj.transaction.tx.outs[2].value,
     );
     // for unit test purposes, calculate fee by subtracting satoshisToSend and change amount from totalInputUtxoValue
     const txFee = new BigNumber(totalInputUtxoValue)
         .minus(satoshisToSend)
         .minus(
             new BigNumber(
                 mockOneToManySendXecTxBuilderObj.transaction.tx.outs[3].value,
             ),
         ); // change value
 
     const destinationAddressAndValueArray = validAddressArrayInput;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const changeAddress = wallet.Path1899.cashAddress;
 
     const outputObj = generateTxOutput(
         isOneToMany,
         singleSendValue,
         satoshisToSend,
         totalInputUtxoValue,
         destinationAddress,
         destinationAddressAndValueArray,
         changeAddress,
         txFee,
         txBuilder,
     );
     expect(outputObj.toString()).toStrictEqual(
         mockOneToManySendXecTxBuilderObj.toString(),
     );
 });
 
 it(`generateTxOutput() throws an error on invalid input params for a one to one XEC tx`, () => {
     // txbuilder output params
     const { wallet } = sendBCHMock;
     const isOneToMany = false;
     const singleSendValue = null; // invalid due to singleSendValue being mandatory when isOneToMany is false
     const totalInputUtxoValue =
         mockOneToOneSendXecTxBuilderObj.transaction.inputs[0].value;
     const satoshisToSend = fromXecToSatoshis(new BigNumber(singleSendValue));
     // for unit test purposes, calculate fee by subtracting satoshisToSend from totalInputUtxoValue
     // no change output to be subtracted in this tx
     const txFee = new BigNumber(totalInputUtxoValue).minus(satoshisToSend);
 
     const destinationAddressAndValueArray = null;
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const changeAddress = wallet.Path1899.cashAddress;
 
     let thrownError;
     try {
         generateTxOutput(
             isOneToMany,
             singleSendValue,
             satoshisToSend,
             totalInputUtxoValue,
             null,
             destinationAddressAndValueArray,
             changeAddress,
             txFee,
             txBuilder,
         );
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid tx input parameter');
 });
 
 it(`generateTxOutput() throws an error on invalid input params for a one to many XEC tx`, () => {
     // txbuilder output params
     const { wallet } = sendBCHMock;
     const isOneToMany = true;
     const singleSendValue = null;
     const totalInputUtxoValue =
         mockOneToManySendXecTxBuilderObj.transaction.inputs[0].value +
         mockOneToManySendXecTxBuilderObj.transaction.inputs[1].value +
         mockOneToManySendXecTxBuilderObj.transaction.inputs[2].value;
     const satoshisToSend = new BigNumber(
         mockOneToManySendXecTxBuilderObj.transaction.tx.outs[0].value +
             mockOneToManySendXecTxBuilderObj.transaction.tx.outs[1].value +
             mockOneToManySendXecTxBuilderObj.transaction.tx.outs[2].value,
     );
     // for unit test purposes, calculate fee by subtracting satoshisToSend and change amount from totalInputUtxoValue
     const txFee = new BigNumber(totalInputUtxoValue)
         .minus(satoshisToSend)
         .minus(
             new BigNumber(
                 mockOneToManySendXecTxBuilderObj.transaction.tx.outs[3].value,
             ),
         ); // change value
     const destinationAddressAndValueArray = null; // invalid as this is mandatory when isOneToMany is true
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const changeAddress = wallet.Path1899.cashAddress;
 
     let thrownError;
     try {
         generateTxOutput(
             isOneToMany,
             singleSendValue,
             satoshisToSend,
             totalInputUtxoValue,
             null,
             destinationAddressAndValueArray,
             changeAddress,
             txFee,
             txBuilder,
         );
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid tx input parameter');
 });
 
 it(`signAndBuildTx() successfully returns a raw tx hex for a tx with a single input and a single output`, () => {
     // txbuilder output params
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     // Use legacy sequencing for legacy unit tests
     txBuilder.DEFAULT_SEQUENCE = 0xffffffff;
     const { wallet } = sendBCHMock;
     // add inputs to txBuilder
     txBuilder.addInput(
         mockSingleInputUtxo[0].txid,
         mockSingleInputUtxo[0].vout,
     );
 
     // add outputs to txBuilder
     const outputAddressAndValue = mockSingleOutput.split(',');
     txBuilder.addOutput(
         cashaddr.toLegacy(outputAddressAndValue[0]), // address
         parseInt(fromXecToSatoshis(new BigNumber(outputAddressAndValue[1]))), // value
     );
     const rawTxHex = signAndBuildTx(mockSingleInputUtxo, txBuilder, wallet);
     expect(rawTxHex).toStrictEqual(
         '0200000001582dfa42e2778a2e6b7d32fb1bf4cefc0be9d10a36538e9503465df99cd4a60d000000006b483045022100b4ee5268cb64c4f097e739df7c6934d1df7e75a4f217d5824db18ae2e12554b102204faf039738181aae80c064b928b3d8079a82cdb080ce9a2d5453939a588f4372412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff0158020000000000001976a9149846b6b38ff713334ac19fe3cf851a1f98c07b0088ac00000000',
     );
 });
 
 it(`signAndBuildTx() successfully returns a raw tx hex for a tx with a single input and multiple outputs`, () => {
     // txbuilder output params
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     // Use legacy sequencing for legacy unit tests
     txBuilder.DEFAULT_SEQUENCE = 0xffffffff;
     const { wallet } = sendBCHMock;
     // add inputs to txBuilder
     txBuilder.addInput(
         mockSingleInputUtxo[0].txid,
         mockSingleInputUtxo[0].vout,
     );
 
     // add outputs to txBuilder
     for (let i = 0; i < mockMultipleOutputs.length; i++) {
         const outputAddressAndValue = mockMultipleOutputs[i].split(',');
         txBuilder.addOutput(
             cashaddr.toLegacy(outputAddressAndValue[0]), // address
             parseInt(
                 fromXecToSatoshis(new BigNumber(outputAddressAndValue[1])),
             ), // value
         );
     }
 
     const rawTxHex = signAndBuildTx(mockSingleInputUtxo, txBuilder, wallet);
     expect(rawTxHex).toStrictEqual(
         '0200000001582dfa42e2778a2e6b7d32fb1bf4cefc0be9d10a36538e9503465df99cd4a60d000000006b483045022100df29734c4fb348b0e8b613ce522c10c5ac14cb3ecd32843dc7fcf004d60f1b8a022023c4ae02b38c7272e29f344902ae2afa4db1ec37d582a31c16650a0abc4f480c412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff0326020000000000001976a914f627e51001a51a1a92d8927808701373cf29267f88ac26020000000000001976a9140b7d35fda03544a08e65464d54cfae4257eb6db788ac26020000000000001976a9149846b6b38ff713334ac19fe3cf851a1f98c07b0088ac00000000',
     );
 });
 
 it(`signAndBuildTx() successfully returns a raw tx hex for a tx with multiple inputs and a single output`, () => {
     // txbuilder output params
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     // Use legacy sequencing for legacy unit tests
     txBuilder.DEFAULT_SEQUENCE = 0xffffffff;
     const { wallet } = sendBCHMock;
     // add inputs to txBuilder
     for (let i = 0; i < mockMultipleInputUtxos.length; i++) {
         txBuilder.addInput(
             mockMultipleInputUtxos[i].txid,
             mockMultipleInputUtxos[i].vout,
         );
     }
     // add outputs to txBuilder
     const outputAddressAndValue = mockSingleOutput.split(',');
     txBuilder.addOutput(
         cashaddr.toLegacy(outputAddressAndValue[0]), // address
         parseInt(fromXecToSatoshis(new BigNumber(outputAddressAndValue[1]))), // value
     );
 
     const rawTxHex = signAndBuildTx(mockMultipleInputUtxos, txBuilder, wallet);
     expect(rawTxHex).toStrictEqual(
         '0200000003582dfa42e2778a2e6b7d32fb1bf4cefc0be9d10a36538e9503465df99cd4a60d000000006a4730440220541366dd5ea25d65d3044dbde16fc6118ab1aee07c7d0d4c25c9e8aa299f040402203ed2f540948197d4c6a4ae963ad187d145a9fb339e311317b03c6172732e267b412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff7313e804af08113dfa290515390a8ec3ac01448118f2eb556ee168a96ee6acdd000000006b483045022100c1d02c5023f83b87a4f2dd26a7306ed9be9d53ab972bd935b440e45eb54a304302200b99aa2f1a728b3bb1dcbff80742c5fcab991bb74e80fa231255a31d58a6ff7d412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff960dd2f0c47e8a3cf1486b046d879f45a047da3b51aedfb5594138ac857214f1000000006b483045022100bd24d11d7070988848cb4aa2b10748aa0aeb79dc8af39c1f22dc1034b3121e5f02201491026e5f8f6eb566eb17cb195e3da3ff0d9cf01bdd34c944964d33a8d3b1ad412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff0158020000000000001976a9149846b6b38ff713334ac19fe3cf851a1f98c07b0088ac00000000',
     );
 });
 
 it(`signAndBuildTx() successfully returns a raw tx hex for a tx with multiple inputs and multiple outputs`, () => {
     // txbuilder output params
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     // Use legacy sequencing for legacy unit tests
     txBuilder.DEFAULT_SEQUENCE = 0xffffffff;
     const { wallet } = sendBCHMock;
     // add inputs to txBuilder
     for (let i = 0; i < mockMultipleInputUtxos.length; i++) {
         txBuilder.addInput(
             mockMultipleInputUtxos[i].txid,
             mockMultipleInputUtxos[i].vout,
         );
     }
     // add outputs to txBuilder
     for (let i = 0; i < mockMultipleOutputs.length; i++) {
         const outputAddressAndValue = mockMultipleOutputs[i].split(',');
         txBuilder.addOutput(
             cashaddr.toLegacy(outputAddressAndValue[0]), // address
             parseInt(
                 fromXecToSatoshis(new BigNumber(outputAddressAndValue[1])),
             ), // value
         );
     }
 
     const rawTxHex = signAndBuildTx(mockMultipleInputUtxos, txBuilder, wallet);
     expect(rawTxHex).toStrictEqual(
         '0200000003582dfa42e2778a2e6b7d32fb1bf4cefc0be9d10a36538e9503465df99cd4a60d000000006a47304402203de4e6a512a6bec1d378b6444008484e1be5a0c621dc4b201d67addefffe864602202daf82e76b7594fe1ab54a49380c6b1226ab65551ae6ab9164216b66266f34a1412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff7313e804af08113dfa290515390a8ec3ac01448118f2eb556ee168a96ee6acdd000000006a473044022029f5fcbc9356beb9eae6b9ff9a479e8c8331b95406b6be456fccf9d90f148ea1022028f4e7fa7234f9429535360c8f5dad303e2c5044431615997861b10f26fa8a88412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff960dd2f0c47e8a3cf1486b046d879f45a047da3b51aedfb5594138ac857214f1000000006a473044022049a67738d99006b3523cff818f3626104cf5106bd463be70d22ad179a8cb403b022025829baf67f964202ea77ea7462a5447e32415e7293cdee382ea7ae9374364e8412102322fe90c5255fe37ab321c386f9446a86e80c3940701d430f22325094fdcec60ffffffff0326020000000000001976a914f627e51001a51a1a92d8927808701373cf29267f88ac26020000000000001976a9140b7d35fda03544a08e65464d54cfae4257eb6db788ac26020000000000001976a9149846b6b38ff713334ac19fe3cf851a1f98c07b0088ac00000000',
     );
 });
 
 it(`signAndBuildTx() throws error on an empty inputUtxo param`, () => {
     // txbuilder output params
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const { wallet } = sendBCHMock;
     let thrownError;
     try {
         signAndBuildTx([], txBuilder, wallet);
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid buildTx parameter');
 });
 
 it(`signAndBuildTx() throws error on a null inputUtxo param`, () => {
     // txbuilder output params
     let txBuilder = utxolib.bitgo.createTransactionBuilderForNetwork(
         utxolib.networks.ecash,
     );
     const inputUtxo = null; // invalid input param
     const { wallet } = sendBCHMock;
     let thrownError;
     try {
         signAndBuildTx(inputUtxo, txBuilder, wallet);
     } catch (err) {
         thrownError = err;
     }
     expect(thrownError.message).toStrictEqual('Invalid buildTx parameter');
 });
 
 describe('Correctly executes cash utility functions', () => {
     it(`Correctly converts smallest base unit to smallest decimal for cashDecimals = 2`, () => {
         expect(fromSatoshisToXec(1, 2)).toStrictEqual(new BigNumber(0.01));
     });
     it(`Correctly converts largest base unit to smallest decimal for cashDecimals = 2`, () => {
         expect(fromSatoshisToXec(1000000012345678, 2)).toStrictEqual(
             new BigNumber(10000000123456.78),
         );
     });
     it(`Correctly converts smallest base unit to smallest decimal for cashDecimals = 8`, () => {
         expect(fromSatoshisToXec(1, 8)).toStrictEqual(
             new BigNumber(0.00000001),
         );
     });
     it(`Correctly converts largest base unit to smallest decimal for cashDecimals = 8`, () => {
         expect(fromSatoshisToXec(1000000012345678, 8)).toStrictEqual(
             new BigNumber(10000000.12345678),
         );
     });
     it(`Accepts a cachedWalletState that has not preserved BigNumber object types, and returns the same wallet state with BigNumber type re-instituted`, () => {
         expect(loadStoredWallet(cachedUtxos)).toStrictEqual(
             utxosLoadedFromCache,
         );
     });
     it(`loadStoredWallet accepts undefined wallet state as input and outputs a zero balance wallet state`, () => {
         expect(loadStoredWallet(undefined)).toStrictEqual({
             balances: {
                 totalBalanceInSatoshis: '0',
                 totalBalance: '0',
             },
         });
     });
     it(`Correctly determines a wallet's balance from its set of non-eToken utxos (nonSlpUtxos)`, () => {
         expect(
             getWalletBalanceFromUtxos(
                 validStoredWalletAfter20221123Streamline.state.nonSlpUtxos,
             ),
         ).toStrictEqual(validStoredWallet.state.balances);
     });
     it(`Correctly determines a wallet's zero balance from its empty set of non-eToken utxos (nonSlpUtxos)`, () => {
         expect(
             getWalletBalanceFromUtxos(utxosLoadedFromCache.nonSlpUtxos),
         ).toStrictEqual(utxosLoadedFromCache.balances);
     });
     it(`Recognizes a stored wallet as valid if it has all required fields prior to 20221123 updated format`, () => {
         expect(isValidStoredWallet(validStoredWallet)).toBe(true);
     });
     it(`Recognizes a stored wallet as valid if it has all required fields in 20221123 updated format`, () => {
         expect(
             isValidStoredWallet(validStoredWalletAfter20221123Streamline),
         ).toBe(true);
     });
     it(`Recognizes a stored wallet as invalid if it is missing required fields`, () => {
         expect(isValidStoredWallet(invalidStoredWallet)).toBe(false);
     });
     it(`Recognizes a stored wallet as invalid if it includes hydratedUtxoDetails in the state field`, () => {
         expect(isValidStoredWallet(invalidpreChronikStoredWallet)).toBe(false);
     });
     it(`Recognizes a stored wallet as invalid if it's missing the Path1899 and mnemonic keys`, () => {
         expect(
             isValidStoredWallet(invalidStoredWalletMissingPath1899AndMnemonic),
         ).toBe(false);
     });
     it(`Converts a legacy BCH amount to an XEC amount`, () => {
         expect(fromLegacyDecimals(0.00000546, 2)).toStrictEqual(5.46);
     });
     it(`Leaves a legacy BCH amount unchanged if appConfig.cashDecimals is 8`, () => {
         expect(fromLegacyDecimals(0.00000546, 8)).toStrictEqual(0.00000546);
     });
     it(`convertToEcashPrefix converts a bitcoincash: prefixed address to an ecash: prefixed address`, () => {
         expect(
             convertToEcashPrefix(
                 'bitcoincash:qz2708636snqhsxu8wnlka78h6fdp77ar5ulhz04hr',
             ),
         ).toBe('ecash:qz2708636snqhsxu8wnlka78h6fdp77ar59jrf5035');
     });
     it(`convertToEcashPrefix returns an ecash: prefix address unchanged`, () => {
         expect(
             convertToEcashPrefix(
                 'ecash:qz2708636snqhsxu8wnlka78h6fdp77ar59jrf5035',
             ),
         ).toBe('ecash:qz2708636snqhsxu8wnlka78h6fdp77ar59jrf5035');
     });
     it(`Recognizes a wallet with missing Path1889 is a Legacy Wallet and requires migration`, () => {
         expect(isLegacyMigrationRequired(missingPath1899Wallet)).toBe(true);
     });
     it(`Recognizes a wallet with missing PublicKey in Path1889 is a Legacy Wallet and requires migration`, () => {
         expect(
             isLegacyMigrationRequired(missingPublicKeyInPath1899Wallet),
         ).toBe(true);
     });
     it(`Recognizes a wallet with missing PublicKey in Path145 is a Legacy Wallet and requires migration`, () => {
         expect(isLegacyMigrationRequired(missingPublicKeyInPath145Wallet)).toBe(
             true,
         );
     });
     it(`Recognizes a wallet with missing PublicKey in Path245 is a Legacy Wallet and requires migration`, () => {
         expect(isLegacyMigrationRequired(missingPublicKeyInPath245Wallet)).toBe(
             true,
         );
     });
     it(`Recognizes a wallet with missing Hash160 values is a Legacy Wallet and requires migration`, () => {
         expect(isLegacyMigrationRequired(missingHash160)).toBe(true);
     });
     it(`Recognizes a latest, current wallet that does not require migration`, () => {
         expect(isLegacyMigrationRequired(notLegacyWallet)).toBe(false);
     });
 
     test('toHash160() converts a valid bitcoincash: prefix address to a hash160', () => {
         const result = toHash160(
             'bitcoincash:qq9h6d0a5q65fgywv4ry64x04ep906mdku7ymranw3',
         );
         expect(result).toStrictEqual(
             '0b7d35fda03544a08e65464d54cfae4257eb6db7',
         );
     });
     test('toHash160 throws error if input address is an invalid bitcoincash: address', () => {
         const address = 'bitcoincash:qqd3qnINVALIDDDDDDDDDza25m';
 
         let errorThrown;
         try {
             toHash160(address);
         } catch (err) {
             errorThrown = err.message;
         }
         expect(errorThrown).toStrictEqual('Invalid address: ' + address + '.');
     });
     test('toHash160() converts a valid ecash: prefix address to a hash160', () => {
         const result = toHash160(
             'ecash:qq9h6d0a5q65fgywv4ry64x04ep906mdku8f0gxfgx',
         );
         expect(result).toStrictEqual(
             '0b7d35fda03544a08e65464d54cfae4257eb6db7',
         );
     });
     test('toHash160 throws error if input address is an invalid ecash address', () => {
         const address = 'ecash:qqd3qn4zINVALIDDDDDtfza25m';
 
         let errorThrown;
         try {
             toHash160(address);
         } catch (err) {
             errorThrown = err.message;
         }
         expect(errorThrown).toStrictEqual('Invalid address: ' + address + '.');
     });
     test('toHash160() converts a valid etoken: address to a hash160', () => {
         const result = toHash160(
             'etoken:qq9h6d0a5q65fgywv4ry64x04ep906mdkufhx2swv3',
         );
         expect(result).toStrictEqual(
             '0b7d35fda03544a08e65464d54cfae4257eb6db7',
         );
     });
     test('toHash160 throws error if input address is an invalid etoken: address', () => {
         const address = 'etoken:qq9h6d0a5INVALIDDDDDDx2swv3';
 
         let errorThrown;
         try {
             toHash160(address);
         } catch (err) {
             errorThrown = err.message;
         }
         expect(errorThrown).toStrictEqual('Invalid address: ' + address + '.');
     });
     test('toHash160() converts a valid simpleledger: address to a hash160', () => {
         const result = toHash160(
             'simpleledger:qq9h6d0a5q65fgywv4ry64x04ep906mdkujlscgns0',
         );
         expect(result).toStrictEqual(
             '0b7d35fda03544a08e65464d54cfae4257eb6db7',
         );
     });
     test('toHash160 throws error if input address is an invalid simpleledger: address', () => {
         const address = 'simpleledger:qq9h6d0a5qINVALIDDDjlscgns0';
 
         let errorThrown;
         try {
             toHash160(address);
         } catch (err) {
             errorThrown = err.message;
         }
         expect(errorThrown).toStrictEqual('Invalid address: ' + address + '.');
     });
 
     test('parseOpReturn() successfully parses a short cashtab message', async () => {
         const result = parseOpReturn(shortCashtabMessageInputHex);
         expect(result).toStrictEqual(mockParsedShortCashtabMessageArray);
     });
 
     test('parseOpReturn() successfully parses a long cashtab message where an additional PUSHDATA1 is present', async () => {
         const result = parseOpReturn(longCashtabMessageInputHex);
         expect(result).toStrictEqual(mockParsedLongCashtabMessageArray);
     });
 
     test('parseOpReturn() successfully parses a short external message', async () => {
         const result = parseOpReturn(shortExternalMessageInputHex);
         expect(result).toStrictEqual(mockParsedShortExternalMessageArray);
     });
 
     test('parseOpReturn() successfully parses a long external message where an additional PUSHDATA1 is present', async () => {
         const result = parseOpReturn(longExternalMessageInputHex);
         expect(result).toStrictEqual(mockParsedLongExternalMessageArray);
     });
 
     test('parseOpReturn() successfully parses an external message that is segmented into separate short parts', async () => {
         const result = parseOpReturn(shortSegmentedExternalMessageInputHex);
         expect(result).toStrictEqual(
             mockParsedShortSegmentedExternalMessageArray,
         );
     });
 
     test('parseOpReturn() successfully parses an external message that is segmented into separate long parts', async () => {
         const result = parseOpReturn(longSegmentedExternalMessageInputHex);
         expect(result).toStrictEqual(
             mockParsedLongSegmentedExternalMessageArray,
         );
     });
 
     test('parseOpReturn() successfully parses an external message that is segmented into separate long and short parts', async () => {
         const result = parseOpReturn(mixedSegmentedExternalMessageInputHex);
         expect(result).toStrictEqual(
             mockParsedMixedSegmentedExternalMessageArray,
         );
     });
 
     test('parseOpReturn() successfully parses an eToken output', async () => {
         const result = parseOpReturn(eTokenInputHex);
         expect(result).toStrictEqual(mockParsedETokenOutputArray);
     });
 
     test('parseOpReturn() successfully parses an airdrop transaction', async () => {
         const result = parseOpReturn(mockAirdropHexOutput);
         // verify the hex output is parsed correctly
         expect(result).toStrictEqual(mockParsedAirdropMessageArray);
         // verify airdrop hex prefix is contained in the array returned from parseOpReturn()
         expect(
             result.find(
                 element => element === opreturnConfig.appPrefixesHex.airdrop,
             ),
         ).toStrictEqual(opreturnConfig.appPrefixesHex.airdrop);
     });
 
     test('convertEtokenToEcashAddr successfully converts a valid eToken address to eCash', async () => {
         const result = convertEtokenToEcashAddr(
             'etoken:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gcldpffcs',
         );
         expect(result).toStrictEqual(
             'ecash:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gkpyrlwu8',
         );
     });
 
     test('convertEtokenToEcashAddr successfully converts prefixless eToken address as input', async () => {
         const result = convertEtokenToEcashAddr(
             'qpatql05s9jfavnu0tv6lkjjk25n6tmj9gcldpffcs',
         );
         expect(result).toStrictEqual(
             'ecash:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gkpyrlwu8',
         );
     });
 
     test('convertEtokenToEcashAddr throws error with an invalid eToken address as input', async () => {
         const result = convertEtokenToEcashAddr('etoken:qpj9gcldpffcs');
         expect(result).toStrictEqual(
             new Error(
                 'cashMethods.convertToEcashAddr() error: etoken:qpj9gcldpffcs is not a valid etoken address',
             ),
         );
     });
 
     test('convertEtokenToEcashAddr throws error with an ecash address as input', async () => {
         const result = convertEtokenToEcashAddr(
             'ecash:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gkpyrlwu8',
         );
         expect(result).toStrictEqual(
             new Error(
                 'cashMethods.convertToEcashAddr() error: ecash:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gkpyrlwu8 is not a valid etoken address',
             ),
         );
     });
 
     test('convertEtokenToEcashAddr throws error with null input', async () => {
         const result = convertEtokenToEcashAddr(null);
         expect(result).toStrictEqual(
             new Error(
                 'cashMethods.convertToEcashAddr() error: No etoken address provided',
             ),
         );
     });
 
     test('convertEtokenToEcashAddr throws error with empty string input', async () => {
         const result = convertEtokenToEcashAddr('');
         expect(result).toStrictEqual(
             new Error(
                 'cashMethods.convertToEcashAddr() error: No etoken address provided',
             ),
         );
     });
 
     test('convertEcashtoEtokenAddr successfully converts a valid ecash address into an etoken address', async () => {
         const eCashAddress = 'ecash:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gkpyrlwu8';
         const eTokenAddress =
             'etoken:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gcldpffcs';
         const result = convertEcashtoEtokenAddr(eCashAddress);
         expect(result).toStrictEqual(eTokenAddress);
     });
 
     test('convertEcashtoEtokenAddr successfully converts a valid prefix-less ecash address into an etoken address', async () => {
         const eCashAddress = 'qpatql05s9jfavnu0tv6lkjjk25n6tmj9gkpyrlwu8';
         const eTokenAddress =
             'etoken:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gcldpffcs';
         const result = convertEcashtoEtokenAddr(eCashAddress);
         expect(result).toStrictEqual(eTokenAddress);
     });
 
     test('convertEcashtoEtokenAddr throws error with invalid ecash address input', async () => {
         const eCashAddress = 'ecash:qpaNOTVALIDADDRESSwu8';
         const result = convertEcashtoEtokenAddr(eCashAddress);
         expect(result).toStrictEqual(
             new Error(eCashAddress + ' is not a valid ecash address'),
         );
     });
 
     test('convertEcashtoEtokenAddr throws error with a valid etoken address input', async () => {
         const eTokenAddress =
             'etoken:qpatql05s9jfavnu0tv6lkjjk25n6tmj9gcldpffcs';
         const result = convertEcashtoEtokenAddr(eTokenAddress);
         expect(result).toStrictEqual(
             new Error(eTokenAddress + ' is not a valid ecash address'),
         );
     });
 
     test('convertEcashtoEtokenAddr throws error with a valid bitcoincash address input', async () => {
         const bchAddress =
             'bitcoincash:qpatql05s9jfavnu0tv6lkjjk25n6tmj9g0vsgy56s';
         const result = convertEcashtoEtokenAddr(bchAddress);
         expect(result).toStrictEqual(
             new Error(bchAddress + ' is not a valid ecash address'),
         );
     });
 
     test('convertEcashtoEtokenPrefix throws error with null ecash address input', async () => {
         const eCashAddress = null;
         const result = convertEcashtoEtokenAddr(eCashAddress);
         expect(result).toStrictEqual(
             new Error(eCashAddress + ' is not a valid ecash address'),
         );
     });
 
     it(`flattenContactList flattens contactList array by returning an array of addresses`, () => {
         expect(
             flattenContactList([
                 {
                     address: 'ecash:qpdkc5p7f25hwkxsr69m3evlj4h7wqq9xcgmjc8sxr',
                     name: 'Alpha',
                 },
                 {
                     address: 'ecash:qpq235n3l3u6ampc8slapapnatwfy446auuv64ylt2',
                     name: 'Beta',
                 },
                 {
                     address: 'ecash:qz50e58nkeg2ej2f34z6mhwylp6ven8emy8pp52r82',
                     name: 'Gamma',
                 },
             ]),
         ).toStrictEqual([
             'ecash:qpdkc5p7f25hwkxsr69m3evlj4h7wqq9xcgmjc8sxr',
             'ecash:qpq235n3l3u6ampc8slapapnatwfy446auuv64ylt2',
             'ecash:qz50e58nkeg2ej2f34z6mhwylp6ven8emy8pp52r82',
         ]);
     });
 
     it(`flattenContactList flattens contactList array of length 1 by returning an array of 1 address`, () => {
         expect(
             flattenContactList([
                 {
                     address: 'ecash:qpdkc5p7f25hwkxsr69m3evlj4h7wqq9xcgmjc8sxr',
                     name: 'Alpha',
                 },
             ]),
         ).toStrictEqual(['ecash:qpdkc5p7f25hwkxsr69m3evlj4h7wqq9xcgmjc8sxr']);
     });
     it(`flattenContactList returns an empty array for invalid input`, () => {
         expect(flattenContactList(false)).toStrictEqual([]);
     });
     it(`getHashArrayFromWallet returns false for a legacy wallet`, () => {
         expect(
             getHashArrayFromWallet(mockLegacyWallets.legacyAlphaMainnet),
         ).toBe(false);
     });
     it(`Successfully extracts a hash160 array from a migrated wallet object`, () => {
         expect(
             getHashArrayFromWallet(
                 mockLegacyWallets.migratedLegacyAlphaMainnet,
             ),
         ).toStrictEqual([
             '960c9ed561f1699f0c49974d50b3bb7cdc118625',
             '2be0e0c999e7e77a443ea726f82c441912fca92b',
             'ba8257db65f40359989c7b894c5e88ed7b6344f6',
         ]);
     });
     it(`isActiveWebsocket returns true for an active chronik websocket connection`, () => {
         expect(isActiveWebsocket(activeWebsocketAlpha)).toBe(true);
     });
     it(`isActiveWebsocket returns false for a disconnected chronik websocket connection`, () => {
         expect(isActiveWebsocket(disconnectedWebsocketAlpha)).toBe(false);
     });
     it(`isActiveWebsocket returns false for a null chronik websocket connection`, () => {
         expect(isActiveWebsocket(null)).toBe(false);
     });
     it(`isActiveWebsocket returns false for an active websocket connection with no subscriptions`, () => {
         expect(isActiveWebsocket(unsubscribedWebsocket)).toBe(false);
     });
     it(`getCashtabByteCount for 2 inputs, 2 outputs returns the same value as BCH.BitcoinCash.getByteCount(
             { P2PKH: utxos.length },
             { P2PKH: p2pkhOutputNumber },
         );`, () => {
         expect(getCashtabByteCount(2, 2)).toBe(374);
     });
     it(`getCashtabByteCount for 1 input, 2 outputs returns the same value as BCH.BitcoinCash.getByteCount(
             { P2PKH: utxos.length },
             { P2PKH: p2pkhOutputNumber },
         );`, () => {
         expect(getCashtabByteCount(1, 2)).toBe(226);
     });
     it(`getCashtabByteCount for 173 input, 1 outputs returns the same value as BCH.BitcoinCash.getByteCount(
             { P2PKH: utxos.length },
             { P2PKH: p2pkhOutputNumber },
         );`, () => {
         expect(getCashtabByteCount(173, 1)).toBe(25648);
     });
     it(`getCashtabByteCount for 1 input, 2000 outputs returns the same value as BCH.BitcoinCash.getByteCount(
             { P2PKH: utxos.length },
             { P2PKH: p2pkhOutputNumber },
         );`, () => {
         expect(getCashtabByteCount(1, 2000)).toBe(68158);
     });
     it('calculates fee correctly for 2 P2PKH outputs', () => {
         const utxosMock = [{}, {}];
         expect(calcFee(utxosMock, 2, appConfig.defaultFee)).toBe(752);
     });
     it(`Converts a hash160 to an ecash address`, () => {
         expect(
             hash160ToAddress('76458db0ed96fe9863fc1ccec9fa2cfab884b0f6'),
         ).toBe('ecash:qpmytrdsakt0axrrlswvaj069nat3p9s7cjctmjasj');
     });
     it(`outputScriptToAddress determines P2PKH address type from output script and returns the ecash address`, () => {
         expect(
             outputScriptToAddress(
                 '76a914da45fd71b76e34c88e97ccbebb454d7cd395e52c88ac',
             ),
         ).toBe('ecash:qrdytlt3kahrfjywjlxtaw69f47d89099s393kne5c');
     });
     it(`outputScriptToAddress determines P2SH address type from output script and returns the ecash address`, () => {
         expect(
             outputScriptToAddress(
                 'a914c5e60aad8d98f298a76434750630dc1b46a2382187',
             ),
         ).toBe('ecash:prz7vz4d3kv09x98vs682p3smsd5dg3cyykjye6grt');
     });
     it(`outputScriptToAddress throws correct error for an output script that does not parse as P2PKH or P2SH`, () => {
         let thrownError;
         try {
             outputScriptToAddress('notAnOutputScript');
         } catch (err) {
             thrownError = err;
         }
         expect(thrownError.message).toBe('Unrecognized outputScript format');
     });
     it(`outputScriptToAddress throws correct error for an output script that for some reason is bracketed by P2PKH markers but is not a valid hash160`, () => {
         let thrownError;
         try {
             outputScriptToAddress(
                 '76a914da45fd71b76eeeeeeeee34c88e97ccbebb454d7cd395e52c88ac',
             );
         } catch (err) {
             thrownError = err;
         }
         expect(thrownError.message).toBe('Parsed hash160 is incorrect length');
     });
     it(`toSatoshis returns expected integers for all possible decimal places at extreme limit of XEC supply`, () => {
         const xecMaxSupply = 21000000000000;
         const xecMaxSupplyLessOne = xecMaxSupply - 1;
         for (let i = 1; i < 100; i += 1) {
             const thisDecimal = parseFloat((i / 100).toFixed(2));
             const testValue = xecMaxSupplyLessOne + thisDecimal;
             const result = toSatoshis(testValue);
             // Confirm you get an integer
             expect(Number.isInteger(result)).toBe(true);
             // Confirm you aren't rounding to the wrong decimal
             expect(result.toString().slice(-2)).toBe(
                 thisDecimal.toFixed(2).slice(-2),
             );
         }
     });
 });
diff --git a/cashtab/src/utils/cashMethods.js b/cashtab/src/utils/cashMethods.js
index fe8531737..10da56401 100644
--- a/cashtab/src/utils/cashMethods.js
+++ b/cashtab/src/utils/cashMethods.js
@@ -1,1352 +1,1237 @@
 import {
     isValidXecAddress,
     isValidEtokenAddress,
     isValidContactList,
     isValidBchAddress,
 } from 'utils/validation';
 import BigNumber from 'bignumber.js';
 import cashaddr from 'ecashaddrjs';
 import bs58 from 'bs58';
 import * as slpMdm from 'slp-mdm';
 import * as utxolib from '@bitgo/utxo-lib';
 import { opReturn as opreturnConfig } from 'config/opreturn';
 import appConfig from 'config/app';
 
-export const getMessageByteSize = (
-    msgInputStr,
-    encryptionFlag,
-    encryptedEj,
-) => {
-    if (!msgInputStr || msgInputStr.trim() === '') {
-        return 0;
-    }
-
-    // generate the OP_RETURN script
-    let opReturnData;
-    if (encryptionFlag && encryptedEj) {
-        opReturnData = generateOpReturnScript(
-            msgInputStr,
-            encryptionFlag, // encryption flag
-            false, // airdrop use
-            null, // airdrop use
-            encryptedEj, // serialized encryption data object
-            false, // alias registration flag
-        );
-    } else {
-        opReturnData = generateOpReturnScript(
-            msgInputStr,
-            encryptionFlag, // encryption use
-            false, // airdrop use
-            null, // airdrop use
-            null, // serialized encryption data object
-            false, // alias registration flag
-        );
-    }
-    // extract the msg input from the OP_RETURN script and check the backend size
-    const hexString = opReturnData.toString('hex'); // convert to hex
-    const opReturnMsg = parseOpReturn(hexString)[1]; // extract the message
-    const msgInputByteSize = opReturnMsg.length / 2; // calculate the byte size
-
-    return msgInputByteSize;
-};
-
 // function is based on BCH-JS' generateBurnOpReturn() however it's been trimmed down for Cashtab use
 // Reference: https://github.com/Permissionless-Software-Foundation/bch-js/blob/62e56c832b35731880fe448269818b853c76dd80/src/slp/tokentype1.js#L217
 export const generateBurnOpReturn = (tokenUtxos, burnQty) => {
     try {
         if (!tokenUtxos || !burnQty) {
             throw new Error('Invalid burn token parameter');
         } // sendToken component already prevents burning of a value greater than the token utxo total held by the wallet
 
         const tokenId = tokenUtxos[0].tokenId;
         const decimals = tokenUtxos[0].decimals;
 
         // account for token decimals
         const finalBurnTokenQty = new BigNumber(burnQty).times(10 ** decimals);
 
         // Calculate the total amount of tokens owned by the wallet.
         const totalTokens = tokenUtxos.reduce(
             (tot, txo) =>
                 tot.plus(new BigNumber(txo.tokenQty).times(10 ** decimals)),
             new BigNumber(0),
         );
 
         // calculate the token change
         const tokenChange = totalTokens.minus(finalBurnTokenQty);
         const tokenChangeStr = tokenChange.toString();
 
         // Generate the burn OP_RETURN as a Buffer
         // No need for separate .send() calls for change and non-change burns as
         // nil change values do not generate token outputs as the full balance is burnt
         const script = slpMdm.TokenType1.send(tokenId, [
             new slpMdm.BN(tokenChangeStr),
         ]);
 
         return script;
     } catch (err) {
         console.log('Error in generateBurnOpReturn(): ' + err);
         throw err;
     }
 };
 
 // Function originally based on BCH-JS' generateSendOpReturn function however trimmed down for Cashtab
 // Reference: https://github.com/Permissionless-Software-Foundation/bch-js/blob/62e56c832b35731880fe448269818b853c76dd80/src/slp/tokentype1.js#L95
 export const generateSendOpReturn = (tokenUtxos, sendQty) => {
     try {
         if (!tokenUtxos || !sendQty) {
             throw new Error('Invalid send token parameter');
         }
 
         const tokenId = tokenUtxos[0].tokenId;
         const decimals = tokenUtxos[0].decimals;
 
         // account for token decimals
         const finalSendTokenQty = new BigNumber(sendQty).times(10 ** decimals);
         const finalSendTokenQtyStr = finalSendTokenQty.toString();
 
         // Calculate the total amount of tokens owned by the wallet.
         const totalTokens = tokenUtxos.reduce(
             (tot, txo) =>
                 tot.plus(new BigNumber(txo.tokenQty).times(10 ** decimals)),
             new BigNumber(0),
         );
 
         // calculate token change
         const tokenChange = totalTokens.minus(finalSendTokenQty);
         const tokenChangeStr = tokenChange.toString();
 
         // When token change output is required
         let script, outputs;
         if (tokenChange > 0) {
             outputs = 2;
             // Generate the OP_RETURN as a Buffer.
             script = slpMdm.TokenType1.send(tokenId, [
                 new slpMdm.BN(finalSendTokenQtyStr),
                 new slpMdm.BN(tokenChangeStr),
             ]);
         } else {
             // no token change needed
             outputs = 1;
             // Generate the OP_RETURN as a Buffer.
             script = slpMdm.TokenType1.send(tokenId, [
                 new slpMdm.BN(finalSendTokenQtyStr),
             ]);
         }
 
         return { script, outputs };
     } catch (err) {
         console.log('Error in generateSendOpReturn(): ' + err);
         throw err;
     }
 };
 
 // function is based on BCH-JS' generateGenesisOpReturn() however it's been trimmed down for Cashtab use
 // Reference: https://github.com/Permissionless-Software-Foundation/bch-js/blob/62e56c832b35731880fe448269818b853c76dd80/src/slp/tokentype1.js#L286
 export const generateGenesisOpReturn = configObj => {
     try {
         if (!configObj) {
             throw new Error('Invalid token configuration');
         }
 
         // adjust initial quantity for token decimals
         const initialQty = new BigNumber(configObj.initialQty)
             .times(10 ** configObj.decimals)
             .toString();
 
         const script = slpMdm.TokenType1.genesis(
             configObj.ticker,
             configObj.name,
             configObj.documentUrl,
             configObj.documentHash,
             configObj.decimals,
             configObj.mintBatonVout,
             new slpMdm.BN(initialQty),
         );
 
         return script;
     } catch (err) {
         console.log('Error in generateGenesisOpReturn(): ' + err);
         throw err;
     }
 };
 
 export const getUtxoWif = (utxo, wallet) => {
     if (!wallet) {
         throw new Error('Invalid wallet parameter');
     }
     const accounts = [wallet.Path245, wallet.Path145, wallet.Path1899];
     const wif = accounts
         .filter(acc => acc.cashAddress === utxo.address)
         .pop().fundingWif;
     return wif;
 };
 
 export const signUtxosByAddress = (inputUtxos, wallet, txBuilder) => {
     for (let i = 0; i < inputUtxos.length; i++) {
         const utxo = inputUtxos[i];
         const accounts = [wallet.Path245, wallet.Path145, wallet.Path1899];
 
         const wif = accounts
             .filter(acc => acc.cashAddress === utxo.address)
             .pop().fundingWif;
 
         const utxoECPair = utxolib.ECPair.fromWIF(wif, utxolib.networks.ecash);
 
         // Specify hash type
         // This should be handled at the utxo-lib level, pending latest published version
         const hashTypes = {
             SIGHASH_ALL: 0x01,
             SIGHASH_FORKID: 0x40,
         };
 
         txBuilder.sign(
             i, // vin
             utxoECPair, // keyPair
             undefined, // redeemScript
             hashTypes.SIGHASH_ALL | hashTypes.SIGHASH_FORKID, // hashType
             parseInt(utxo.value), // value
         );
     }
 
     return txBuilder;
 };
 
 export const getCashtabByteCount = (p2pkhInputCount, p2pkhOutputCount) => {
     // Simplifying bch-js function for P2PKH txs only, as this is all Cashtab supports for now
     // https://github.com/Permissionless-Software-Foundation/bch-js/blob/master/src/bitcoincash.js#L408
     // The below magic numbers refer to:
     // const types = {
     //     inputs: {
     //         'P2PKH': 148 * 4,
     //     },
     //     outputs: {
     //         P2PKH: 34 * 4,
     //     },
     // };
 
     const inputCount = new BigNumber(p2pkhInputCount);
     const outputCount = new BigNumber(p2pkhOutputCount);
     const inputWeight = new BigNumber(148 * 4);
     const outputWeight = new BigNumber(34 * 4);
     const nonSegwitWeightConstant = new BigNumber(10 * 4);
     let totalWeight = new BigNumber(0);
     totalWeight = totalWeight
         .plus(inputCount.times(inputWeight))
         .plus(outputCount.times(outputWeight))
         .plus(nonSegwitWeightConstant);
     const byteCount = totalWeight.div(4).integerValue(BigNumber.ROUND_CEIL);
 
     return Number(byteCount);
 };
 
 export const calcFee = (
     utxos,
     p2pkhOutputNumber = 2,
     satoshisPerByte = appConfig.defaultFee,
     opReturnByteCount = 0,
 ) => {
     const byteCount = getCashtabByteCount(utxos.length, p2pkhOutputNumber);
     const txFee = Math.ceil(satoshisPerByte * (byteCount + opReturnByteCount));
     return txFee;
 };
 
 export const generateTokenTxOutput = (
     txBuilder,
     tokenAction,
     legacyCashOriginAddress,
     tokenUtxosBeingSpent = [], // optional - send or burn tx only
     remainderXecValue = new BigNumber(0), // optional - only if > dust
     tokenConfigObj = {}, // optional - genesis only
     tokenRecipientAddress = false, // optional - send tx only
     tokenAmount = false, // optional - send or burn amount for send/burn tx only
 ) => {
     try {
         if (!tokenAction || !legacyCashOriginAddress || !txBuilder) {
             throw new Error('Invalid token tx output parameter');
         }
 
         let script, opReturnObj, destinationAddress;
         switch (tokenAction) {
             case 'GENESIS':
                 script = generateGenesisOpReturn(tokenConfigObj);
                 destinationAddress = legacyCashOriginAddress;
                 break;
             case 'SEND':
                 opReturnObj = generateSendOpReturn(
                     tokenUtxosBeingSpent,
                     tokenAmount.toString(),
                 );
                 script = opReturnObj.script;
                 destinationAddress = tokenRecipientAddress;
                 break;
             case 'BURN':
                 script = generateBurnOpReturn(
                     tokenUtxosBeingSpent,
                     tokenAmount,
                 );
                 destinationAddress = legacyCashOriginAddress;
                 break;
             default:
                 throw new Error('Invalid token transaction type');
         }
 
         // OP_RETURN needs to be the first output in the transaction.
         txBuilder.addOutput(script, 0);
 
         // add XEC dust output as fee for genesis, send or burn token output
         txBuilder.addOutput(
             cashaddr.toLegacy(destinationAddress),
             parseInt(appConfig.etokenSats),
         );
 
         // Return any token change back to the sender for send and burn txs
         if (
             tokenAction !== 'GENESIS' ||
             (opReturnObj && opReturnObj.outputs > 1)
         ) {
             // add XEC dust output as fee
             txBuilder.addOutput(
                 cashaddr.toLegacy(tokenUtxosBeingSpent[0].address), // etoken address
                 parseInt(appConfig.etokenSats),
             );
         }
 
         // Send xec change to own address
         if (remainderXecValue.gte(new BigNumber(appConfig.dustSats))) {
             txBuilder.addOutput(
                 cashaddr.toLegacy(legacyCashOriginAddress),
                 parseInt(remainderXecValue),
             );
         }
     } catch (err) {
         console.log(`generateTokenTxOutput() error: ` + err);
         throw err;
     }
 
     return txBuilder;
 };
 
 export const generateTxInput = (
     isOneToMany,
     utxos,
     txBuilder,
     destinationAddressAndValueArray,
     satoshisToSend,
     feeInSatsPerByte,
     opReturnByteCount,
 ) => {
     let txInputObj = {};
     const inputUtxos = [];
     let txFee = 0;
     let totalInputUtxoValue = new BigNumber(0);
     try {
         if (
             (isOneToMany && !destinationAddressAndValueArray) ||
             !utxos ||
             !txBuilder ||
             !satoshisToSend ||
             !feeInSatsPerByte
         ) {
             throw new Error('Invalid tx input parameter');
         }
 
         // A normal tx will have 2 outputs, destination and change
         // A one to many tx will have n outputs + 1 change output, where n is the number of recipients
         const txOutputs = isOneToMany
             ? destinationAddressAndValueArray.length + 1
             : 2;
         for (let i = 0; i < utxos.length; i++) {
             const utxo = utxos[i];
             totalInputUtxoValue = totalInputUtxoValue.plus(utxo.value);
             const vout = utxo.outpoint.outIdx;
             const txid = utxo.outpoint.txid;
             // add input with txid and index of vout
             txBuilder.addInput(txid, vout);
 
             inputUtxos.push(utxo);
             txFee = calcFee(
                 inputUtxos,
                 txOutputs,
                 feeInSatsPerByte,
                 opReturnByteCount,
             );
 
             if (totalInputUtxoValue.minus(satoshisToSend).minus(txFee).gte(0)) {
                 break;
             }
         }
     } catch (err) {
         console.log(`generateTxInput() error: ` + err);
         throw err;
     }
     txInputObj.txBuilder = txBuilder;
     txInputObj.totalInputUtxoValue = totalInputUtxoValue;
     txInputObj.inputUtxos = inputUtxos;
     txInputObj.txFee = txFee;
     return txInputObj;
 };
 
 export const generateTokenTxInput = (
     tokenAction, // GENESIS, SEND or BURN
     totalXecUtxos,
     totalTokenUtxos,
     tokenId,
     tokenAmount, // optional - only for sending or burning
     feeInSatsPerByte,
     txBuilder,
 ) => {
     let totalXecInputUtxoValue = new BigNumber(0);
     let remainderXecValue = new BigNumber(0);
     let remainderTokenValue = new BigNumber(0);
     let totalXecInputUtxos = [];
     let txFee = 0;
     let tokenUtxosBeingSpent = [];
 
     try {
         if (
             !tokenAction ||
             !totalXecUtxos ||
             (tokenAction !== 'GENESIS' && !tokenId) ||
             !feeInSatsPerByte ||
             !txBuilder
         ) {
             throw new Error('Invalid token tx input parameter');
         }
 
         // collate XEC UTXOs for this token tx
         const txOutputs =
             tokenAction === 'GENESIS'
                 ? 2 // one for genesis OP_RETURN output and one for change
                 : 4; // for SEND/BURN token txs see T2645 on why this is not dynamically generated
         for (let i = 0; i < totalXecUtxos.length; i++) {
             const thisXecUtxo = totalXecUtxos[i];
             totalXecInputUtxoValue = totalXecInputUtxoValue.plus(
                 new BigNumber(thisXecUtxo.value),
             );
             const vout = thisXecUtxo.outpoint.outIdx;
             const txid = thisXecUtxo.outpoint.txid;
             // add input with txid and index of vout
             txBuilder.addInput(txid, vout);
 
             totalXecInputUtxos.push(thisXecUtxo);
             txFee = calcFee(totalXecInputUtxos, txOutputs, feeInSatsPerByte);
 
             remainderXecValue =
                 tokenAction === 'GENESIS'
                     ? totalXecInputUtxoValue
                           .minus(new BigNumber(appConfig.etokenSats))
                           .minus(new BigNumber(txFee))
                     : totalXecInputUtxoValue
                           .minus(new BigNumber(appConfig.etokenSats * 2)) // one for token send/burn output, one for token change
                           .minus(new BigNumber(txFee));
 
             if (remainderXecValue.gte(0)) {
                 break;
             }
         }
 
         if (remainderXecValue.lt(0)) {
             throw new Error(`Insufficient funds`);
         }
 
         let filteredTokenInputUtxos = [];
         let finalTokenAmountSpent = new BigNumber(0);
         let tokenAmountBeingSpent = new BigNumber(tokenAmount);
 
         if (tokenAction === 'SEND' || tokenAction === 'BURN') {
             // filter for token UTXOs matching the token being sent/burnt
             filteredTokenInputUtxos = totalTokenUtxos.filter(utxo => {
                 if (
                     utxo && // UTXO is associated with a token.
                     utxo.slpMeta.tokenId === tokenId && // UTXO matches the token ID.
                     !utxo.slpToken.isMintBaton // UTXO is not a minting baton.
                 ) {
                     return true;
                 }
                 return false;
             });
             if (filteredTokenInputUtxos.length === 0) {
                 throw new Error(
                     'No token UTXOs for the specified token could be found.',
                 );
             }
 
             // collate token UTXOs to cover the token amount being sent/burnt
             for (let i = 0; i < filteredTokenInputUtxos.length; i++) {
                 finalTokenAmountSpent = finalTokenAmountSpent.plus(
                     new BigNumber(filteredTokenInputUtxos[i].tokenQty),
                 );
                 txBuilder.addInput(
                     filteredTokenInputUtxos[i].outpoint.txid,
                     filteredTokenInputUtxos[i].outpoint.outIdx,
                 );
                 tokenUtxosBeingSpent.push(filteredTokenInputUtxos[i]);
                 if (tokenAmountBeingSpent.lte(finalTokenAmountSpent)) {
                     break;
                 }
             }
 
             // calculate token change
             remainderTokenValue = finalTokenAmountSpent.minus(
                 new BigNumber(tokenAmount),
             );
             if (remainderTokenValue.lt(0)) {
                 throw new Error(
                     'Insufficient token UTXOs for the specified token amount.',
                 );
             }
         }
     } catch (err) {
         console.log(`generateTokenTxInput() error: ` + err);
         throw err;
     }
 
     return {
         txBuilder: txBuilder,
         inputXecUtxos: totalXecInputUtxos,
         inputTokenUtxos: tokenUtxosBeingSpent,
         remainderXecValue: remainderXecValue,
         remainderTokenValue: remainderTokenValue,
     };
 };
 
 export const getChangeAddressFromInputUtxos = (inputUtxos, wallet) => {
     if (!inputUtxos || !wallet) {
         throw new Error('Invalid getChangeAddressFromWallet input parameter');
     }
 
     // Assume change address is input address of utxo at index 0
     let changeAddress;
 
     // Validate address
     try {
         changeAddress = inputUtxos[0].address;
         if (
             !isValidXecAddress(changeAddress) &&
             !isValidBchAddress(changeAddress)
         ) {
             throw new Error('Invalid change address');
         }
     } catch (err) {
         throw new Error('Invalid input utxo');
     }
     return changeAddress;
 };
 
 /**
  * Get the total XEC amount sent in a one-to-many XEC tx
  * @param {array} destinationAddressAndValueArray
  * Array constructed by user input of addresses and values
  * e.g. [
  *  "<address>, <value>",
  *   "<address>, <value>"
  *  ]
  * @returns {number} total value of XEC
  */
 export const sumOneToManyXec = destinationAddressAndValueArray => {
     return destinationAddressAndValueArray.reduce((prev, curr) => {
         return parseFloat(prev) + parseFloat(curr.split(',')[1]);
     }, 0);
 };
 
 /**
  * Return an integer that is the given amountXEC in satoshis
  * @param {Number} amountXec
  * @returns {Integer} satoshis
  */
 export const toSatoshis = amountXec => {
     const SATOSHIS_PER_XEC = 100;
     // Math.round returns the nearest integer value
     // e.g. in JS, 151.52 * 100 = 15152.000000000002
     // We need to return 15152
     return Math.round(SATOSHIS_PER_XEC * amountXec);
 };
 /*
  * Generates an OP_RETURN script for a version 0 alias registration tx
  *
  * Returns the final encoded script object ready to be added as a transaction output
  */
 export const generateAliasOpReturnScript = (alias, address) => {
     // Note: utxolib.script.compile(script) will add pushdata bytes for each buffer
     // utxolib.script.compile(script) will not add pushdata bytes for raw data
 
     // Initialize script array with OP_RETURN byte (6a) as rawdata (i.e. you want compiled result of 6a, not 016a)
     let script = [opreturnConfig.opReturnPrefixDec];
 
     // Push alias protocol identifier
     script.push(
         Buffer.from(opreturnConfig.appPrefixesHex.aliasRegistration, 'hex'), // '.xec'
     );
 
     // Push alias protocol tx version to stack
     // Per spec, push this as OP_0
     script.push(0);
 
     // Push alias to the stack
     script.push(Buffer.from(alias, 'utf8'));
 
     // Get the type and hash of the address in string format
     const { type, hash } = cashaddr.decode(address, true);
 
     // Determine address type and corresponding address version byte
     let addressVersionByte;
     // Version bytes per cashaddr spec,https://github.com/bitcoincashorg/bitcoincash.org/blob/master/spec/cashaddr.md
     if (type === 'p2pkh') {
         addressVersionByte = '00'; // one byte 0 in hex
     } else if (type === 'p2sh') {
         addressVersionByte = '08'; // one byte 8 in hex
     } else {
         throw new Error('Unsupported address type');
     }
 
     // Push <addressVersionByte> and <addressPayload>
     script.push(Buffer.from(`${addressVersionByte}${hash}`, 'hex'));
 
     return utxolib.script.compile(script);
 };
-/*
- * Generates an OP_RETURN script to reflect the various send XEC permutations
- * involving messaging, encryption, eToken IDs and airdrop flags.
- *
- * Returns the final encoded script object
- */
-export const generateOpReturnScript = (
-    optionalOpReturnMsg,
-    encryptionFlag,
-    airdropFlag,
-    airdropTokenId,
-    encryptedEj,
-    optionalAliasRegistrationFlag = false,
-) => {
-    // encrypted mesage is mandatory when encryptionFlag is true
-    // airdrop token id is mandatory when airdropFlag is true
-    if ((encryptionFlag && !encryptedEj) || (airdropFlag && !airdropTokenId)) {
-        throw new Error('Invalid OP RETURN script input');
-    }
-
-    // Note: script.push(Buffer.from(opreturnConfig.opReturnPrefixHex, 'hex')); actually evaluates to '016a'
-    // instead of keeping the hex string intact. This behavour is specific to the initial script array element.
-    // To get around this, the bch-js approach of directly using the opReturn prefix in decimal form for the initial entry is used here.
-    let script = [opreturnConfig.opReturnPrefixDec]; // initialize script with the OP_RETURN op code (6a) in decimal form (106)
-
-    try {
-        if (encryptionFlag) {
-            // if the user has opted to encrypt this message
-
-            // add the encrypted cashtab messaging prefix and encrypted msg to script
-            script.push(
-                Buffer.from(
-                    opreturnConfig.appPrefixesHex.cashtabEncrypted,
-                    'hex',
-                ), // 65746162
-            );
-
-            // add the encrypted message to script
-            script.push(Buffer.from(encryptedEj));
-        } else {
-            // this is an un-encrypted message
-
-            if (airdropFlag) {
-                // if this was routed from the airdrop component
-                // add the airdrop prefix to script
-                script.push(
-                    Buffer.from(opreturnConfig.appPrefixesHex.airdrop, 'hex'), // drop
-                );
-                // add the airdrop token ID to script
-                script.push(Buffer.from(airdropTokenId, 'hex'));
-            }
-
-            if (optionalAliasRegistrationFlag) {
-                script.push(
-                    Buffer.from(
-                        opreturnConfig.appPrefixesHex.aliasRegistration,
-                        'hex',
-                    ), // '.xec'
-                );
-            } else {
-                // add the cashtab prefix to script
-                script.push(
-                    Buffer.from(opreturnConfig.appPrefixesHex.cashtab, 'hex'), // 00746162
-                );
-            }
-            // add the un-encrypted message to script if supplied
-            if (optionalOpReturnMsg) {
-                script.push(Buffer.from(optionalOpReturnMsg));
-            }
-        }
-    } catch (err) {
-        console.log('Error in generateOpReturnScript(): ' + err);
-        throw err;
-    }
-
-    return utxolib.script.compile(script);
-};
 
 export const generateTxOutput = (
     isOneToMany,
     singleSendValue,
     satoshisToSend,
     totalInputUtxoValue,
     destinationAddress,
     destinationAddressAndValueArray,
     changeAddress,
     txFee,
     txBuilder,
 ) => {
     try {
         if (
             (isOneToMany && !destinationAddressAndValueArray) ||
             (!isOneToMany && !destinationAddress && !singleSendValue) ||
             !changeAddress ||
             !satoshisToSend ||
             !totalInputUtxoValue ||
             !txFee ||
             !txBuilder
         ) {
             throw new Error('Invalid tx input parameter');
         }
 
         // amount to send back to the remainder address.
         const remainder = new BigNumber(totalInputUtxoValue)
             .minus(satoshisToSend)
             .minus(txFee);
         if (remainder.lt(0)) {
             throw new Error(`Insufficient funds`);
         }
 
         if (isOneToMany) {
             // for one to many mode, add the multiple outputs from the array
             let arrayLength = destinationAddressAndValueArray.length;
             for (let i = 0; i < arrayLength; i++) {
                 // add each send tx from the array as an output
                 let outputAddress = destinationAddressAndValueArray[i]
                     .split(',')[0]
                     .trim();
                 let outputValue = new BigNumber(
                     destinationAddressAndValueArray[i].split(',')[1],
                 );
                 txBuilder.addOutput(
                     cashaddr.toLegacy(outputAddress),
                     parseInt(fromXecToSatoshis(outputValue)),
                 );
             }
         } else {
             // for one to one mode, add output w/ single address and amount to send
             txBuilder.addOutput(
                 cashaddr.toLegacy(destinationAddress),
                 parseInt(fromXecToSatoshis(singleSendValue)),
             );
         }
 
         // if a remainder exists, return to change address as the final output
         if (remainder.gte(new BigNumber(appConfig.dustSats))) {
             txBuilder.addOutput(
                 cashaddr.toLegacy(changeAddress),
                 parseInt(remainder),
             );
         }
     } catch (err) {
         console.log('Error in generateTxOutput(): ' + err);
         throw err;
     }
 
     return txBuilder;
 };
 
 export const signAndBuildTx = (inputUtxos, txBuilder, wallet) => {
     if (!inputUtxos || inputUtxos.length === 0 || !txBuilder || !wallet) {
         throw new Error('Invalid buildTx parameter');
     }
 
     // Sign each XEC UTXO being consumed and refresh transactionBuilder
     txBuilder = signUtxosByAddress(inputUtxos, wallet, txBuilder);
 
     let hex;
     try {
         // build tx
         const tx = txBuilder.build();
         // output rawhex
         hex = tx.toHex();
     } catch (err) {
         throw new Error('Transaction build failed');
     }
     return hex;
 };
 
 export function parseOpReturn(hexStr) {
     if (
         !hexStr ||
         typeof hexStr !== 'string' ||
         hexStr.substring(0, 2) !== opreturnConfig.opReturnPrefixHex
     ) {
         return false;
     }
 
     hexStr = hexStr.slice(2); // remove the first byte i.e. 6a
 
     /*
      * @Return: resultArray is structured as follows:
      *  resultArray[0] is the transaction type i.e. eToken prefix, cashtab prefix, external message itself if unrecognized prefix
      *  resultArray[1] is the actual cashtab message or the 2nd part of an external message
      *  resultArray[2 - n] are the additional messages for future protcols
      */
     let resultArray = [];
     let message = '';
     let hexStrLength = hexStr.length;
 
     for (let i = 0; hexStrLength !== 0; i++) {
         // part 1: check the preceding byte value for the subsequent message
         let byteValue = hexStr.substring(0, 2);
         let msgByteSize = 0;
         if (byteValue === opreturnConfig.opPushDataOne) {
             // if this byte is 4c then the next byte is the message byte size - retrieve the message byte size only
             msgByteSize = parseInt(hexStr.substring(2, 4), 16); // hex base 16 to decimal base 10
             hexStr = hexStr.slice(4); // strip the 4c + message byte size info
         } else {
             // take the byte as the message byte size
             msgByteSize = parseInt(hexStr.substring(0, 2), 16); // hex base 16 to decimal base 10
             hexStr = hexStr.slice(2); // strip the message byte size info
         }
 
         // part 2: parse the subsequent message based on bytesize
         const msgCharLength = 2 * msgByteSize;
         message = hexStr.substring(0, msgCharLength);
         if (i === 0 && message === opreturnConfig.appPrefixesHex.eToken) {
             // add the extracted eToken prefix to array then exit loop
             resultArray[i] = opreturnConfig.appPrefixesHex.eToken;
             break;
         } else if (
             i === 0 &&
             message === opreturnConfig.appPrefixesHex.cashtab
         ) {
             // add the extracted Cashtab prefix to array
             resultArray[i] = opreturnConfig.appPrefixesHex.cashtab;
         } else if (
             i === 0 &&
             message === opreturnConfig.appPrefixesHex.cashtabEncrypted
         ) {
             // add the Cashtab encryption prefix to array
             resultArray[i] = opreturnConfig.appPrefixesHex.cashtabEncrypted;
         } else if (
             i === 0 &&
             message === opreturnConfig.appPrefixesHex.airdrop
         ) {
             // add the airdrop prefix to array
             resultArray[i] = opreturnConfig.appPrefixesHex.airdrop;
         } else {
             // this is either an external message or a subsequent cashtab message loop to extract the message
             resultArray[i] = message;
         }
 
         // strip out the parsed message
         hexStr = hexStr.slice(msgCharLength);
         hexStrLength = hexStr.length;
     }
     return resultArray;
 }
 
 export const fromLegacyDecimals = (
     amount,
     cashDecimals = appConfig.cashDecimals,
 ) => {
     // Input 0.00000546 BCH
     // Output 5.46 XEC or 0.00000546 BCH, depending on appConfig.cashDecimals
     const amountBig = new BigNumber(amount);
     const conversionFactor = new BigNumber(10 ** (8 - cashDecimals));
     const amountSmallestDenomination = amountBig
         .times(conversionFactor)
         .toNumber();
     return amountSmallestDenomination;
 };
 
 export const fromSatoshisToXec = (
     amount,
     cashDecimals = appConfig.cashDecimals,
 ) => {
     const amountBig = new BigNumber(amount);
     const multiplier = new BigNumber(10 ** (-1 * cashDecimals));
     const amountInBaseUnits = amountBig.times(multiplier);
     return amountInBaseUnits;
 };
 
 export const fromXecToSatoshis = (
     sendAmount,
     cashDecimals = appConfig.cashDecimals,
 ) => {
     // Replace the BCH.toSatoshi method with an equivalent function that works for arbitrary decimal places
     // Example, for an 8 decimal place currency like Bitcoin
     // Input: a BigNumber of the amount of Bitcoin to be sent
     // Output: a BigNumber of the amount of satoshis to be sent, or false if input is invalid
 
     // Validate
     // Input should be a BigNumber with no more decimal places than cashDecimals
     const isValidSendAmount =
         BigNumber.isBigNumber(sendAmount) && sendAmount.dp() <= cashDecimals;
     if (!isValidSendAmount) {
         return false;
     }
     const conversionFactor = new BigNumber(10 ** cashDecimals);
     const sendAmountSmallestDenomination = sendAmount.times(conversionFactor);
     return sendAmountSmallestDenomination;
 };
 
 export const flattenContactList = contactList => {
     /*
     Converts contactList from array of objects of type {address: <valid XEC address>, name: <string>} to array of addresses only
 
     If contact list is invalid, returns and empty array
     */
     if (!isValidContactList(contactList)) {
         return [];
     }
     let flattenedContactList = [];
     for (let i = 0; i < contactList.length; i += 1) {
         const thisAddress = contactList[i].address;
         flattenedContactList.push(thisAddress);
     }
     return flattenedContactList;
 };
 
 export const loadStoredWallet = walletStateFromStorage => {
     // Accept cached tokens array that does not save BigNumber type of BigNumbers
     // Return array with BigNumbers converted
     // See BigNumber.js api for how to create a BigNumber object from an object
     // https://mikemcl.github.io/bignumber.js/
     const liveWalletState =
         typeof walletStateFromStorage !== 'undefined'
             ? walletStateFromStorage
             : {};
 
     const keysInLiveWalletState = Object.keys(liveWalletState);
 
     // Newly created wallets may not have a state field
 
     // You only need to do this if you are loading a wallet
     // that hasn't yet saved tokens[i].balance as a string
     // instead of a BigNumber
     if (keysInLiveWalletState.includes('tokens')) {
         const { tokens } = liveWalletState;
         if (
             tokens.length > 0 &&
             tokens[0] &&
             tokens[0].balance &&
             typeof tokens[0].balance !== 'string'
         ) {
             for (let i = 0; i < tokens.length; i += 1) {
                 const thisTokenBalance = tokens[i].balance;
                 thisTokenBalance._isBigNumber = true;
                 tokens[i].balance = new BigNumber(thisTokenBalance);
             }
         }
     }
 
     // Also confirm balance is correct
     // Necessary step in case appConfig.decimals changed since last startup
     let nonSlpUtxosToParseForBalance;
     let balancesRebased;
     if (keysInLiveWalletState.length !== 0) {
         if (keysInLiveWalletState.includes('slpBalancesAndUtxos')) {
             // If this wallet still includes the wallet.state.slpBalancesAndUtxos field
             nonSlpUtxosToParseForBalance =
                 liveWalletState.slpBalancesAndUtxos.nonSlpUtxos;
         } else {
             nonSlpUtxosToParseForBalance = liveWalletState.nonSlpUtxos;
         }
         balancesRebased = getWalletBalanceFromUtxos(
             nonSlpUtxosToParseForBalance,
         );
     } else {
         balancesRebased = {
             totalBalanceInSatoshis: '0',
             totalBalance: '0',
         };
     }
 
     liveWalletState.balances = balancesRebased;
     return liveWalletState;
 };
 
 export const getWalletBalanceFromUtxos = nonSlpUtxos => {
     const totalBalanceInSatoshis = nonSlpUtxos.reduce(
         (previousBalance, utxo) =>
             previousBalance.plus(new BigNumber(utxo.value)),
         new BigNumber(0),
     );
     return {
         totalBalanceInSatoshis: totalBalanceInSatoshis.toString(),
         totalBalance: fromSatoshisToXec(totalBalanceInSatoshis).toString(),
     };
 };
 
 export const isValidStoredWallet = walletStateFromStorage => {
     return (
         typeof walletStateFromStorage === 'object' &&
         'state' in walletStateFromStorage &&
         'mnemonic' in walletStateFromStorage &&
         'name' in walletStateFromStorage &&
         'Path245' in walletStateFromStorage &&
         'Path145' in walletStateFromStorage &&
         'Path1899' in walletStateFromStorage &&
         typeof walletStateFromStorage.state === 'object' &&
         'balances' in walletStateFromStorage.state &&
         !('hydratedUtxoDetails' in walletStateFromStorage.state) &&
         ('slpBalancesAndUtxos' in walletStateFromStorage.state ||
             ('slpUtxos' in walletStateFromStorage.state &&
                 'nonSlpUtxos' in walletStateFromStorage.state)) &&
         'tokens' in walletStateFromStorage.state
     );
 };
 
 export const getWalletState = wallet => {
     if (!wallet || !wallet.state) {
         return {
             balances: { totalBalance: 0, totalBalanceInSatoshis: 0 },
             hydratedUtxoDetails: {},
             tokens: [],
             slpUtxos: [],
             nonSlpUtxos: [],
             parsedTxHistory: [],
             utxos: [],
         };
     }
 
     return wallet.state;
 };
 
 export function convertEtokenToEcashAddr(eTokenAddress) {
     if (!eTokenAddress) {
         return new Error(
             `cashMethods.convertToEcashAddr() error: No etoken address provided`,
         );
     }
 
     // Confirm input is a valid eToken address
     const isValidInput = isValidEtokenAddress(eTokenAddress);
     if (!isValidInput) {
         return new Error(
             `cashMethods.convertToEcashAddr() error: ${eTokenAddress} is not a valid etoken address`,
         );
     }
 
     // Check for etoken: prefix
     const isPrefixedEtokenAddress = eTokenAddress.slice(0, 7) === 'etoken:';
 
     // If no prefix, assume it is checksummed for an etoken: prefix
     const testedEtokenAddr = isPrefixedEtokenAddress
         ? eTokenAddress
         : `etoken:${eTokenAddress}`;
 
     let ecashAddress;
     try {
         const { type, hash } = cashaddr.decode(testedEtokenAddr);
         ecashAddress = cashaddr.encode('ecash', type, hash);
     } catch (err) {
         return err;
     }
 
     return ecashAddress;
 }
 
 export function convertToEcashPrefix(bitcoincashPrefixedAddress) {
     // Prefix-less addresses may be valid, but the cashaddr.decode function used below
     // will throw an error without a prefix. Hence, must ensure prefix to use that function.
     const hasPrefix = bitcoincashPrefixedAddress.includes(':');
     if (hasPrefix) {
         // Is it bitcoincash: or simpleledger:
         const { type, hash, prefix } = cashaddr.decode(
             bitcoincashPrefixedAddress,
         );
 
         let newPrefix;
         if (prefix === 'bitcoincash') {
             newPrefix = 'ecash';
         } else if (prefix === 'simpleledger') {
             newPrefix = 'etoken';
         } else {
             return bitcoincashPrefixedAddress;
         }
 
         const convertedAddress = cashaddr.encode(newPrefix, type, hash);
 
         return convertedAddress;
     } else {
         return bitcoincashPrefixedAddress;
     }
 }
 
 export function convertEcashtoEtokenAddr(eCashAddress) {
     const isValidInput = isValidXecAddress(eCashAddress);
     if (!isValidInput) {
         return new Error(`${eCashAddress} is not a valid ecash address`);
     }
 
     // Check for ecash: prefix
     const isPrefixedEcashAddress = eCashAddress.slice(0, 6) === 'ecash:';
 
     // If no prefix, assume it is checksummed for an ecash: prefix
     const testedEcashAddr = isPrefixedEcashAddress
         ? eCashAddress
         : `ecash:${eCashAddress}`;
 
     let eTokenAddress;
     try {
         const { type, hash } = cashaddr.decode(testedEcashAddr);
         eTokenAddress = cashaddr.encode('etoken', type, hash);
     } catch (err) {
         return new Error('eCash to eToken address conversion error');
     }
     return eTokenAddress;
 }
 
 // converts ecash, etoken, bitcoincash and simpleledger addresses to hash160
 export function toHash160(addr) {
     try {
         // decode address hash
         const { hash } = cashaddr.decode(addr);
         // encode the address hash to legacy format (bitcoin)
         const legacyAdress = bs58.encode(hash);
         // convert legacy to hash160
         const addrHash160 = Buffer.from(bs58.decode(legacyAdress)).toString(
             'hex',
         );
         return addrHash160;
     } catch (err) {
         console.log('Error converting address to hash160');
         throw err;
     }
 }
 
 /* Converts a serialized buffer containing encrypted data into an object
  * that can be interpreted by the ecies-lite library.
  *
  * For reference on the parsing logic in this function refer to the link below on the segment of
  * ecies-lite's encryption function where the encKey, macKey, iv and cipher are sliced and concatenated
  * https://github.com/tibetty/ecies-lite/blob/8fd97e80b443422269d0223ead55802378521679/index.js#L46-L55
  *
  * A similar PSF implmentation can also be found at:
  * https://github.com/Permissionless-Software-Foundation/bch-encrypt-lib/blob/master/lib/encryption.js
  *
  * For more detailed overview on the ecies encryption scheme, see https://cryptobook.nakov.com/asymmetric-key-ciphers/ecies-public-key-encryption
  */
 export const convertToEncryptStruct = encryptionBuffer => {
     // based on ecies-lite's encryption logic, the encryption buffer is concatenated as follows:
     //  [ epk + iv + ct + mac ]  whereby:
     // - The first 32 or 64 chars of the encryptionBuffer is the epk
     // - Both iv and ct params are 16 chars each, hence their combined substring is 32 chars from the end of the epk string
     //    - within this combined iv/ct substring, the first 16 chars is the iv param, and ct param being the later half
     // - The mac param is appended to the end of the encryption buffer
 
     // validate input buffer
     if (!encryptionBuffer) {
         throw new Error(
             'cashmethods.convertToEncryptStruct() error: input must be a buffer',
         );
     }
 
     try {
         // variable tracking the starting char position for string extraction purposes
         let startOfBuf = 0;
 
         // *** epk param extraction ***
         // The first char of the encryptionBuffer indicates the type of the public key
         // If the first char is 4, then the public key is 64 chars
         // If the first char is 3 or 2, then the public key is 32 chars
         // Otherwise this is not a valid encryption buffer compatible with the ecies-lite library
         let publicKey;
         switch (encryptionBuffer[0]) {
             case 4:
                 publicKey = encryptionBuffer.slice(0, 65); //  extract first 64 chars as public key
                 break;
             case 3:
             case 2:
                 publicKey = encryptionBuffer.slice(0, 33); //  extract first 32 chars as public key
                 break;
             default:
                 throw new Error(`Invalid type: ${encryptionBuffer[0]}`);
         }
 
         // *** iv and ct param extraction ***
         startOfBuf += publicKey.length; // sets the starting char position to the end of the public key (epk) in order to extract subsequent iv and ct substrings
         const encryptionTagLength = 32; // the length of the encryption tag (i.e. mac param) computed from each block of ciphertext, and is used to verify no one has tampered with the encrypted data
         const ivCtSubstring = encryptionBuffer.slice(
             startOfBuf,
             encryptionBuffer.length - encryptionTagLength,
         ); // extract the substring containing both iv and ct params, which is after the public key but before the mac param i.e. the 'encryption tag'
         const ivbufParam = ivCtSubstring.slice(0, 16); // extract the first 16 chars of substring as the iv param
         const ctbufParam = ivCtSubstring.slice(16); // extract the last 16 chars as substring the ct param
 
         // *** mac param extraction ***
         const macParam = encryptionBuffer.slice(
             encryptionBuffer.length - encryptionTagLength,
             encryptionBuffer.length,
         ); // extract the mac param appended to the end of the buffer
 
         return {
             iv: ivbufParam,
             epk: publicKey,
             ct: ctbufParam,
             mac: macParam,
         };
     } catch (err) {
         console.error(`useBCH.convertToEncryptStruct() error: `, err);
         throw err;
     }
 };
 
 export const isLegacyMigrationRequired = wallet => {
     // If the wallet does not have Path1899,
     // Or each Path1899, Path145, Path245 does not have a public key
     // Then it requires migration
     if (
         !wallet.Path1899 ||
         !wallet.Path1899.publicKey ||
         !wallet.Path1899.hash160 ||
         !wallet.Path145.publicKey ||
         !wallet.Path145.hash160 ||
         !wallet.Path245.publicKey ||
         !wallet.Path245.hash160
     ) {
         return true;
     }
 
     return false;
 };
 
 export const getHashArrayFromWallet = wallet => {
     // If the wallet has wallet.Path1899.hash160, it's migrated and will have all of them
     // Return false for an umigrated wallet
     const hash160Array =
         wallet && wallet.Path1899 && 'hash160' in wallet.Path1899
             ? [
                   wallet.Path245.hash160,
                   wallet.Path145.hash160,
                   wallet.Path1899.hash160,
               ]
             : false;
     return hash160Array;
 };
 
 export const isActiveWebsocket = ws => {
     // Return true if websocket is connected and subscribed
     // Otherwise return false
     return (
         ws !== null &&
         ws &&
         '_ws' in ws &&
         'readyState' in ws._ws &&
         ws._ws.readyState === 1 &&
         '_subs' in ws &&
         ws._subs.length > 0
     );
 };
 
 export const hash160ToAddress = hash160 => {
     const buffer = Buffer.from(hash160, 'hex');
 
     // Because ecashaddrjs only accepts Uint8Array as input type, convert
     const hash160ArrayBuffer = new ArrayBuffer(buffer.length);
     const hash160Uint8Array = new Uint8Array(hash160ArrayBuffer);
     for (let i = 0; i < hash160Uint8Array.length; i += 1) {
         hash160Uint8Array[i] = buffer[i];
     }
 
     // Encode ecash: address
     const ecashAddr = cashaddr.encode('ecash', 'P2PKH', hash160Uint8Array);
 
     return ecashAddr;
 };
 
 export const outputScriptToAddress = outputScript => {
     // returns P2SH or P2PKH address
 
     // P2PKH addresses are in outputScript of type 76a914...88ac
     // P2SH addresses are in outputScript of type a914...87
 
     // Return false if cannot determine P2PKH or P2SH address
 
     const typeTestSlice = outputScript.slice(0, 4);
 
     let addressType;
     let hash160;
     switch (typeTestSlice) {
         case '76a9':
             addressType = 'P2PKH';
             hash160 = outputScript.substring(
                 outputScript.indexOf('76a914') + '76a914'.length,
                 outputScript.lastIndexOf('88ac'),
             );
             break;
         case 'a914':
             addressType = 'P2SH';
             hash160 = outputScript.substring(
                 outputScript.indexOf('a914') + 'a914'.length,
                 outputScript.lastIndexOf('87'),
             );
             break;
         default:
             throw new Error('Unrecognized outputScript format');
     }
 
     // Test hash160 for correct length
     if (hash160.length !== 40) {
         throw new Error('Parsed hash160 is incorrect length');
     }
 
     const buffer = Buffer.from(hash160, 'hex');
 
     // Because ecashaddrjs only accepts Uint8Array as input type, convert
     const hash160ArrayBuffer = new ArrayBuffer(buffer.length);
     const hash160Uint8Array = new Uint8Array(hash160ArrayBuffer);
 
     for (let i = 0; i < hash160Uint8Array.length; i += 1) {
         hash160Uint8Array[i] = buffer[i];
     }
 
     // Encode ecash: address
     const ecashAddress = cashaddr.encode(
         'ecash',
         addressType,
         hash160Uint8Array,
     );
 
     return ecashAddress;
 };
 
 export function parseAddressForParams(addressString) {
     // Build return obj
     const addressInfo = {
         address: '',
         queryString: null,
         amount: null,
     };
     // Parse address string for parameters
     const paramCheck = addressString.split('?');
 
     let cleanAddress = paramCheck[0];
     addressInfo.address = cleanAddress;
 
     // Check for parameters
     // only the amount param is currently supported
     let queryString = null;
     let amount = null;
     if (paramCheck.length > 1) {
         queryString = paramCheck[1];
         addressInfo.queryString = queryString;
 
         const addrParams = new URLSearchParams(queryString);
 
         if (addrParams.has('amount')) {
             // Amount in XEC
             try {
                 amount = new BigNumber(
                     parseFloat(addrParams.get('amount')),
                 ).toString();
             } catch (err) {
                 amount = null;
             }
         }
     }
 
     addressInfo.amount = amount;
     return addressInfo;
 }