Skip to content

Commit 49609b0

Browse files
committed
feat(groq): implement comprehensive error handling
- Add Groq-specific error classes for better error classification: GroqError as base, GroqAPIError for API errors, GroqAuthenticationError (401), GroqRateLimitError (429), GroqTimeoutError, GroqConnectionError, and GroqValidationError - Improve error handling in createChatCompletion with proper error mapping, retry logic, comprehensive logging, and fixed cache-related scoping - Update TODO.md to reflect completed error handling tasks test: Add initial Groq client integration tests - Add basic client initialization and chat completion tests - Create comprehensive test plan for Groq client - Add GROQ_API_KEY to .env.example - Set up test infrastructure with logging and environment handling test: Add initial Groq client integration tests - Add basic client initialization and chat completion tests - Create comprehensive test plan for Groq client - Add GROQ_API_KEY to .env.example - Set up test infrastructure with logging and environment handling add groq groq vibes well move groq client to external
1 parent 6330f3a commit 49609b0

12 files changed

+874
-424
lines changed

lib/a11y/utils.ts

+212-95
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,8 @@ import {
66
PlaywrightCommandMethodNotSupportedException,
77
PlaywrightCommandException,
88
} from "@/types/playwright";
9+
import { actMethods, getAvailableMethods } from "../actions";
10+
import { exhaustiveMatchingGuard } from "../utils";
911

1012
// Parser function for str output
1113
export function formatSimplifiedTree(
@@ -464,24 +466,54 @@ export async function performPlaywrightMethod(
464466
},
465467
});
466468

