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 MREGE: Mfa/demo #1398

Draft
wants to merge 86 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 63 commits
Commits
Show all changes
86 commits
Select commit Hold shift + click to select a range
8ca926a
feat: add mfa totp to the email otp flow
blakecduncan Feb 25, 2025
a5038db
chore: fix rn build after mfa changes
blakecduncan Feb 25, 2025
bc887b6
feat: support mfa for magic link
blakecduncan Feb 26, 2025
e572159
feat: update auto generated docs
blakecduncan Feb 26, 2025
dc2c484
chore: fix signer build from missing listener
blakecduncan Feb 26, 2025
c021c79
feat: update api endpoints for mfa
blakecduncan Feb 26, 2025
e4f65fc
chore: update signer docs
blakecduncan Feb 26, 2025
8113753
chore: fix build
blakecduncan Feb 26, 2025
af789b6
chore: update signer docs
blakecduncan Feb 26, 2025
180935e
feat: wip mfa card
RobChangCA Feb 26, 2025
3e06598
chore: update types to match api responses
blakecduncan Feb 27, 2025
40da2d2
chore: auto generated docs pages
blakecduncan Feb 27, 2025
ae47938
chore: use AlchemyMfaStatus type
blakecduncan Feb 27, 2025
901747a
feat: med-fi full flow
RobChangCA Feb 27, 2025
3e89cd6
feat: verify handler for otp
RobChangCA Feb 27, 2025
6cbf5b8
chore: use correct mfa verify api endpoint
blakecduncan Feb 27, 2025
24aeb93
chore: fix verify mfa return value
blakecduncan Feb 27, 2025
fca6b30
chore: auto generated docs pages
blakecduncan Feb 27, 2025
9f73714
chore: fix rn build
blakecduncan Feb 27, 2025
4eea86c
Merge branch 'blake/add-mfa-totp' into mfa/demo
blakecduncan Feb 27, 2025
f477d36
feat: add mfa crud functionality to the UI
blakecduncan Feb 27, 2025
0543c03
feat: dark mode, magic link
RobChangCA Feb 27, 2025
98f3890
feat: add mfa to email otp authentication
blakecduncan Feb 28, 2025
27ce637
Merge branch 'mfa/demo' of https://github.com/alchemyplatform/aa-sdk …
blakecduncan Feb 28, 2025
a5a47f3
chore: fix build
blakecduncan Feb 28, 2025
c33ff93
chore: temporarily using otp email instead of magic link
blakecduncan Feb 28, 2025
07516d2
chore: remove image from qr code
blakecduncan Feb 28, 2025
b8ee838
chore: remove unused import
blakecduncan Feb 28, 2025
0c20735
fix: increase card dimensions
RobChangCA Feb 28, 2025
af94812
fix: min height to cards
RobChangCA Feb 28, 2025
13023b4
chore: fix auth mfa otp endpoint
blakecduncan Feb 28, 2025
cf2a82f
Merge branch 'rob/card-dimensions' into mfa/demo
blakecduncan Feb 28, 2025
e475a71
chore: auto generated docs pages
blakecduncan Feb 28, 2025
46b3bd7
feat: minor ui tweaks
RobChangCA Feb 28, 2025
263e202
feat: check status of mfa in mfa card
blakecduncan Feb 28, 2025
7c43f87
chore: remove unused import
blakecduncan Feb 28, 2025
9a1e543
chore: remove image from qr code
blakecduncan Feb 28, 2025
ef174aa
feat: ui tweaks
RobChangCA Feb 28, 2025
4f04878
fix: image file name
RobChangCA Feb 28, 2025
7e82e4e
feat: add mfa for magic link in the demo
blakecduncan Mar 4, 2025
171ff78
Merge branch 'main' into mfa/demo
RobChangCA Mar 4, 2025
bde11ff
feat: cleanup mfa logic in signer (#1409)
blakecduncan Mar 4, 2025
4b7973d
feat: add mfa hook (#1410)
blakecduncan Mar 4, 2025
7edd6d0
feat: pass multiFactorType to api
blakecduncan Mar 4, 2025
0460be1
chore: update error message
blakecduncan Mar 5, 2025
c88b569
feat: split mfa, clean up animation, error state (#1418)
RobChangCA Mar 6, 2025
e99325c
feat: update the stampedRequest on mfa calls
blakecduncan Mar 11, 2025
75fcbed
feat: update mfa request payload
blakecduncan Mar 11, 2025
6eb9300
Merge branch 'main' into mfa/demo
blakecduncan Mar 13, 2025
e9fdbfd
feat: fix build and wrap cards
blakecduncan Mar 13, 2025
c3ce923
docs: auto generated docs
blakecduncan Mar 13, 2025
250c912
docs: add mfa docs
blakecduncan Mar 13, 2025
2af3651
feat: use mfa hooks in the mfaModal
blakecduncan Mar 14, 2025
ba0e67b
docs: auto generated docs
blakecduncan Mar 14, 2025
bc22a0d
fix: fix lint error
blakecduncan Mar 14, 2025
71775b4
feat: use staging signer
blakecduncan Mar 17, 2025
0f7c6b9
feat: update hook exports
blakecduncan Mar 17, 2025
13bdeaf
feat: fix error states for totp
blakecduncan Mar 18, 2025
efd6edd
feat: fix text overflow on small screens
blakecduncan Mar 18, 2025
9719e29
docs: dogfooding feedback
blakecduncan Mar 18, 2025
4eb58b6
docs: add social login with mfa docs page
blakecduncan Mar 19, 2025
f0f569c
docs: add reference to mfa in the getting started docs
blakecduncan Mar 19, 2025
6980b35
docs: dogfooding feedback
blakecduncan Mar 19, 2025
914b565
feat: clean up useMfa hook and documenation after dogfooding
blakecduncan Mar 20, 2025
e5cb8a3
Merge branch 'main' into mfa/demo
RobChangCA Mar 20, 2025
5ec990b
feat: card alignment - switch to grid
RobChangCA Mar 21, 2025
b9a5603
test: add tests for the authenticate method
blakecduncan Mar 21, 2025
709eec1
test: clean up mocks and base test
blakecduncan Mar 21, 2025
b66f88b
Merge branch 'mfa/demo' of github.com:alchemyplatform/aa-sdk into mfa…
blakecduncan Mar 21, 2025
13bf95e
feat: mobile auth store (#1477)
RobChangCA Mar 24, 2025
b36865a
feat: use new verify mfa api
blakecduncan Mar 25, 2025
47d0d0d
feat: fix build
blakecduncan Mar 26, 2025
f043d8a
feat: use new verify mfa api in the demo
blakecduncan Mar 27, 2025
3325da7
feat: fix the demo app
blakecduncan Mar 27, 2025
4d3fefb
feat: remove unused comment
blakecduncan Mar 27, 2025
4aaa975
test: fix base.test.ts
blakecduncan Mar 28, 2025
cca8114
feat: change modal background to a blur
blakecduncan Apr 1, 2025
126216d
docs: refactor docs after mfa api changes
blakecduncan Apr 1, 2025
6f4cb87
Merge branch 'main' into mfa/demo
blakecduncan Apr 1, 2025
92f70a4
fix: ui merge issues
blakecduncan Apr 1, 2025
30c4e0b
Merge branch 'blake/test-modal-blur' into mfa/demo
blakecduncan Apr 1, 2025
6c20e2b
fix: fix magic link auth card flow
blakecduncan Apr 2, 2025
a10199d
docs: make mfa docs more clear
blakecduncan Apr 2, 2025
fa4f9b5
docs: fix signer mfa docs with updates
blakecduncan Apr 2, 2025
2aa64ac
docs: fix code example build
blakecduncan Apr 2, 2025
9ea28df
fix: fix card ui jumping issue
blakecduncan Apr 2, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 14 additions & 2 deletions account-kit/react/src/components/auth/card/loading/otp.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { useState } from "react";
import { useSigner } from "../../../../hooks/useSigner.js";
import { EmailIllustration } from "../../../../icons/illustrations/email.js";
import { ls } from "../../../../strings.js";
import {
Expand All @@ -20,6 +21,7 @@ export const LoadingOtp = () => {
const [otpCode, setOtpCode] = useState<OTPCodeType>(initialOTPValue);
const [errorText, setErrorText] = useState(authStep.error?.message || "");
const [titleText, setTitleText] = useState(ls.loadingOtp.title);
const signer = useSigner();

const resetOTP = (errorText = "") => {
setOtpCode(initialOTPValue);
Expand All @@ -46,14 +48,24 @@ export const LoadingOtp = () => {
},
});

const setValue = (otpCode: OTPCodeType) => {
const setValue = async (otpCode: OTPCodeType) => {
setOtpCode(otpCode);
if (isOTPCodeType(otpCode)) {
const otp = otpCode.join("");

setAuthStep({ ...authStep, status: AuthStepStatus.verifying });
setTitleText(ls.loadingOtp.verifying);
authenticate({ type: "otp", otpCode: otp });
const { mfaRequired, mfaFactorId } = signer?.getMfaStatus() ?? {};
if (mfaRequired) {
setAuthStep({
type: "totp_verify",
previousStep: "otp",
factorId: mfaFactorId ?? "",
otpCode: otp,
});
} else {
authenticate({ type: "otp", otpCode: otp });
}
}
};

Expand Down
101 changes: 101 additions & 0 deletions account-kit/react/src/components/auth/card/loading/totp.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
import { useState } from "react";
import { useAuthContext } from "../../context.js";
import { useAuthenticate } from "../../../../hooks/useAuthenticate.js";
import {
OTPInput,
initialOTPValue,
isOTPCodeType,
type OTPCodeType,
} from "../../../otp-input/otp-input.js";
import { Spinner } from "../../../../icons/spinner.js";
import { ThreeStarsIcon } from "../../../../icons/threeStars.js";

export const LoadingTotp = () => {
const { authStep, setAuthStep } = useAuthContext("totp_verify");
const [totpCode, setTotpCode] = useState<OTPCodeType>(initialOTPValue);
const [errorText, setErrorText] = useState(authStep.error?.message || "");
const [isSubmitting, setIsSubmitting] = useState(false);
const { authenticateAsync } = useAuthenticate({
onMutate: async (params) => {
if (params.type === "email" && "email" in params) {
if (params.emailMode === "magicLink") {
setAuthStep({ type: "email_verify", email: params.email });
}
}
},
onSuccess: () => {
setIsSubmitting(false);
setAuthStep({ type: "complete" });
},
onError: (err) => {
console.error("TOTP verify error", err);
setIsSubmitting(false);
setErrorText("The code you entered is incorrect");
setTotpCode(initialOTPValue);
},
});

const setValue = async (otpCode: OTPCodeType) => {
setTotpCode(otpCode);
if (isOTPCodeType(otpCode)) {
setIsSubmitting(true);
const otp = otpCode.join("");
handleVerify(otp);
}
};

// Called when all digits are typed or user hits "Verify" button
const handleVerify = async (codeString: string) => {
try {
if (authStep.previousStep === "magicLink") {
await authenticateAsync({
type: "email",
email: authStep.email,
emailMode: "magicLink",
redirectParams: new URLSearchParams(),
multiFactors: [
{
multiFactorId: authStep.factorId,
multiFactorCode: codeString,
},
],
});
} else if (authStep.previousStep === "otp") {
await authenticateAsync({
type: "otp",
otpCode: authStep.otpCode ?? "",
multiFactors: [
{
multiFactorId: authStep.factorId,
multiFactorCode: codeString,
},
],
});
} else {
throw new Error("Invalid previous step");
}
} catch (err) {
console.error("TOTP submission error", err);
}
};

return (
<div className="flex flex-col items-center">
<div className="relative h-12 w-12 mb-5">
<Spinner className="absolute" />
<ThreeStarsIcon className="absolute left-1/2 top-1/2 -translate-x-1/2 -translate-y-1/2" />
</div>
<h3 className="text-fg-primary font-semibold text-lg mb-2">
Enter authentication code
</h3>
<OTPInput
value={totpCode}
setValue={setValue}
errorText={errorText}
setErrorText={setErrorText}
disabled={isSubmitting}
handleReset={() => setTotpCode(initialOTPValue)}
/>
</div>
);
};
3 changes: 3 additions & 0 deletions account-kit/react/src/components/auth/card/steps.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,12 +7,15 @@ import { LoadingPasskeyAuth } from "./loading/passkey.js";
import { MainAuthContent } from "./main.js";
import { PasskeyAdded } from "./passkey-added.js";
import { LoadingOtp } from "./loading/otp.js";
import { LoadingTotp } from "./loading/totp.js";

export const Step = () => {
const { authStep } = useAuthContext();
switch (authStep.type) {
case "email_verify":
return <LoadingEmail />;
case "totp_verify":
return <LoadingTotp />;
case "otp_verify":
return <LoadingOtp />;
case "passkey_verify":
Expand Down
14 changes: 14 additions & 0 deletions account-kit/react/src/components/auth/context.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,20 @@ export type AuthStep =
error?: Error;
status?: AuthStepStatus;
}
| {
type: "totp_verify";
previousStep: "magicLink";
factorId: string;
email: string;
error?: Error;
}
| {
type: "totp_verify";
previousStep: "otp";
factorId: string;
otpCode: string;
error?: Error;
}
| { type: "passkey_verify"; error?: Error }
| { type: "passkey_create"; error?: Error }
| { type: "passkey_create_success" }
Expand Down
30 changes: 25 additions & 5 deletions account-kit/react/src/components/auth/sections/EmailAuth.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { useForm } from "@tanstack/react-form";
import { zodValidator } from "@tanstack/zod-form-adapter";
import { memo } from "react";
import { memo, useEffect } from "react";
import { z } from "zod";
import { useAuthenticate } from "../../../hooks/useAuthenticate.js";
import { useSigner } from "../../../hooks/useSigner.js";
Expand All @@ -12,6 +12,8 @@ import { IS_SIGNUP_QP } from "../../constants.js";
import { Input } from "../../input.js";
import { useAuthContext } from "../context.js";
import type { AuthType } from "../types.js";
import { useSignerStatus } from "../../../hooks/useSignerStatus.js";
import { AlchemySignerStatus, MfaRequiredError } from "@account-kit/signer";

type EmailAuthProps = Extract<AuthType, { type: "email" }>;

Expand All @@ -24,14 +26,13 @@ export const EmailAuth = memo(
buttonLabel = ls.login.email.button,
placeholder = ls.login.email.placeholder,
}: EmailAuthProps) => {
const { status } = useSignerStatus();
const { setAuthStep } = useAuthContext();
const signer = useSigner();
const { authenticateAsync, isPending } = useAuthenticate({
onMutate: async (params) => {
if (params.type === "email" && "email" in params) {
if (params.emailMode === "magicLink") {
setAuthStep({ type: "email_verify", email: params.email });
} else {
if (params.emailMode === "otp") {
setAuthStep({ type: "otp_verify", email: params.email });
}
}
Expand Down Expand Up @@ -64,15 +65,34 @@ export const EmailAuth = memo(
emailMode,
redirectParams,
});
if (emailMode === "magicLink") {
setAuthStep({ type: "email_verify", email });
}
} catch (e) {
const error = e instanceof Error ? e : new Error("An Unknown error");
if (e instanceof MfaRequiredError) {
const { multiFactorId } = e.multiFactors[0];
setAuthStep({
type: "totp_verify",
previousStep: "magicLink",
factorId: multiFactorId,
email,
});
return;
}

const error = e instanceof Error ? e : new Error("An Unknown error");
setAuthStep({ type: "initial", error });
}
},
validatorAdapter: zodValidator(),
});

useEffect(() => {
if (status === AlchemySignerStatus.AWAITING_EMAIL_AUTH) {
setAuthStep({ type: "email_verify", email: form.state.values.email });
}
}, [status]);
Copy link
Collaborator

Choose a reason for hiding this comment

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

⚠️ [eslint] <react-hooks/exhaustive-deps> reported by reviewdog 🐶
React Hook useEffect has missing dependencies: 'form.state.values.email' and 'setAuthStep'. Either include them or remove the dependency array.


return (
<form
className="w-full"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ const RenderFooterText = ({ authStep }: FooterProps) => {
case "oauth_completing":
return <OAuthContactSupport />;
case "email_completing":
case "totp_verify":
case "passkey_create_success":
case "eoa_connect":
case "pick_eoa":
Expand Down
1 change: 1 addition & 0 deletions account-kit/react/src/hooks/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,3 +22,4 @@ export * from "./useSmartAccountClient.js";
export * from "./useUiConfig.js";
export * from "./useUser.js";
export * from "./useWaitForUserOperationTransaction.js";
export * from "./useMFA.js";
Loading