6
6
PlaywrightCommandMethodNotSupportedException ,
7
7
PlaywrightCommandException ,
8
8
} from "@/types/playwright" ;
9
+ import { actMethods , getAvailableMethods } from "../actions" ;
10
+ import { exhaustiveMatchingGuard } from "../utils" ;
9
11
10
12
// Parser function for str output
11
13
export function formatSimplifiedTree (
@@ -464,24 +466,54 @@ export async function performPlaywrightMethod(
464
466
} ,
465
467
} ) ;
466
468
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 ) {
485
517
logger ( {
486
518
category : "action" ,
487
519
message : "error scrolling element into view" ,
@@ -501,89 +533,173 @@ export async function performPlaywrightMethod(
501
533
} ,
502
534
} ,
503
535
} ) ;
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
- } ) ;
525
536
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 ;
537
540
}
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
+ } ) ;
558
572
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
+ } ,
578
588
} ,
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
+ } ,
582
609
} ,
583
- } ,
584
- } ) ;
610
+ } ) ;
585
611
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
+ }
587
703
}
588
704
} else if ( typeof locator [ method as keyof typeof locator ] === "function" ) {
589
705
// Log current URL before action
@@ -746,6 +862,7 @@ export async function performPlaywrightMethod(
746
862
747
863
throw new PlaywrightCommandMethodNotSupportedException (
748
864
`Method ${ method } not supported` ,
865
+ getAvailableMethods ( locator ) ,
749
866
) ;
750
867
}
751
868
}
0 commit comments