Page MenuHomePhabricator

[Cashtab] Support custom eToken icons
ClosedPublic

Authored by bytesofman on Tue, Nov 23, 20:30.

Details

Reviewers
emack
Group Reviewers
Restricted Project
Commits
rABCba44769b7430: [Cashtab] Support custom eToken icons
Summary

T1743

I've coded and deployed a simple back-end engine that will accept user submission of token icons. The icons are then manually reviewed and approved. Approval is by clicking on a button in a Telegram channel, so can be quite quick without too much extra work.

This diff implements user tools for creating and uploading a customized token icon. For now, only creating the icon on token creation is supported. In the future I will add a way for users to upload an icon for a previously created token.

While the diff seems like a complicated change, most of the code comes from previous production mint.bitcoin.com code, available from https://github.com/bitcoin-com/mint

Test Plan

npm start

  1. Navigate to eTokens page and fill out the form to create a new eToken
  2. Upload your icon. Test the customization settings (zoom, rotate, square vs circle crop)
  3. Create the token. Note the notifications -- you should see one for icon submission success.
  4. Ping me on Telegram to verify icon is approved (@bytesofman)
  5. Verify you can see the icon displayed on the eTokens page

Diff Detail

Repository
rABC Bitcoin ABC
Lint
Lint Not Applicable
Unit
Tests Not Applicable

Event Timeline

Failed tests logs:

====== CashTab Unit Tests:  Wallet with BCH balances and tokens and state field ======
Error: expect(received).toMatchSnapshot()

Snapshot name: `Wallet with BCH balances and tokens and state field 1`

- Snapshot  - 13
+ Received  +  1

@@ -163,23 +163,11 @@
                          onMouseUp={[Function]}
                          style={null}
                        >
                          <span
                            className="ant-input-prefix"
-                         >
-                           <img
-                             alt="identicon of tokenId bd1acc4c986de57af8d6d2a64aecad8c30ee80f37ae9d066d758923732ddc9ba "
-                             heigh="16"
-                             src=""
-                             style={
-                               Object {
-                                 "borderRadius": "50%",
-                               }
-                             }
-                             width="16"
-                           />
-                         </span>
+                         />
                          <input
                            className="ant-input"
                            name="value"
                            onBlur={[Function]}
                            onChange={[Function]}
    at Object.<anonymous> (/work/web/cashtab/src/components/Send/__tests__/SendToken.test.js:76:18)
    at Object.asyncJestTest (/work/web/cashtab/node_modules/jest-jasmine2/build/jasmineAsyncInstall.js:106:37)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:45:12
    at new Promise (<anonymous>)
    at mapper (/work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:28:19)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:75:41
    at processTicksAndRejections (node:internal/process/task_queues:94:5)Error: expect(received).toMatchSnapshot()

Snapshot name: `Wallet with BCH balances and tokens and state field 1`

- Snapshot  - 18
+ Received  +  6

