Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

[DO NOT MERGE - Iceboxed] Add Iframe and Summary card for the KYC #127

Closed
wants to merge 25 commits into from

Conversation

gianfra-t
Copy link
Contributor

@gianfra-t gianfra-t commented Sep 2, 2024

Available to test UI as offramp is mocked.

Issue #102. Any amount will work and launch the iframe page, which will complete after some seconds mocking the user finishing the process.

Changes

  • Removes the redirect action for "Start Offramping" button that used to open the KYC site on a different page.
  • Remove "Confirm" button and replace with only one click on the "Start Offramping" button to start the process.
  • The KYC is now embedded in an iframe inside the application, which closes and continues to the progress page once the sep24process is completed (response from anchor).

Copy link

netlify bot commented Sep 2, 2024

Deploy Preview for pendulum-pay ready!

Name Link
🔨 Latest commit f78ae4c
🔍 Latest deploy log https://app.netlify.com/sites/pendulum-pay/deploys/66f1ac6b78ae0f0008b53c44
😎 Deploy Preview https://deploy-preview-127--pendulum-pay.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

src/hooks/useMainProcess.ts Outdated Show resolved Hide resolved
@@ -49,6 +52,7 @@ export const ProgressPage: FC<ProgressPageProps> = ({ setOfframpingPhase, offram

const main = (
<main>
<SigningBox step={signingPhase} />
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Signing now happens in the process page, as it would otherwise not be visible given the iframe screen has already popped up.

@gianfra-t gianfra-t changed the title Add Iframe for the KYC [DO NOT MERGE - TESTING UI]. Add Iframe for the KYC Sep 3, 2024
@gianfra-t gianfra-t changed the title [DO NOT MERGE - TESTING UI]. Add Iframe for the KYC [DO NOT MERGE - MISSING CARD DESIGN]. Add Iframe for the KYC Sep 5, 2024
@gianfra-t gianfra-t changed the title [DO NOT MERGE - MISSING CARD DESIGN]. Add Iframe for the KYC Add Iframe and Summary card for the KYC Sep 9, 2024
@gianfra-t
Copy link
Contributor Author

@pendulum-chain/devs and @pendulum-chain/product I think this is ready for review and final adjustments. Once the UI is fine I'll remove the commented part used for mocking.

@ebma ebma linked an issue Sep 10, 2024 that may be closed by this pull request
@ebma
Copy link
Member

ebma commented Sep 10, 2024

I didn't look at the code yet but during testing I ran into the 403 error quite quickly.
image

I entered my credentials in the form of the iframe (name, email, IBAN) and I did that even within a single minute I think. But after clicking the 'Submit' button I encountered the 403.

It doesn't matter how fast or slow I do it, it never works and I always encounter 'Error: 403: Session is not authenticated'.

@gianfra-t
Copy link
Contributor Author

gianfra-t commented Sep 10, 2024

EDIT: Nevermind, I understand what you mean. I didn't test it since I cannot make an offramp anyway, but I can reach the error you also see.

I'm a bit confused @ebma since I can see it always working
But remember that this is mocked so perhaps the wallet is not connected and it is letting you go through anyway, not being able to sign the sep10 challenge?

image.

src/components/Iframe/index.tsx Show resolved Hide resolved
}) => {
const [cachedSrc, setCachedSrc] = useState<string | null>(null);

useEffect(() => {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we can remove the whole cachedSrc logic. We are not using it anywhere

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

True, the idea was to save the kyc url since it was being reset, but since the iframe loads only once, we can remove this.

Copy link
Contributor Author

@gianfra-t gianfra-t Sep 20, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Separated. Also added reusable inner cards.

setCachedSrc(src);
}
}, [src, cachedSrc]);
let main = (
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

const main

<div className="flex justify-center items-center relative w-full h-[50vh] md:w-[535px] ">
<iframe
src={src}
style={{ border: 'none' }}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can't we just use border-0 class?

@@ -0,0 +1,51 @@
import React, { useState, useEffect } from 'react';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we use preact not react, please import preact/compat


return (

<div className="bg-white p-4 rounded-lg shadow-lg w-full">
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is quite big component, wdyt about breaking it down into smaller parts?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You're right. I wanted to get first some feedback on the alignment with the requirements but I agree it needs breaking down for readability.

@@ -36,6 +37,7 @@ export const SwapPage = () => {
const [isQuoteSubmitted, setIsQuoteSubmitted] = useState(false);
const formRef = useRef<HTMLDivElement | null>(null);
const [api, setApi] = useState<ApiPromise | null>(null);
const [isIframVisible, setIframVisible] = useState(false);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it is unused

@gianfra-t
Copy link
Contributor Author

gianfra-t commented Sep 20, 2024

@pendulum-chain/devs now we need to wait for Mykobo to implement the cookies fix in production and we can test this with a full offramp.

UPDATE: this has been unblocked.

Copy link
Member

@ebma ebma left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added minor changes to improve the styling.
And I added code to prevent the content in the summary card to fluctuate. Because I noticed that the quote changed during the time I had the iframe open (over the course of a couple minutes). Let me know what you think.

Besides that, it looks good to me 👍 can confirm the iframe works, and it also recovers properly when the user enters a wrong amount on Mykobo's UI. In that case the toast error is thrown and when the user clicks on the 'Start offramping' button again, a fresh URL is shown in the iframe and the user can continue without the 403 error.

I think this is good to go once the mock changes are reverted.

<span className="text-md md:text-xl font-medium text-blue-800">{amount}</span>
<div className="flex items-center mr-2">
<img src={icon} className="w-4 md:w-6 h-4 md:h-6" />
<span className="ml-1 text-blue-800">{symbol}</span>
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We use everywhere text-blue-700

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

to avoid such confusion I think we should make a class like

vortex-text-blue {
@apply .text-blue-700
}

@vadaynujra
Copy link

vadaynujra commented Sep 23, 2024

I tested twice just now and got the same error, after filling up the first page with the anchor. It might not be related to the iframe implementation but something to do with the anchor or my profile with them?
image

@gianfra-t
Copy link
Contributor Author

gianfra-t commented Sep 23, 2024

@ebma thanks for the improvements. I never thought about the live price change, that would have been very confusing. I will remove the commented guards.

@vadaynujra It looks like it is related. What browser are you using? Do you remember if your wallet connected? Since I haven't experienced this issue since Mykobo changed their back-end.

Copy link
Collaborator

@TorstenStueber TorstenStueber left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tested and it worked well. Great job.

My main remark is about the size of the iframe. I personally don't like that I need to scroll a small box. I didn't test on mobile but can imagine that I would like it even less there.

Another important point is that the SigningBox it not shown anymore when the user signs the Polygon transactions. However, instead of moving the signing box to the IframeComponent, I think it now makes sense to already move to progress screen immediately when SEP 24 is completed and move the SigningBox to the progress screen as well (otherwise I find it slightly confusing UX that after finishing SEP24 I am still shown the iframe for quite some time).

@@ -125,7 +125,7 @@ export interface ExecutionContext {
setSigningPhase: (n: SigningPhase) => void;
}

const OFFRAMPING_STATE_LOCAL_STORAGE_KEY = 'offrampingState';
export const OFFRAMPING_STATE_LOCAL_STORAGE_KEY = 'offrampingState';
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why is this exported now? It is not used outside this file.

if (config.test.mockSep24) {
return { url: 'https://www.example.com', id: '1234' };
}
// if (config.test.mockSep24) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please remove, not comment out.

@@ -227,11 +227,13 @@ export async function sep24Second(
const { sep24Url } = tomlValues;

if (config.test.mockSep24) {
// sleep 15 seconds, mock user completion of KYC
await new Promise((resolve) => setTimeout(resolve, 150000));
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is actually 150 seconds, not 15 seconds.

return {
amount: sessionParams.offrampAmount,
memo: 'MYK1722323689',
memoType: 'text',
offrampingAccount: (await fetchSigningServiceAccountId()).stellar.public,
offrampingAccount: 'GSAPDFASJFPASFOKNASOFKNAS',
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why this change? The previous setting is better in my opinion as it sends the funds back to the funding account I use locally.

pending={isInitiating || offrampingStarted || offrampingState !== undefined}
/>
)}
<SwapSubmitButton
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is a breaking change here. The button text has been changed in the mean time, can you please change accordingly?

(also the logic for the pending state)

@@ -25,6 +25,7 @@ import { SuccessPage } from '../success';
import { FailurePage } from '../failure';
import { useInputTokenBalance } from '../../hooks/useInputTokenBalance';
import { UserBalance } from '../../components/UserBalance';
import { IframeComponent, IframeProps } from '../../components/Iframe';
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The IframeProps import not used.

<h2 className="text-md mb-4 mt-4">{subtitle}</h2>
</div>
</div>
<div className="flex justify-center items-center relative w-full h-[50vh] md:w-[535px] ">
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I remember that we discussed once that we want to reduce double scroll and the that the design solution was to prevent scrolling of the outer screen by making the iframe always fit into the screen (i.e., scrolling only happens within the iframe).

We then discussed the other option: make the iframe large enough that it will not scroll and only the outer screen is scrolling.

I actually remember that we decided to use that latter option, am I wrong (I don't find notes about this).

Same for the width of the iframe: it feels right now like looking through a peephole and I don't think that this is such a good UX.


const receiveAmount = calculateTotalReceive(toAmount.toString(), OUTPUT_TOKEN_CONFIG[assetOut]);

const approximateExchangeRate = roundDownToSignificantDecimals(toAmount.div(fromAmount), 2).toString();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wouldn't it be better to also use 4 decimals as on the details box on the main page? E.g., reuse the ExchangeRate component.


const receiveAmount = calculateTotalReceive(toAmount.toString(), OUTPUT_TOKEN_CONFIG[assetOut]);

const approximateExchangeRate = roundDownToSignificantDecimals(toAmount.div(fromAmount), 2).toString();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please use toFixed(2, 0) instead of toString() for every number stringification of amounts in this component. (see #171)

<LowerSummaryCard
label="Offramp fees"
amount={offrampFees}
startIcon={<LocalGasStationIcon className="text-blue-700" fontSize="small" />}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I argue that we should remove the gas station icon: #102 (comment)

@vadaynujra
Copy link

@gianfra-t I used Google Chrome

@vadaynujra
Copy link

Tried again and got the same error. Attached the console logs. I also tried the flow using production to ensure there wasn't any issue with my details and didn't get the 403 error there.
image

@gianfra-t
Copy link
Contributor Author

gianfra-t commented Sep 24, 2024

@TorstenStueber I will address your comments but regarding this one:

instead of moving the signing box to the IframeComponent, I think it now makes sense to already move to progress screen immediately when SEP 24 is completed and move the SigningBox to the progress screen as well

I agree and in fact the solution is quite quick, we just need to delete this condition, the includes one.

That said, what was the rationale behind signing on the main screen (even before the iframe)? Because if it is such that the user sees the values when it is signing, then we should stay here (and potentially remove the Iframe and show a message).

@TorstenStueber
Copy link
Collaborator

Yes, good question. I don't know what the reasoning was. @prayagd @vadaynujra can you give the reasoning?

(I guess it was because of the numbers, as you said. Would it not make sense to show your SummaryCard then also on the progress screen? It is nice to see how much is currently offramping).

@TorstenStueber
Copy link
Collaborator

What shall we do with this PR? Close it?

@gianfra-t gianfra-t changed the title Add Iframe and Summary card for the KYC [DO NOT MERGE - Iceboxed] Add Iframe and Summary card for the KYC Sep 26, 2024
@gianfra-t
Copy link
Contributor Author

gianfra-t commented Sep 26, 2024

What shall we do with this PR? Close it?

I changed the title but we can also close it, we can always reopen it.

@TorstenStueber
Copy link
Collaborator

I would also suggest to just close it instead of keeping it open with the warning in the title.

@ebma
Copy link
Member

ebma commented Sep 26, 2024

@gianfra-t before closing it, can you leave a comment with a brief summary that explains why we cannot proceed with this change? Namely pointing out the restrictions with the browser settings that interferes with the cookies in the iframe etc.

@TorstenStueber
Copy link
Collaborator

@gianfra-t let's close this PR? We can still reopen later if we want to go back to the iframe.

@gianfra-t
Copy link
Contributor Author

Closing this. We could later use the UI perhaps.

Just to recap, the reason this didn't work is because the anchor (Mykobo) is using cookies for CSRF protection, and these are being blocked when using the iframe solution (the URL of the iframe is different to Vortex). Specifically, this happens in browsers with stricter policies, which would make the solution not viable for many users.

@gianfra-t gianfra-t closed this Nov 4, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Implement the SEP24 flow in an iframe
5 participants