Page MenuHomePhabricator

[Cashtab][Alias] Add alias status check to wallet contacts
ClosedPublic

Authored by emack on Jul 29 2023, 15:50.

Details

Summary

T3216

Following the removal of caching-driven registration status checks in D14318, this diff implements this check using the alias-server /alias endpoint.

This API call is only made if the contact address input is parsed to be a valid alias format.

Test Plan
  • Set alias flag to true in Ticker.js
  • npm start
  • Navigate to Settings and click on Add Contact. Use an invalid alias input for eCash Address and ensure the Invalid eCash address or alias error is displayed
  • Use a valid and registered alias input and ensure it is accepted and saved as a new contact
  • Click on the send icon for the valid and registered alias contact from above and ensure successful XEC send tx to this alias
  • Use a valid but unregistered alias and ensure the Invalid eCash address or alias error is displayed
  • Regression test with a valid eCash address and a valid prefix-less eCash address as input and ensure both passes validation
  • Regression test with an invalid eCash/alias address and ensure the Invalid eCash address or alias error is displayed

Diff Detail

Repository
rABC Bitcoin ABC
Branch
aliasContacts
Lint
Lint Passed
Unit
No Test Coverage
Build Status
Buildable 24650
Build 48896: Build Diffcashtab-tests
Build 48895: arc lint + arc unit

Event Timeline