@@ -1,15 +1,15 @@
  Array [
    <div
-     className="sc-kAzzGY kdOdTZ"
+     className="sc-kgoBCf kfjVlq"
    >
      0.06
       
      XEC
    </div>,
    <div
-     className="sc-chPdSV elAazB"
+     className="sc-kGXeez kiHoFA"
    >
      $ 
      NaN
       
      USD
@@ -61,29 +61,17 @@
      <a
        href="/send-token/bd1acc4c986de57af8d6d2a64aecad8c30ee80f37ae9d066d758923732ddc9ba"
        onClick={[Function]}
      >
        <div
-         className="sc-eNQAEJ jnzudA"
+         className="sc-kEYyzF RTiPg"
        >
          <div
-           className="sc-ckVGcZ ealBRQ"
-         >
-           <img
-             alt="identicon of tokenId bd1acc4c986de57af8d6d2a64aecad8c30ee80f37ae9d066d758923732ddc9ba "
-             height="32"
-             src=""
-             style={
-               Object {
-                 "borderRadius": "50%",
-               }
-             }
-             width="32"
-           />
-         </div>
+           className="sc-eNQAEJ iEWhkj"
+         />
          <div
-           className="sc-jKJlTe bUCZgB"
+           className="sc-hMqMXs jcgLFh"
          >
            6.001
             
            <strong>
              TBS
    at Object.<anonymous> (/work/web/cashtab/src/components/Tokens/__tests__/Tokens.test.js:96:18)
    at Object.asyncJestTest (/work/web/cashtab/node_modules/jest-jasmine2/build/jasmineAsyncInstall.js:106:37)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:45:12
    at new Promise (<anonymous>)
    at mapper (/work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:28:19)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:75:41
    at processTicksAndRejections (node:internal/process/task_queues:94:5)
====== CashTab Unit Tests:  Wallet without BCH balance ======
Error: expect(received).toMatchSnapshot()

Snapshot name: `Wallet without BCH balance 1`

- Snapshot  - 3
+ Received  + 3

@@ -1,15 +1,15 @@
  Array [
    <div
-     className="sc-kgoBCf fswuXg"
+     className="sc-kpOJdX GoEyB"
    >
      You need some 
      XEC
       in your wallet to create tokens.
    </div>,
    <div
-     className="sc-kAzzGY kdOdTZ"
+     className="sc-kgoBCf kfjVlq"
    >
      0
       
      XEC
    </div>,
@@ -55,11 +55,11 @@
          Create eToken
        </div>
      </div>
    </div>,
    <p
-     className="sc-kpOJdX bjUfnF"
+     className="sc-ckVGcZ eEJdrC"
    >
      You need at least
       
      5.5
       
    at Object.<anonymous> (/work/web/cashtab/src/components/Tokens/__tests__/Tokens.test.js:54:18)
    at Object.asyncJestTest (/work/web/cashtab/node_modules/jest-jasmine2/build/jasmineAsyncInstall.js:106:37)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:45:12
    at new Promise (<anonymous>)
    at mapper (/work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:28:19)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:75:41
    at processTicksAndRejections (node:internal/process/task_queues:94:5)
====== CashTab Unit Tests:  Wallet with BCH balances ======
Error: expect(received).toMatchSnapshot()

Snapshot name: `Wallet with BCH balances 1`

- Snapshot  - 3
+ Received  + 3

@@ -1,15 +1,15 @@
  Array [
    <div
-     className="sc-kgoBCf fswuXg"
+     className="sc-kpOJdX GoEyB"
    >
      You need some 
      XEC
       in your wallet to create tokens.
    </div>,
    <div
-     className="sc-kAzzGY kdOdTZ"
+     className="sc-kgoBCf kfjVlq"
    >
      0
       
      XEC
    </div>,
@@ -55,11 +55,11 @@
          Create eToken
        </div>
      </div>
    </div>,
    <p
-     className="sc-kpOJdX bjUfnF"
+     className="sc-ckVGcZ eEJdrC"
    >
      You need at least
       
      5.5
       
    at Object.<anonymous> (/work/web/cashtab/src/components/Tokens/__tests__/Tokens.test.js:68:18)
    at Object.asyncJestTest (/work/web/cashtab/node_modules/jest-jasmine2/build/jasmineAsyncInstall.js:106:37)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:45:12
    at new Promise (<anonymous>)
    at mapper (/work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:28:19)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:75:41
    at processTicksAndRejections (node:internal/process/task_queues:94:5)
====== CashTab Unit Tests:  Wallet with BCH balances and tokens ======
Error: expect(received).toMatchSnapshot()

Snapshot name: `Wallet with BCH balances and tokens 1`

- Snapshot  - 3
+ Received  + 3

@@ -1,15 +1,15 @@
  Array [
    <div
-     className="sc-kgoBCf fswuXg"
+     className="sc-kpOJdX GoEyB"
    >
      You need some 
      XEC
       in your wallet to create tokens.
    </div>,
    <div
-     className="sc-kAzzGY kdOdTZ"
+     className="sc-kgoBCf kfjVlq"
    >
      0
       
      XEC
    </div>,
@@ -55,11 +55,11 @@
          Create eToken
        </div>
      </div>
    </div>,
    <p
-     className="sc-kpOJdX bjUfnF"
+     className="sc-ckVGcZ eEJdrC"
    >
      You need at least
       
      5.5
       
    at Object.<anonymous> (/work/web/cashtab/src/components/Tokens/__tests__/Tokens.test.js:82:18)
    at Object.asyncJestTest (/work/web/cashtab/node_modules/jest-jasmine2/build/jasmineAsyncInstall.js:106:37)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:45:12
    at new Promise (<anonymous>)
    at mapper (/work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:28:19)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:75:41
    at processTicksAndRejections (node:internal/process/task_queues:94:5)
====== CashTab Unit Tests:  Without wallet defined ======
Error: expect(received).toMatchSnapshot()

Snapshot name: `Without wallet defined 1`

- Snapshot  - 3
+ Received  + 3

@@ -1,15 +1,15 @@
  Array [
    <div
-     className="sc-kgoBCf fswuXg"
+     className="sc-kpOJdX GoEyB"
    >
      You need some 
      XEC
       in your wallet to create tokens.
    </div>,
    <div
-     className="sc-kAzzGY kdOdTZ"
+     className="sc-kgoBCf kfjVlq"
    >
      0
       
      XEC
    </div>,
@@ -55,11 +55,11 @@
          Create eToken
        </div>
      </div>
    </div>,
    <p
-     className="sc-kpOJdX bjUfnF"
+     className="sc-ckVGcZ eEJdrC"
    >
      You need at least
       
      5.5
       
    at Object.<anonymous> (/work/web/cashtab/src/components/Tokens/__tests__/Tokens.test.js:114:18)
    at Object.asyncJestTest (/work/web/cashtab/node_modules/jest-jasmine2/build/jasmineAsyncInstall.js:106:37)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:45:12
    at new Promise (<anonymous>)
    at mapper (/work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:28:19)
    at /work/web/cashtab/node_modules/jest-jasmine2/build/queueRunner.js:75:41
    at processTicksAndRejections (node:internal/process/task_queues:94:5)

Each failure log is accessible here:
CashTab Unit Tests: Wallet with BCH balances and tokens and state field
CashTab Unit Tests: Wallet without BCH balance
CashTab Unit Tests: Wallet with BCH balances
CashTab Unit Tests: Wallet with BCH balances and tokens
CashTab Unit Tests: Without wallet defined

emack requested changes to this revision.Tue, Nov 23, 23:58

A few areas of observation:

  1. On initial load of wallet, getting a bunch of 403 errors in the browser console, without doing anything but doesn't seem to be impacting functions. Might need to handle those backend calls.

image.png (303×319 px, 56 KB)

  1. It doesn't like svg images, suggest explicitly limiting upload file types to png and jpgs, and adding this instruction to the UI.

image.png (311×428 px, 63 KB)

  1. Expand test plan to include attempts to upload videos, gifs, and images of various dimensions/sizes and round/square modes. I tested an ultra high quality native 4k wallpaper (2MB) and it resized/cropped it all ok on Round mode and console.log output looks normal.
  1. However, uploading that same 4k wallpaper image in Square mode consistently encountered backend errors. Suggest upgrading the backend error handling messages to make our future debugging experience easier

image.png (397×297 px, 55 KB)

  1. Tested with an 8k Ultra HD image (7680x4320), 9.3MB, round mode, icon submitted, approved and displayed all ok, which is weird considering you've adjusted the backend to only handle 2.5MB images or less. Please check this backend logic as well as adding frontend validation that stops the user submitting any files larger than 2.5MB.
  1. The 'you can only upload image files' notification error persisted on screen without going away. The error notification for beforeTokenIconUpload() is set to 0, not sure if that's the issue.

image.png (107×409 px, 8 KB)

  1. When clicking on the newly created eToken and the user is taken to SendToken.js, the token details table should include an additional row that displays the icon like the mockimage below

image.png (350×529 px, 26 KB)

  1. The document URI is incorrectly rendered as part of a tx url on be.cash explorer, however I think this might be an existing issue.

image.png (125×493 px, 13 KB)

  1. Remainder to adjust backend to approve icons by default and we can deny by exception.
This revision now requires changes to proceed.Tue, Nov 23, 23:58

A few areas of observation:

  1. On initial load of wallet, getting a bunch of 403 errors in the browser console, without doing anything but doesn't seem to be impacting functions. Might need to handle those backend calls.

image.png (303×319 px, 56 KB)

I've created a task to handle this separate from this diff (T1981)

  1. It doesn't like svg images, suggest explicitly limiting upload file types to png and jpgs, and adding this instruction to the UI.

image.png (311×428 px, 63 KB)

Refactored so that only png and jpg files are accepted

  1. Expand test plan to include attempts to upload videos, gifs, and images of various dimensions/sizes and round/square modes. I tested an ultra high quality native 4k wallpaper (2MB) and it resized/cropped it all ok on Round mode and console.log output looks normal.

Testing with a non-jpg/png filetype to confirm it does not upload to the server is sufficient. Annoyingly, file is still displayed under the Dragger, but the relevant state fields are not holding any data.

  1. However, uploading that same 4k wallpaper image in Square mode consistently encountered backend errors. Suggest upgrading the backend error handling messages to make our future debugging experience easier

image.png (397×297 px, 55 KB)

See the changes I made to the showCroppedImage function. needs to be tested. Looks like the previous version was only resizing the image if it was rounded. This would explain the server errors. Rounded images were resized to 128x128, but square images were not...so square images could still be above backend size restrictions on upload, while round images would not be.

I also bumped the resize to 512x512 so we can support higher resolutions in the future. Created a task for this -- current way it works of only using the 32x32 image is not great (T1987)

  1. Tested with an 8k Ultra HD image (7680x4320), 9.3MB, round mode, icon submitted, approved and displayed all ok, which is weird considering you've adjusted the backend to only handle 2.5MB images or less. Please check this backend logic as well as adding frontend validation that stops the user submitting any files larger than 2.5MB.

Should be handled by the forced resize to 512x512, now applying to both round and square images

  1. The 'you can only upload image files' notification error persisted on screen without going away. The error notification for beforeTokenIconUpload() is set to 0, not sure if that's the issue.

image.png (107×409 px, 8 KB)

I replaced this with a modal

  1. When clicking on the newly created eToken and the user is taken to SendToken.js, the token details table should include an additional row that displays the icon like the mockimage below

image.png (350×529 px, 26 KB)

Yes. Created a separate task for this, does not need to be in this diff. T1988

  1. The document URI is incorrectly rendered as part of a tx url on be.cash explorer, however I think this might be an existing issue.

image.png (125×493 px, 13 KB)

Not sure exactly what you mean here -- could you create a new task for this?

  1. Remainder to adjust backend to approve icons by default and we can deny by exception.

Good idea, will do.

emack requested changes to this revision.Thu, Nov 25, 01:27

Resolved issues:

  • Icon square mode uploads successfully to the server
  • The '...upload image files error' notification duration now fixed
  • Non-jpg/png files now triggering validation error as expected
  • Tested in firefox/chrome
  • Will raise a separate task for the document URI issue

New issues:

  • When testing on a physical mobile device the Icon Upload Error dialog is given for both jpg and png files. Steps to reproduce issue:
    1. npm start
    2. access the cashtab instance via chrome on a mobile device
    3. attempt to upload a jpg or png image from the mobile device

The jpg image used is below, 98.8 KB in size and 549 x 766 pixels in resolution, 96 dpi.

20211109_165125 - Copy.jpg (766×549 px, 98 KB)

Screenshot_20211125-115557_Chrome - Copy.jpg (2×1 px, 135 KB)

  • When testing in extension mode, once the user selects the image to upload, the plugin disappears, which is normal since you're clicking outside of the plugin. However this resets the token creation process which makes it impossible for plugin users to create icons. Suggest addition persistence logic specific to the token creation screen is needed here in extension mode.
This revision now requires changes to proceed.Thu, Nov 25, 01:27

Fixing issue of token icon creation not working on mobile

Issue with mobile upload not working appears related to the manually-implemented file types. When I tested on mobile device, dev console was showing image/jpeg instead of image/jpg -- adding this filetype fixed the issue for me.

Re: chrome extension issue. Valid issue though I think can still be a workaround if user opens the extension in its own tab. I created a task to handle this issue separately T1999. Want to get this deployed soon so that I can start iterating on the front-end; most users are on the web wallet.

Resolved issues:

  • Icon square mode uploads successfully to the server
  • The '...upload image files error' notification duration now fixed
  • Non-jpg/png files now triggering validation error as expected
  • Tested in firefox/chrome
  • Will raise a separate task for the document URI issue

New issues:

  • When testing on a physical mobile device the Icon Upload Error dialog is given for both jpg and png files. Steps to reproduce issue:
    1. npm start
    2. access the cashtab instance via chrome on a mobile device
    3. attempt to upload a jpg or png image from the mobile device

The jpg image used is below, 98.8 KB in size and 549 x 766 pixels in resolution, 96 dpi.

20211109_165125 - Copy.jpg (766×549 px, 98 KB)

Screenshot_20211125-115557_Chrome - Copy.jpg (2×1 px, 135 KB)

  • When testing in extension mode, once the user selects the image to upload, the plugin disappears, which is normal since you're clicking outside of the plugin. However this resets the token creation process which makes it impossible for plugin users to create icons. Suggest addition persistence logic specific to the token creation screen is needed here in extension mode.
This revision is now accepted and ready to land.Thu, Nov 25, 22:05