refactor: standardize test IDs in E2E testing
This commit is contained in:
+20
-39
@@ -27,16 +27,15 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
|
|
||||||
logger.info(`>> [Draft] Current URL after click: ${page.url()}`);
|
logger.info(`>> [Draft] Current URL after click: ${page.url()}`);
|
||||||
|
|
||||||
// Wait for the recipient input to be present in the DOM
|
// Editor page
|
||||||
|
await expect(page.getByTestId("recipient-input")).toBeVisible();
|
||||||
const recipientInput = page.getByTestId("recipient-input");
|
const recipientInput = page.getByTestId("recipient-input");
|
||||||
await recipientInput.waitFor({ state: "visible", timeout: 20000 });
|
|
||||||
|
|
||||||
const recipientName = "Dear Friend";
|
const recipientName = "Dear Friend";
|
||||||
await recipientInput.fill(recipientName);
|
await recipientInput.fill(recipientName);
|
||||||
|
|
||||||
// Initial load: verify textarea value (populated by Fabric when focused)
|
// Initial load: verify textarea value (populated by Fabric when focused)
|
||||||
const canvasInput = page.locator("textarea");
|
const canvasInput = page.locator("textarea");
|
||||||
await canvasInput.waitFor({ state: "attached" });
|
|
||||||
await canvasInput.focus();
|
await canvasInput.focus();
|
||||||
await expect(canvasInput).toHaveValue(/Take a deep breath/i);
|
await expect(canvasInput).toHaveValue(/Take a deep breath/i);
|
||||||
|
|
||||||
@@ -50,7 +49,7 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
await page.getByTestId("draft-btn").click();
|
await page.getByTestId("draft-btn").click();
|
||||||
|
|
||||||
// Verify Success Modal/Alert
|
// Verify Success Modal/Alert
|
||||||
await expect(page.getByText(/your letter is saved/i)).toBeVisible();
|
await expect(page.getByTestId("save-success-toast")).toBeVisible();
|
||||||
|
|
||||||
// Verify URL updated with a UUID
|
// Verify URL updated with a UUID
|
||||||
await expect(page).toHaveURL(/\/quill\/[0-9a-f-]{36}/);
|
await expect(page).toHaveURL(/\/quill\/[0-9a-f-]{36}/);
|
||||||
@@ -62,13 +61,7 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
await page.goto(savedUrl);
|
await page.goto(savedUrl);
|
||||||
|
|
||||||
// Wait for initial load overlay to appear and then definitely disappear
|
// Wait for initial load overlay to appear and then definitely disappear
|
||||||
await page
|
await expect(page.getByTestId("opening-draft-overlay")).toBeHidden();
|
||||||
.getByText(/opening your draft/i)
|
|
||||||
.waitFor({ state: "visible", timeout: 2000 })
|
|
||||||
.catch(() => {});
|
|
||||||
await expect(page.getByText(/opening your draft/i)).toBeHidden({
|
|
||||||
timeout: 10000,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Check recipient
|
// Check recipient
|
||||||
await expect(page.getByTestId("recipient-input")).toHaveValue(recipientName);
|
await expect(page.getByTestId("recipient-input")).toHaveValue(recipientName);
|
||||||
@@ -77,9 +70,7 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
// We wait for the content to appear in the textarea.
|
// We wait for the content to appear in the textarea.
|
||||||
// toHaveValue will poll until it matches or timeouts.
|
// toHaveValue will poll until it matches or timeouts.
|
||||||
await canvasInput.focus();
|
await canvasInput.focus();
|
||||||
await expect(canvasInput).toHaveValue(/This is a secret draft/i, {
|
await expect(canvasInput).toHaveValue(/This is a secret draft/i);
|
||||||
timeout: 10000,
|
|
||||||
});
|
|
||||||
await expect(canvasInput).toHaveValue(/It should persist/i);
|
await expect(canvasInput).toHaveValue(/It should persist/i);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -96,7 +87,6 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
await page.getByTestId("write-letter-btn").click();
|
await page.getByTestId("write-letter-btn").click();
|
||||||
|
|
||||||
const recipientInput = page.getByTestId("recipient-input");
|
const recipientInput = page.getByTestId("recipient-input");
|
||||||
await recipientInput.waitFor({ state: "visible", timeout: 10000 });
|
|
||||||
await recipientInput.fill("A Secret Guest");
|
await recipientInput.fill("A Secret Guest");
|
||||||
|
|
||||||
const canvasInput = page.locator("textarea");
|
const canvasInput = page.locator("textarea");
|
||||||
@@ -110,40 +100,36 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
|
|
||||||
// Should show sealed confirmation modal
|
// Should show sealed confirmation modal
|
||||||
logger.info(">> [Seal] Verifying sealed modal...");
|
logger.info(">> [Seal] Verifying sealed modal...");
|
||||||
await expect(page.getByText(/your letter is sealed/i)).toBeVisible({
|
await expect(page.getByTestId("post-seal-modal")).toBeVisible();
|
||||||
timeout: 10000,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Navigate to Reader via "View letter"
|
// Navigate to Reader via "View letter"
|
||||||
await page.getByTestId("view-letter-btn").click();
|
await page.getByTestId("view-letter-btn").click();
|
||||||
|
|
||||||
// Should be on Reader URL
|
// Should be on Reader URL
|
||||||
await expect(page).toHaveURL(/\/read\/[a-f0-9-]{36}$/, { timeout: 15000 });
|
await expect(page).toHaveURL(/\/read\/[a-f0-9-]{36}$/);
|
||||||
|
|
||||||
// Open the envelope to reveal the letter
|
// Open the envelope to reveal the letter
|
||||||
await expect(page.getByText(/breaking the seal/i)).toBeHidden({
|
await expect(page.getByTestId("decryption-overlay")).toBeHidden();
|
||||||
timeout: 10000,
|
|
||||||
});
|
|
||||||
// Flip the envelope to show the seal and reveal the letter
|
// Flip the envelope to show the seal and reveal the letter
|
||||||
await revealEnvelope(page);
|
await revealEnvelope(page);
|
||||||
await expect(page.getByTestId("envelope-letter")).toBeHidden({ timeout: 20000 });
|
await expect(page.getByTestId("envelope-letter")).toBeHidden();
|
||||||
|
|
||||||
// Share on demand
|
// Share on demand
|
||||||
logger.info(">> [Seal] Clicking Share button in Reader...");
|
logger.info(">> [Seal] Clicking Share button in Reader...");
|
||||||
await page.getByTestId("share-letter-btn").click();
|
await page.getByTestId("share-letter-btn").click();
|
||||||
|
|
||||||
// Verify share modal with a valid link
|
// Verify share modal with a valid link
|
||||||
await expect(page.getByText(/send this letter/i)).toBeVisible();
|
await expect(page.getByTestId("share-letter-modal")).toBeVisible();
|
||||||
const linkInput = page.locator("#share-link-input");
|
const linkInput = page.locator("#share-link-input");
|
||||||
const linkValue = await linkInput.inputValue();
|
const linkValue = await linkInput.inputValue();
|
||||||
expect(linkValue).toContain("/read/");
|
expect(linkValue).toContain("/read/");
|
||||||
expect(linkValue).toContain("#");
|
expect(linkValue).toContain("#");
|
||||||
logger.info(`>> [Seal] Sharing link: ${linkValue}`);
|
logger.info(`>> [Seal] Sharing link: ${linkValue}`);
|
||||||
|
|
||||||
await expect(page.getByRole("button", { name: /copy/i })).toBeVisible();
|
await expect(page.getByTestId("copy-link-btn")).toBeVisible();
|
||||||
// Assuming Close button in ShareModal might need a testid too, but for now let's use text if unique or add testid
|
// Assuming Close button in ShareModal might need a testid too, but for now let's use text if unique or add testid
|
||||||
await page.getByRole("button", { name: /close/i }).click();
|
await page.getByTestId("modal-close-btn").click();
|
||||||
await expect(page.getByText(/send this letter/i)).toBeHidden();
|
await expect(page.getByTestId("share-letter-modal")).toBeHidden();
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should allow author to access sealed letter from drawer without sharing key", async ({
|
test("should allow author to access sealed letter from drawer without sharing key", async ({
|
||||||
@@ -161,7 +147,6 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
await page.getByTestId("write-letter-btn").click();
|
await page.getByTestId("write-letter-btn").click();
|
||||||
|
|
||||||
const recipientInput = page.getByTestId("recipient-input");
|
const recipientInput = page.getByTestId("recipient-input");
|
||||||
await recipientInput.waitFor({ state: "visible" });
|
|
||||||
await recipientInput.fill(recipientName);
|
await recipientInput.fill(recipientName);
|
||||||
|
|
||||||
const canvasInput = page.locator("textarea");
|
const canvasInput = page.locator("textarea");
|
||||||
@@ -173,34 +158,30 @@ test.describe("Letter Drafting (Real Backend)", () => {
|
|||||||
await page.getByTestId("seal-confirm-btn").click();
|
await page.getByTestId("seal-confirm-btn").click();
|
||||||
|
|
||||||
// Sealed modal should appear — click "Keep it" to go to Drawer
|
// Sealed modal should appear — click "Keep it" to go to Drawer
|
||||||
await expect(page.getByText(/your letter is sealed/i)).toBeVisible({
|
await expect(page.getByTestId("post-seal-modal")).toBeVisible();
|
||||||
timeout: 10000,
|
|
||||||
});
|
|
||||||
await page.getByTestId("keep-it-btn").click();
|
await page.getByTestId("keep-it-btn").click();
|
||||||
|
|
||||||
// Open "Kept" section - search for the section with id='kept' and click its toggle button
|
// Open "Kept" section - search for the section with id='kept' and click its toggle button
|
||||||
logger.info(">> [Drawer] Opening Kept section...");
|
logger.info(">> [Drawer] Opening Kept section...");
|
||||||
const keptSection = page.locator("#kept");
|
await page.getByTestId("drawer-section-kept").click();
|
||||||
await keptSection.getByRole("button", { name: /kept/i }).click();
|
|
||||||
|
|
||||||
// Find the sealed letter in the drawer by recipient name and click it
|
// Find the sealed letter in the drawer by recipient name and click it
|
||||||
logger.info(">> [Drawer] Clicking sealed letter in drawer...");
|
logger.info(">> [Drawer] Clicking sealed letter in drawer...");
|
||||||
const sealedItem = page
|
const sealedItem = page
|
||||||
.getByRole("button", { name: new RegExp(recipientName, "i") })
|
.getByTestId(/^letter-item-/)
|
||||||
|
.filter({ hasText: recipientName })
|
||||||
.first();
|
.first();
|
||||||
await sealedItem.click();
|
await sealedItem.click();
|
||||||
|
|
||||||
// Verify it opens the Reader without a hash
|
// Verify it opens the Reader without a hash
|
||||||
logger.info(">> [Drawer] Verifying Reader page...");
|
logger.info(">> [Drawer] Verifying Reader page...");
|
||||||
// Give it a bit more time for decryption
|
// Give it a bit more time for decryption
|
||||||
await expect(page).toHaveURL(/\/read\/[a-f0-9-]{36}$/, { timeout: 15000 });
|
await expect(page).toHaveURL(/\/read\/[a-f0-9-]{36}$/);
|
||||||
// Reveal and check decrypted content in Reader
|
// Reveal and check decrypted content in Reader
|
||||||
await expect(page.getByText(/breaking the seal/i)).toBeHidden({
|
await expect(page.getByTestId("decryption-overlay")).toBeHidden();
|
||||||
timeout: 10000,
|
|
||||||
});
|
|
||||||
// Flip the envelope and reveal the letter
|
// Flip the envelope and reveal the letter
|
||||||
await revealEnvelope(page);
|
await revealEnvelope(page);
|
||||||
await expect(page.getByTestId("envelope-letter")).toBeHidden({ timeout: 20000 });
|
await expect(page.getByTestId("envelope-letter")).toBeHidden();
|
||||||
|
|
||||||
// Also check if we are redirected to the Reader if we manually go to the Editor URL
|
// Also check if we are redirected to the Reader if we manually go to the Editor URL
|
||||||
const readerUrl = page.url();
|
const readerUrl = page.url();
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ async function registerAndLogin(
|
|||||||
|
|
||||||
await page.goto(activationLink);
|
await page.goto(activationLink);
|
||||||
|
|
||||||
await expect(page.getByText(/account activated/i)).toBeVisible();
|
await expect(page.getByTestId("activation-success-header")).toBeVisible();
|
||||||
await page.getByTestId("start-writing-btn").click();
|
await page.getByTestId("start-writing-btn").click();
|
||||||
|
|
||||||
// Dismiss the Welcom Modal and Perform Login
|
// Dismiss the Welcom Modal and Perform Login
|
||||||
|
|||||||
@@ -35,6 +35,7 @@ export function DrawerSection({
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
|
data-testid={`drawer-section-${id}`}
|
||||||
className={`w-full p-[24px_28px] cursor-pointer flex items-center gap-5 transition-all duration-2000 ease-in-out outline-none focus-visible:ring-2 focus-visible:ring-primary/50 border border-base-content/10 text-left bg-linear-to-r from-transparent to-base-100/40`}
|
className={`w-full p-[24px_28px] cursor-pointer flex items-center gap-5 transition-all duration-2000 ease-in-out outline-none focus-visible:ring-2 focus-visible:ring-primary/50 border border-base-content/10 text-left bg-linear-to-r from-transparent to-base-100/40`}
|
||||||
>
|
>
|
||||||
<div className="flex-1">
|
<div className="flex-1">
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ export function LetterItem({
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
onClick={handleNavigate}
|
onClick={handleNavigate}
|
||||||
|
data-testid={`letter-item-${id}`}
|
||||||
className={`${isLocked ? "pointer-events-none" : ""} p-4 border-base-content/3 flex items-start gap-4 hover:bg-base-300 transition-all delay-75 duration-100 group text-left cursor-pointer w-9/12 mx-auto hover:scale-120 hover:h-24 hover:-translate-y-3 hover:pb-4 hover:border-x-5 hover:border-t-5 border-t-2 hover:-mb-2`}
|
className={`${isLocked ? "pointer-events-none" : ""} p-4 border-base-content/3 flex items-start gap-4 hover:bg-base-300 transition-all delay-75 duration-100 group text-left cursor-pointer w-9/12 mx-auto hover:scale-120 hover:h-24 hover:-translate-y-3 hover:pb-4 hover:border-x-5 hover:border-t-5 border-t-2 hover:-mb-2`}
|
||||||
>
|
>
|
||||||
<div className="text-[0.85rem] italic text-base-content/40 flex-1 truncate group-hover:text-base-content/60 transition-none animate-[opacity_200ms_linear_forwards]">
|
<div className="text-[0.85rem] italic text-base-content/40 flex-1 truncate group-hover:text-base-content/60 transition-none animate-[opacity_200ms_linear_forwards]">
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ export function PostSealModal({
|
|||||||
type = "KEPT",
|
type = "KEPT",
|
||||||
}: PostSealModalProps) {
|
}: PostSealModalProps) {
|
||||||
return (
|
return (
|
||||||
<Modal isOpen={!!sealedTargetId}>
|
<Modal isOpen={!!sealedTargetId} data-testid="post-seal-modal">
|
||||||
<LockIcon size={32} weight="duotone" className="text-primary mt-3" />
|
<LockIcon size={32} weight="duotone" className="text-primary mt-3" />
|
||||||
<h3 className="font-serif text-2xl">Your letter is sealed</h3>
|
<h3 className="font-serif text-2xl">Your letter is sealed</h3>
|
||||||
<p className="text-base-content/60">
|
<p className="text-base-content/60">
|
||||||
|
|||||||
@@ -14,7 +14,11 @@ export function ShareModal({ shareLink, setShareLink }: ShareModalProps) {
|
|||||||
};
|
};
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Modal isOpen={!!shareLink} onClose={() => setShareLink(null)}>
|
<Modal
|
||||||
|
isOpen={!!shareLink}
|
||||||
|
onClose={() => setShareLink(null)}
|
||||||
|
data-testid="share-letter-modal"
|
||||||
|
>
|
||||||
<div className="flex flex-col items-center justify-center text-center gap-6 py-4">
|
<div className="flex flex-col items-center justify-center text-center gap-6 py-4">
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
<PaperPlaneTiltIcon
|
<PaperPlaneTiltIcon
|
||||||
@@ -47,6 +51,7 @@ export function ShareModal({ shareLink, setShareLink }: ShareModalProps) {
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
onClick={copyToClipboard}
|
onClick={copyToClipboard}
|
||||||
|
data-testid="copy-link-btn"
|
||||||
className="btn btn-primary font-sans btn-sm rounded-tl-xl rounded-bl-xl rounded-tr-full rounded-br-full"
|
className="btn btn-primary font-sans btn-sm rounded-tl-xl rounded-bl-xl rounded-tr-full rounded-br-full"
|
||||||
>
|
>
|
||||||
Copy
|
Copy
|
||||||
|
|||||||
@@ -5,17 +5,27 @@ interface ModalProps {
|
|||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
onClose?: () => void;
|
onClose?: () => void;
|
||||||
children: ReactNode;
|
children: ReactNode;
|
||||||
|
"data-testid"?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function Modal({ isOpen, onClose, children }: ModalProps) {
|
export function Modal({
|
||||||
|
isOpen,
|
||||||
|
onClose,
|
||||||
|
children,
|
||||||
|
"data-testid": testId,
|
||||||
|
}: ModalProps) {
|
||||||
if (!isOpen) return null;
|
if (!isOpen) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="modal modal-open modal-middle backdrop-blur-md before:absolute before:top-0 before:left-0 before:w-full before:h-full before:content-[''] before:opacity-[0.03] before:z-10 before:pointer-events-none before:bg-[url('assets/noise.gif')]">
|
<div
|
||||||
|
data-testid={testId}
|
||||||
|
className="modal modal-open modal-middle backdrop-blur-md before:absolute before:top-0 before:left-0 before:w-full before:h-full before:content-[''] before:opacity-[0.03] before:z-10 before:pointer-events-none before:bg-[url('assets/noise.gif')]"
|
||||||
|
>
|
||||||
<div className="modal-box relative bg-base-100/60 flex flex-col items-center text-center gap-6">
|
<div className="modal-box relative bg-base-100/60 flex flex-col items-center text-center gap-6">
|
||||||
{onClose && (
|
{onClose && (
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
|
data-testid="modal-close-btn"
|
||||||
className="btn btn-sm btn-circle btn-ghost absolute right-2 top-2 z-20"
|
className="btn btn-sm btn-circle btn-ghost absolute right-2 top-2 z-20"
|
||||||
onClick={onClose}
|
onClick={onClose}
|
||||||
aria-label="Close"
|
aria-label="Close"
|
||||||
|
|||||||
@@ -52,13 +52,13 @@ export default function Activate() {
|
|||||||
className="text-success"
|
className="text-success"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<h2 className="font-display text-xl text-success">
|
<h2 data-testid="activation-success-header" className="font-display text-xl text-success">
|
||||||
Account Activated!
|
You're in.
|
||||||
</h2>
|
</h2>
|
||||||
<p className="opacity-70 leading-relaxed">
|
<p className="opacity-70 leading-relaxed">
|
||||||
Welcome to <Logo scale={1} />
|
Welcome to <Logo scale={1} />
|
||||||
<br />
|
<br />
|
||||||
Your identity is now verified and ready for timeless letters.
|
Just one more step and you can start writing timeless letters.
|
||||||
</p>
|
</p>
|
||||||
<div className="divider opacity-10 my-0"></div>
|
<div className="divider opacity-10 my-0"></div>
|
||||||
<button
|
<button
|
||||||
@@ -72,7 +72,7 @@ export default function Activate() {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
Start Writing
|
I'm ready
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -376,7 +376,10 @@ export default function Editor() {
|
|||||||
weight="bold"
|
weight="bold"
|
||||||
className="animate-spin text-primary"
|
className="animate-spin text-primary"
|
||||||
/>
|
/>
|
||||||
<p className="text-xxs uppercase tracking-widester font-bold text-base-content/40">
|
<p
|
||||||
|
data-testid="opening-draft-overlay"
|
||||||
|
className="text-xxs uppercase tracking-widester font-bold text-base-content/40"
|
||||||
|
>
|
||||||
Opening your draft...
|
Opening your draft...
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -406,6 +409,7 @@ export default function Editor() {
|
|||||||
{saveOverlay === "SAVED" && (
|
{saveOverlay === "SAVED" && (
|
||||||
<div
|
<div
|
||||||
role="alert"
|
role="alert"
|
||||||
|
data-testid="save-success-toast"
|
||||||
className={`alert alert-success shadow-lg transition-all ease-in-out duration-2000 ${
|
className={`alert alert-success shadow-lg transition-all ease-in-out duration-2000 ${
|
||||||
showSaveOverlay
|
showSaveOverlay
|
||||||
? "opacity-100 scale-100 translate-y-0"
|
? "opacity-100 scale-100 translate-y-0"
|
||||||
|
|||||||
@@ -217,7 +217,10 @@ export default function Reader() {
|
|||||||
<Logo />
|
<Logo />
|
||||||
<div className="flex flex-col items-center gap-2">
|
<div className="flex flex-col items-center gap-2">
|
||||||
<span className="loading loading-ring loading-md text-primary/40"></span>
|
<span className="loading loading-ring loading-md text-primary/40"></span>
|
||||||
<p className="text-xs uppercase tracking-widest text-base-content/20 animate-pulse">
|
<p
|
||||||
|
data-testid="decryption-overlay"
|
||||||
|
className="text-xs uppercase tracking-widest text-base-content/20 animate-pulse"
|
||||||
|
>
|
||||||
Breaking the seal...
|
Breaking the seal...
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user