emack requested review of this revision.Jul 29 2023, 15:50
Fabien requested changes to this revision.Jul 31 2023, 09:04
Fabien added a subscriber: Fabien.
Fabien added inline comments.
cashtab/src/components/Configure/Configure.js
1233–1241 ↗(On Diff #41636)
1252 ↗(On Diff #41636)

Nit: You can make the code flow easier to read by reversing the condition:

if (!isValidAlias) {
    setManualContactAddressIsValid(false);
    return;
}

[...]
This revision now requires changes to proceed.Jul 31 2023, 09:04
emack marked 2 inline comments as done.

Updated flow logic

Fabien requested changes to this revision.Jul 31 2023, 14:50
Fabien added inline comments.
cashtab/src/components/Configure/Configure.js
1225 ↗(On Diff #41643)

just do it unconditionally before the check, afaict it's not changed in the code below

1230 ↗(On Diff #41643)

Isn't it the exact same as const aliasName = value.slice(0, -4); ?

This revision now requires changes to proceed.Jul 31 2023, 14:50
emack marked 2 inline comments as done.

Responding to feedback

cashtab/src/components/Configure/Configure.js
1225 ↗(On Diff #41643)

Updated

1230 ↗(On Diff #41643)

Yup, updated

Fabien requested changes to this revision.Aug 1 2023, 05:35

Code looks good. Now the elephant in the room:

  • There is no test
  • How is that in Configure.js ?
This revision now requires changes to proceed.Aug 1 2023, 05:35
  • How is that in Configure.js ?

Configure should probably be renamed to Settings to align with the frontend. (I can throw a diff up for this)
The contact list resides there because:

  • Logical grouping of saved wallets alongside saved contacts
  • Contacts will eventually be applicable to both Send XEC and Send eToken screens, so it wouldn't work putting it exclusively in either of those send components
  • Doesn't belong in the Home (Tx History) nor Receive (QR Code) screens
  • There is no test

This is similar to SendToken.js where internal functions (i.e. handleManualContactAddressInput in this case) in a React component (classes that render things) can't be accessed directly. You can only test the initial state and props upon initialization but cases where you're looking to simulate a user input (entering a contact address) and then verifying it's reaction to that input is a challenge with Cashtab.

In general, this type of testing can usually be achieved by:

  1. Tools such as the React Testing Library - which facilitates the simulation of user inputs and verification of state variables afterward, which theoretically can be used to test this type of diff. However the issue when it comes to Cashtab is that the testing tool is looking for specific labels to latch onto in order to simulate the input (e.g. label="address-input"), whereas Cashtab uses re-usable subcomponents (e.g. <AddressInput>) which the tool can not detect. Cashtab components are essentially a blank page from this tool's POV. The key lesson here for the next app we build is to design it to be fully unit-testable from the outset.
  2. The other (hacky) way of achieving this is by moving the handleManualContactAddressInput function out, to the same level as the Configure class. This will allow it to be declared as an export function and accessed directly however it will cause issues where handleManualContactAddressInput can't access the state variables internal to the Configure component. In order to resolve that it starts looking like a refactor to make unit tests work rather than what makes sense from a component POV.

Hence why these types of diffs are manually tested for the time being.

  • How is that in Configure.js ?

Configure should probably be renamed to Settings to align with the frontend. (I can throw a diff up for this)
The contact list resides there because:

  • Logical grouping of saved wallets alongside saved contacts
  • Contacts will eventually be applicable to both Send XEC and Send eToken screens, so it wouldn't work putting it exclusively in either of those send components
  • Doesn't belong in the Home (Tx History) nor Receive (QR Code) screens

I totally agree that it doesn't belong in any of these files either. But it's not a configuration, nor a setting.
If it's for consistency with saved contacts, which makes sense, then maybe contacts.js ?

  • There is no test

This is similar to SendToken.js where internal functions (i.e. handleManualContactAddressInput in this case) in a React component (classes that render things) can't be accessed directly. You can only test the initial state and props upon initialization but cases where you're looking to simulate a user input (entering a contact address) and then verifying it's reaction to that input is a challenge with Cashtab.

In general, this type of testing can usually be achieved by:

  1. Tools such as the React Testing Library - which facilitates the simulation of user inputs and verification of state variables afterward, which theoretically can be used to test this type of diff. However the issue when it comes to Cashtab is that the testing tool is looking for specific labels to latch onto in order to simulate the input (e.g. label="address-input"), whereas Cashtab uses re-usable subcomponents (e.g. <AddressInput>) which the tool can not detect. Cashtab components are essentially a blank page from this tool's POV. The key lesson here for the next app we build is to design it to be fully unit-testable from the outset.
  2. The other (hacky) way of achieving this is by moving the handleManualContactAddressInput function out, to the same level as the Configure class. This will allow it to be declared as an export function and accessed directly however it will cause issues where handleManualContactAddressInput can't access the state variables internal to the Configure component. In order to resolve that it starts looking like a refactor to make unit tests work rather than what makes sense from a component POV.

Hence why these types of diffs are manually tested for the time being.

Or you can keep it simple, create a function isValidRecipient(value) -> bool that takes either an address or an alias and call it from the handler like setManualContactAddressIsValid(isValidRecipient(value)). This isValidRecipient() function can be unit tested easily as it only requires a mock for the alias server query. Obviously the new function doesn't belong to Configure.js either.

then maybe contacts.js ?

makes sense, will move it (T3254) in a separate diff as a new component accessed via the hamburger

Or you can keep it simple, create a function isValidRecipient(value) -> bool

damn... that is simpler... will give that a go

Created new isValidRecipient which encapsulates the ecash/alias parsing logic

Fabien requested changes to this revision.Aug 1 2023, 15:40

Too many antipatterns in that code

cashtab/src/utils/validation.js
12–15 ↗(On Diff #41662)

You can drop both the variable and the else

17–20 ↗(On Diff #41662)

Dito

25–27 ↗(On Diff #41662)

Interesting how returning a boolean introduced a lot of antipatterns

33–34 ↗(On Diff #41662)
This revision now requires changes to proceed.Aug 1 2023, 15:40
emack marked 4 inline comments as done.

Updated for all feedback except the boolean return statement

cashtab/src/utils/validation.js
25–27 ↗(On Diff #41662)

This suggestion actually ends up returning the value in aliasDetails.address in the unit tests, hence the original inline conditional statement to enforce a boolean evaluation. The other approach is to use the !! truthy operator in front of aliasDetails.address but that's fairly hard to read.

image.png (141×874 px, 22 KB)

cashtab/src/utils/validation.js
25–27 ↗(On Diff #41662)

stupid js... then either !! like suggested (imo it's fine) or:

return aliasDetails && !aliasDetails.error && aliasDetails.address !== "";

if you feel like it's more readable.

Applying !! boolean operator to enforce boolean evaluation

This revision is now accepted and ready to land.Aug 3 2023, 13:18