467-
if (method === "scrollIntoView") {
468-
logger({
469-
category: "action",
470-
message: "scrolling element into view",
471-
level: 2,
472-
auxiliary: {
473-
xpath: {
474-
value: xpath,
475-
type: "string",
476-
},
477-
},
478-
});
479-
try {
480-
await locator
481-
.evaluate((element: HTMLElement) => {
482-
element.scrollIntoView({ behavior: "smooth", block: "center" });
483-
})
484-
.catch((e: Error) => {
469+
if (actMethods.includes(method as (typeof actMethods)[number])) {
470+
const action = method as (typeof actMethods)[number];
471+
switch (action) {
472+
case "goBack": {
473+
await stagehandPage.goBack({
474+
waitUntil: "domcontentloaded",
475+
});
476+
break;
477+
}
478+
case "scrollIntoView": {
479+
logger({
480+
category: "action",
481+
message: "scrolling element into view",
482+
level: 2,
483+
auxiliary: {
484+
xpath: {
485+
value: xpath,
486+
type: "string",
487+
},
488+
},
489+
});
490+
try {
491+
await locator
492+
.evaluate((element: HTMLElement) => {
493+
element.scrollIntoView({ behavior: "smooth", block: "center" });
494+
})
495+
.catch((e: Error) => {
496+
logger({
497+
category: "action",
498+
message: "error scrolling element into view",
499+
level: 1,
500+
auxiliary: {
501+
error: {
502+
value: e.message,
503+
type: "string",
504+
},
505+
trace: {
506+
value: e.stack,
507+
type: "string",
508+
},
509+
xpath: {
510+
value: xpath,
511+
type: "string",
512+
},
513+
},
514+
});
515+
});
516+
} catch (e) {
485517
logger({
486518
category: "action",
487519
message: "error scrolling element into view",
@@ -501,89 +533,173 @@ export async function performPlaywrightMethod(
501533
},
502534
},
503535
});
504-
});
505-
} catch (e) {
506-
logger({
507-
category: "action",
508-
message: "error scrolling element into view",
509-
level: 1,
510-
auxiliary: {
511-
error: {
512-
value: e.message,
513-
type: "string",
514-
},
515-
trace: {
516-
value: e.stack,
517-
type: "string",
518-
},
519-
xpath: {
520-
value: xpath,
521-
type: "string",
522-
},
523-
},
524-
});
525536

526-
throw new PlaywrightCommandException(e.message);
527-
}
528-
} else if (method === "fill" || method === "type") {
529-
try {
530-
await locator.fill("");
531-
await locator.click();
532-
const text = args[0]?.toString();
533-
for (const char of text) {
534-
await stagehandPage.keyboard.type(char, {
535-
delay: Math.random() * 50 + 25,
536-
});
537+
throw new PlaywrightCommandException(e.message);
538+
}
539+
break;
537540
}
538-
} catch (e) {
539-
logger({
540-
category: "action",
541-
message: "error filling element",
542-
level: 1,
543-
auxiliary: {
544-
error: {
545-
value: e.message,
546-
type: "string",
547-
},
548-
trace: {
549-
value: e.stack,
550-
type: "string",
551-
},
552-
xpath: {
553-
value: xpath,
554-
type: "string",
555-
},
556-
},
557-
});
541+
case "fill": // fall through to type
542+
case "type": {
543+
try {
544+
await locator.fill("");
545+
await locator.click();
546+
const text = args[0]?.toString();
547+
for (const char of text) {
548+
await stagehandPage.keyboard.type(char, {
549+
delay: Math.random() * 50 + 25,
550+
});
551+
}
552+
} catch (e) {
553+
logger({
554+
category: "action",
555+
message: "error filling element",
556+
level: 1,
557+
auxiliary: {
558+
error: {
559+
value: e.message,
560+
type: "string",
561+
},
562+
trace: {
563+
value: e.stack,
564+
type: "string",
565+
},
566+
xpath: {
567+
value: xpath,
568+
type: "string",
569+
},
570+
},
571+
});
558572

559-
throw new PlaywrightCommandException(e.message);
560-
}
561-
} else if (method === "press") {
562-
try {
563-
const key = args[0]?.toString();
564-
await stagehandPage.keyboard.press(key);
565-
} catch (e) {
566-
logger({
567-
category: "action",
568-
message: "error pressing key",
569-
level: 1,
570-
auxiliary: {
571-
error: {
572-
value: e.message,
573-
type: "string",
574-
},
575-
trace: {
576-
value: e.stack,
577-
type: "string",
573+
throw new PlaywrightCommandException(e.message);
574+
}
575+
break;
576+
}
577+
// Handle navigation if a new page is opened
578+
case "click": {
579+
logger({
580+
category: "action",
581+
message: "clicking element, checking for page navigation",
582+
level: 1,
583+
auxiliary: {
584+
xpath: {
585+
value: xpath,
586+
type: "string",
587+
},
578588
},
579-
key: {
580-
value: args[0]?.toString() ?? "unknown",
581-
type: "string",
589+
});
590+
591+
const newOpenedTab = await Promise.race([
592+
new Promise<Page | null>((resolve) => {
593+
Promise.resolve(stagehandPage.context()).then((context) => {
594+
context.once("page", (page: Page) => resolve(page));
595+
setTimeout(() => resolve(null), 1_500);
596+
});
597+
}),
598+
]);
599+
600+
logger({
601+
category: "action",
602+
message: "clicked element",
603+
level: 1,
604+
auxiliary: {
605+
newOpenedTab: {
606+
value: newOpenedTab ? "opened a new tab" : "no new tabs opened",
607+
type: "string",
608+
},
582609
},
583-
},
584-
});
610+
});
585611

586-
throw new PlaywrightCommandException(e.message);
612+
if (newOpenedTab) {
613+
logger({
614+
category: "action",
615+
message: "new page detected (new tab) with URL",
616+
level: 1,
617+
auxiliary: {
618+
url: {
619+
value: newOpenedTab.url(),
620+
type: "string",
621+
},
622+
},
623+
});
624+
await newOpenedTab.close();
625+
await stagehandPage.goto(newOpenedTab.url());
626+
await stagehandPage.waitForLoadState("domcontentloaded");
627+
}
628+
629+
await Promise.race([
630+
stagehandPage.waitForLoadState("networkidle"),
631+
new Promise((resolve) => setTimeout(resolve, 5_000)),
632+
]).catch((e) => {
633+
logger({
634+
category: "action",
635+
message: "network idle timeout hit",
636+
level: 1,
637+
auxiliary: {
638+
trace: {
639+
value: e.stack,
640+
type: "string",
641+
},
642+
message: {
643+
value: e.message,
644+
type: "string",
645+
},
646+
},
647+
});
648+
});
649+
650+
logger({
651+
category: "action",
652+
message: "finished waiting for (possible) page navigation",
653+
level: 1,
654+
});
655+
656+
if (stagehandPage.url() !== initialUrl) {
657+
logger({
658+
category: "action",
659+
message: "new page detected with URL",
660+
level: 1,
661+
auxiliary: {
662+
url: {
663+
value: stagehandPage.url(),
664+
type: "string",
665+
},
666+
},
667+
});
668+
}
669+
break;
670+
}
671+
case "press": {
672+
try {
673+
const key = args[0]?.toString();
674+
await stagehandPage.keyboard.press(key);
675+
} catch (e) {
676+
logger({
677+
category: "action",
678+
message: "error pressing key",
679+
level: 1,
680+
auxiliary: {
681+
error: {
682+
value: e.message,
683+
type: "string",
684+
},
685+
trace: {
686+
value: e.stack,
687+
type: "string",
688+
},
689+
key: {
690+
value: args[0]?.toString() ?? "unknown",
691+
type: "string",
692+
},
693+
},
694+
});
695+
696+
throw new PlaywrightCommandException(e.message);
697+
}
698+
break;
699+
}
700+
default: {
701+
throw exhaustiveMatchingGuard(action);
702+
}
587703
}
588704
} else if (typeof locator[method as keyof typeof locator] === "function") {
589705
// Log current URL before action
@@ -746,6 +862,7 @@ export async function performPlaywrightMethod(
746862

747863
throw new PlaywrightCommandMethodNotSupportedException(
748864
`Method ${method} not supported`,
865+
getAvailableMethods(locator),
749866
);
750867
}
751868
}

lib/actions.ts

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
import { Locator } from "@playwright/test";
2+
3+
const defaultMethods: Set<string> = new Set(getAvailableMethods({} as Locator));
4+
5+
export const actMethods = [
6+
"scrollIntoView",
7+
"press",
8+
"click",
9+
"fill",
10+
"type",
11+
"goBack",
12+
] as const;
13+
14+
export function getAvailableMethods(locator: Locator) {
15+
return Object.keys(locator)
16+
.filter(
17+
(key) =>
18+
!defaultMethods.has(key) &&
19+
key in locator &&
20+
typeof locator[key as keyof Locator] === "function",
21+
)
22+
.concat(actMethods);
23+
}

0 commit comments

Comments
 (0)