@octavus/client-sdk 2.7.0 → 2.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.js +162 -101
- package/dist/index.js.map +1 -1
- package/package.json +2 -2
package/dist/index.js
CHANGED
|
@@ -590,8 +590,10 @@ var OctavusChat = class {
|
|
|
590
590
|
};
|
|
591
591
|
if (workerState) {
|
|
592
592
|
const workerPart = state.parts[workerState.partIndex];
|
|
593
|
-
|
|
594
|
-
|
|
593
|
+
state.parts[workerState.partIndex] = {
|
|
594
|
+
...workerPart,
|
|
595
|
+
parts: [...workerPart.parts, operationPart]
|
|
596
|
+
};
|
|
595
597
|
} else {
|
|
596
598
|
state.parts.push(operationPart);
|
|
597
599
|
}
|
|
@@ -611,8 +613,9 @@ var OctavusChat = class {
|
|
|
611
613
|
);
|
|
612
614
|
if (operationPartIndex >= 0) {
|
|
613
615
|
const part = workerPart.parts[operationPartIndex];
|
|
614
|
-
workerPart.parts
|
|
615
|
-
|
|
616
|
+
const updatedParts = [...workerPart.parts];
|
|
617
|
+
updatedParts[operationPartIndex] = { ...part, status: "done" };
|
|
618
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
616
619
|
}
|
|
617
620
|
} else {
|
|
618
621
|
const operationPartIndex = state.parts.findIndex(
|
|
@@ -640,9 +643,9 @@ var OctavusChat = class {
|
|
|
640
643
|
};
|
|
641
644
|
if (workerState) {
|
|
642
645
|
const workerPart = state.parts[workerState.partIndex];
|
|
643
|
-
workerPart.parts
|
|
644
|
-
workerState.currentReasoningPartIndex =
|
|
645
|
-
state.parts[workerState.partIndex] = { ...workerPart };
|
|
646
|
+
const newParts = [...workerPart.parts, reasoningPart];
|
|
647
|
+
workerState.currentReasoningPartIndex = newParts.length - 1;
|
|
648
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: newParts };
|
|
646
649
|
} else {
|
|
647
650
|
state.parts.push(reasoningPart);
|
|
648
651
|
state.currentReasoningPartIndex = state.parts.length - 1;
|
|
@@ -657,15 +660,20 @@ var OctavusChat = class {
|
|
|
657
660
|
if (workerState.currentReasoningPartIndex !== null) {
|
|
658
661
|
const workerPart = state.parts[workerState.partIndex];
|
|
659
662
|
const part = workerPart.parts[workerState.currentReasoningPartIndex];
|
|
660
|
-
|
|
661
|
-
|
|
662
|
-
|
|
663
|
+
const updatedParts = [...workerPart.parts];
|
|
664
|
+
updatedParts[workerState.currentReasoningPartIndex] = {
|
|
665
|
+
...part,
|
|
666
|
+
text: part.text + event.delta
|
|
667
|
+
};
|
|
668
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
663
669
|
}
|
|
664
670
|
} else {
|
|
665
671
|
if (state.currentReasoningPartIndex !== null) {
|
|
666
672
|
const part = state.parts[state.currentReasoningPartIndex];
|
|
667
|
-
|
|
668
|
-
|
|
673
|
+
state.parts[state.currentReasoningPartIndex] = {
|
|
674
|
+
...part,
|
|
675
|
+
text: part.text + event.delta
|
|
676
|
+
};
|
|
669
677
|
}
|
|
670
678
|
if (state.activeBlock) {
|
|
671
679
|
state.activeBlock.reasoning += event.delta;
|
|
@@ -681,15 +689,17 @@ var OctavusChat = class {
|
|
|
681
689
|
if (workerState.currentReasoningPartIndex !== null) {
|
|
682
690
|
const workerPart = state.parts[workerState.partIndex];
|
|
683
691
|
const part = workerPart.parts[workerState.currentReasoningPartIndex];
|
|
684
|
-
|
|
685
|
-
|
|
686
|
-
|
|
692
|
+
const updatedParts = [...workerPart.parts];
|
|
693
|
+
updatedParts[workerState.currentReasoningPartIndex] = {
|
|
694
|
+
...part,
|
|
695
|
+
status: "done"
|
|
696
|
+
};
|
|
697
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
687
698
|
workerState.currentReasoningPartIndex = null;
|
|
688
699
|
}
|
|
689
700
|
} else if (state.currentReasoningPartIndex !== null) {
|
|
690
701
|
const part = state.parts[state.currentReasoningPartIndex];
|
|
691
|
-
|
|
692
|
-
state.parts[state.currentReasoningPartIndex] = { ...part };
|
|
702
|
+
state.parts[state.currentReasoningPartIndex] = { ...part, status: "done" };
|
|
693
703
|
state.currentReasoningPartIndex = null;
|
|
694
704
|
}
|
|
695
705
|
this.updateStreamingMessage();
|
|
@@ -713,11 +723,11 @@ var OctavusChat = class {
|
|
|
713
723
|
};
|
|
714
724
|
if (workerState) {
|
|
715
725
|
const workerPart = state.parts[workerState.partIndex];
|
|
716
|
-
workerPart.parts
|
|
717
|
-
workerState.currentObjectPartIndex =
|
|
726
|
+
const newParts = [...workerPart.parts, objectPart];
|
|
727
|
+
workerState.currentObjectPartIndex = newParts.length - 1;
|
|
718
728
|
workerState.accumulatedJson = "";
|
|
719
729
|
workerState.currentTextPartIndex = null;
|
|
720
|
-
state.parts[workerState.partIndex] = { ...workerPart };
|
|
730
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: newParts };
|
|
721
731
|
} else {
|
|
722
732
|
state.parts.push(objectPart);
|
|
723
733
|
state.currentObjectPartIndex = state.parts.length - 1;
|
|
@@ -733,10 +743,10 @@ var OctavusChat = class {
|
|
|
733
743
|
};
|
|
734
744
|
if (workerState) {
|
|
735
745
|
const workerPart = state.parts[workerState.partIndex];
|
|
736
|
-
workerPart.parts
|
|
737
|
-
workerState.currentTextPartIndex =
|
|
746
|
+
const newParts = [...workerPart.parts, textPart];
|
|
747
|
+
workerState.currentTextPartIndex = newParts.length - 1;
|
|
738
748
|
workerState.currentObjectPartIndex = null;
|
|
739
|
-
state.parts[workerState.partIndex] = { ...workerPart };
|
|
749
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: newParts };
|
|
740
750
|
} else {
|
|
741
751
|
state.parts.push(textPart);
|
|
742
752
|
state.currentTextPartIndex = state.parts.length - 1;
|
|
@@ -757,28 +767,33 @@ var OctavusChat = class {
|
|
|
757
767
|
const part = workerPart.parts[workerState.currentObjectPartIndex];
|
|
758
768
|
const parsed = parsePartialJson(workerState.accumulatedJson);
|
|
759
769
|
if (parsed !== void 0) {
|
|
760
|
-
|
|
761
|
-
|
|
770
|
+
const updatedParts = [...workerPart.parts];
|
|
771
|
+
updatedParts[workerState.currentObjectPartIndex] = { ...part, partial: parsed };
|
|
772
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
762
773
|
}
|
|
763
774
|
} else if (workerState.currentTextPartIndex !== null) {
|
|
764
775
|
const part = workerPart.parts[workerState.currentTextPartIndex];
|
|
765
|
-
|
|
766
|
-
|
|
776
|
+
const updatedParts = [...workerPart.parts];
|
|
777
|
+
updatedParts[workerState.currentTextPartIndex] = {
|
|
778
|
+
...part,
|
|
779
|
+
text: part.text + event.delta
|
|
780
|
+
};
|
|
781
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
767
782
|
}
|
|
768
|
-
state.parts[workerState.partIndex] = { ...workerPart };
|
|
769
783
|
} else {
|
|
770
784
|
if (state.currentObjectPartIndex !== null) {
|
|
771
785
|
state.accumulatedJson += event.delta;
|
|
772
786
|
const part = state.parts[state.currentObjectPartIndex];
|
|
773
787
|
const parsed = parsePartialJson(state.accumulatedJson);
|
|
774
788
|
if (parsed !== void 0) {
|
|
775
|
-
|
|
776
|
-
state.parts[state.currentObjectPartIndex] = { ...part };
|
|
789
|
+
state.parts[state.currentObjectPartIndex] = { ...part, partial: parsed };
|
|
777
790
|
}
|
|
778
791
|
} else if (state.currentTextPartIndex !== null) {
|
|
779
792
|
const part = state.parts[state.currentTextPartIndex];
|
|
780
|
-
|
|
781
|
-
|
|
793
|
+
state.parts[state.currentTextPartIndex] = {
|
|
794
|
+
...part,
|
|
795
|
+
text: part.text + event.delta
|
|
796
|
+
};
|
|
782
797
|
}
|
|
783
798
|
if (state.activeBlock) {
|
|
784
799
|
state.activeBlock.text += event.delta;
|
|
@@ -792,45 +807,57 @@ var OctavusChat = class {
|
|
|
792
807
|
const workerState = workerId ? state.activeWorkers.get(workerId) : void 0;
|
|
793
808
|
if (workerState) {
|
|
794
809
|
const workerPart = state.parts[workerState.partIndex];
|
|
810
|
+
const updatedParts = [...workerPart.parts];
|
|
795
811
|
if (workerState.currentObjectPartIndex !== null) {
|
|
796
812
|
const part = workerPart.parts[workerState.currentObjectPartIndex];
|
|
797
813
|
try {
|
|
798
814
|
const finalObject = JSON.parse(workerState.accumulatedJson);
|
|
799
|
-
|
|
800
|
-
|
|
801
|
-
|
|
815
|
+
updatedParts[workerState.currentObjectPartIndex] = {
|
|
816
|
+
...part,
|
|
817
|
+
object: finalObject,
|
|
818
|
+
partial: finalObject,
|
|
819
|
+
status: "done"
|
|
820
|
+
};
|
|
802
821
|
} catch {
|
|
803
|
-
|
|
804
|
-
|
|
822
|
+
updatedParts[workerState.currentObjectPartIndex] = {
|
|
823
|
+
...part,
|
|
824
|
+
status: "error",
|
|
825
|
+
error: "Failed to parse response as JSON"
|
|
826
|
+
};
|
|
805
827
|
}
|
|
806
|
-
workerPart.parts[workerState.currentObjectPartIndex] = { ...part };
|
|
807
828
|
workerState.currentObjectPartIndex = null;
|
|
808
829
|
workerState.accumulatedJson = "";
|
|
809
830
|
} else if (workerState.currentTextPartIndex !== null) {
|
|
810
831
|
const part = workerPart.parts[workerState.currentTextPartIndex];
|
|
811
|
-
|
|
812
|
-
|
|
832
|
+
updatedParts[workerState.currentTextPartIndex] = {
|
|
833
|
+
...part,
|
|
834
|
+
status: "done"
|
|
835
|
+
};
|
|
813
836
|
workerState.currentTextPartIndex = null;
|
|
814
837
|
}
|
|
815
|
-
state.parts[workerState.partIndex] = { ...workerPart };
|
|
838
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
816
839
|
} else if (state.currentObjectPartIndex !== null) {
|
|
817
840
|
const part = state.parts[state.currentObjectPartIndex];
|
|
818
841
|
try {
|
|
819
842
|
const finalObject = JSON.parse(state.accumulatedJson);
|
|
820
|
-
|
|
821
|
-
|
|
822
|
-
|
|
843
|
+
state.parts[state.currentObjectPartIndex] = {
|
|
844
|
+
...part,
|
|
845
|
+
object: finalObject,
|
|
846
|
+
partial: finalObject,
|
|
847
|
+
status: "done"
|
|
848
|
+
};
|
|
823
849
|
} catch {
|
|
824
|
-
|
|
825
|
-
|
|
850
|
+
state.parts[state.currentObjectPartIndex] = {
|
|
851
|
+
...part,
|
|
852
|
+
status: "error",
|
|
853
|
+
error: "Failed to parse response as JSON"
|
|
854
|
+
};
|
|
826
855
|
}
|
|
827
|
-
state.parts[state.currentObjectPartIndex] = { ...part };
|
|
828
856
|
state.currentObjectPartIndex = null;
|
|
829
857
|
state.accumulatedJson = "";
|
|
830
858
|
} else if (state.currentTextPartIndex !== null) {
|
|
831
859
|
const part = state.parts[state.currentTextPartIndex];
|
|
832
|
-
|
|
833
|
-
state.parts[state.currentTextPartIndex] = { ...part };
|
|
860
|
+
state.parts[state.currentTextPartIndex] = { ...part, status: "done" };
|
|
834
861
|
state.currentTextPartIndex = null;
|
|
835
862
|
}
|
|
836
863
|
this.updateStreamingMessage();
|
|
@@ -845,14 +872,18 @@ var OctavusChat = class {
|
|
|
845
872
|
toolName: event.toolName,
|
|
846
873
|
displayName: event.title,
|
|
847
874
|
args: {},
|
|
875
|
+
result: void 0,
|
|
876
|
+
error: void 0,
|
|
848
877
|
status: "pending",
|
|
849
878
|
thread: threadForPart(state.activeBlock?.thread)
|
|
850
879
|
};
|
|
851
880
|
if (workerState) {
|
|
852
881
|
workerState.toolInputBuffers.set(event.toolCallId, "");
|
|
853
882
|
const workerPart = state.parts[workerState.partIndex];
|
|
854
|
-
|
|
855
|
-
|
|
883
|
+
state.parts[workerState.partIndex] = {
|
|
884
|
+
...workerPart,
|
|
885
|
+
parts: [...workerPart.parts, toolPart]
|
|
886
|
+
};
|
|
856
887
|
} else {
|
|
857
888
|
state.toolInputBuffers.set(event.toolCallId, "");
|
|
858
889
|
state.parts.push(toolPart);
|
|
@@ -878,9 +909,12 @@ var OctavusChat = class {
|
|
|
878
909
|
const toolPart = workerPart.parts[toolPartIndex];
|
|
879
910
|
const parsed = parsePartialJson(accumulated);
|
|
880
911
|
if (parsed !== void 0) {
|
|
881
|
-
|
|
882
|
-
|
|
883
|
-
|
|
912
|
+
const updatedParts = [...workerPart.parts];
|
|
913
|
+
updatedParts[toolPartIndex] = {
|
|
914
|
+
...toolPart,
|
|
915
|
+
args: parsed
|
|
916
|
+
};
|
|
917
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
884
918
|
this.updateStreamingMessage();
|
|
885
919
|
}
|
|
886
920
|
}
|
|
@@ -895,8 +929,10 @@ var OctavusChat = class {
|
|
|
895
929
|
const toolPart = state.parts[toolPartIndex];
|
|
896
930
|
const parsed = parsePartialJson(accumulated);
|
|
897
931
|
if (parsed !== void 0) {
|
|
898
|
-
|
|
899
|
-
|
|
932
|
+
state.parts[toolPartIndex] = {
|
|
933
|
+
...toolPart,
|
|
934
|
+
args: parsed
|
|
935
|
+
};
|
|
900
936
|
this.updateStreamingMessage();
|
|
901
937
|
}
|
|
902
938
|
}
|
|
@@ -916,10 +952,13 @@ var OctavusChat = class {
|
|
|
916
952
|
);
|
|
917
953
|
if (toolPartIndex >= 0) {
|
|
918
954
|
const part = workerPart.parts[toolPartIndex];
|
|
919
|
-
|
|
920
|
-
|
|
921
|
-
|
|
922
|
-
|
|
955
|
+
const updatedParts = [...workerPart.parts];
|
|
956
|
+
updatedParts[toolPartIndex] = {
|
|
957
|
+
...part,
|
|
958
|
+
args: event.input,
|
|
959
|
+
status: "running"
|
|
960
|
+
};
|
|
961
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
923
962
|
this.updateStreamingMessage();
|
|
924
963
|
}
|
|
925
964
|
} else {
|
|
@@ -929,9 +968,11 @@ var OctavusChat = class {
|
|
|
929
968
|
);
|
|
930
969
|
if (toolPartIndex >= 0) {
|
|
931
970
|
const part = state.parts[toolPartIndex];
|
|
932
|
-
|
|
933
|
-
|
|
934
|
-
|
|
971
|
+
state.parts[toolPartIndex] = {
|
|
972
|
+
...part,
|
|
973
|
+
args: event.input,
|
|
974
|
+
status: "running"
|
|
975
|
+
};
|
|
935
976
|
this.updateStreamingMessage();
|
|
936
977
|
}
|
|
937
978
|
}
|
|
@@ -947,10 +988,13 @@ var OctavusChat = class {
|
|
|
947
988
|
);
|
|
948
989
|
if (toolPartIndex >= 0) {
|
|
949
990
|
const part = workerPart.parts[toolPartIndex];
|
|
950
|
-
|
|
951
|
-
|
|
952
|
-
|
|
953
|
-
|
|
991
|
+
const updatedParts = [...workerPart.parts];
|
|
992
|
+
updatedParts[toolPartIndex] = {
|
|
993
|
+
...part,
|
|
994
|
+
result: event.output,
|
|
995
|
+
status: "done"
|
|
996
|
+
};
|
|
997
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
954
998
|
this.updateStreamingMessage();
|
|
955
999
|
}
|
|
956
1000
|
} else {
|
|
@@ -959,9 +1003,11 @@ var OctavusChat = class {
|
|
|
959
1003
|
);
|
|
960
1004
|
if (toolPartIndex >= 0) {
|
|
961
1005
|
const part = state.parts[toolPartIndex];
|
|
962
|
-
|
|
963
|
-
|
|
964
|
-
|
|
1006
|
+
state.parts[toolPartIndex] = {
|
|
1007
|
+
...part,
|
|
1008
|
+
result: event.output,
|
|
1009
|
+
status: "done"
|
|
1010
|
+
};
|
|
965
1011
|
this.updateStreamingMessage();
|
|
966
1012
|
}
|
|
967
1013
|
}
|
|
@@ -977,10 +1023,13 @@ var OctavusChat = class {
|
|
|
977
1023
|
);
|
|
978
1024
|
if (toolPartIndex >= 0) {
|
|
979
1025
|
const part = workerPart.parts[toolPartIndex];
|
|
980
|
-
|
|
981
|
-
|
|
982
|
-
|
|
983
|
-
|
|
1026
|
+
const updatedParts = [...workerPart.parts];
|
|
1027
|
+
updatedParts[toolPartIndex] = {
|
|
1028
|
+
...part,
|
|
1029
|
+
error: event.error,
|
|
1030
|
+
status: "error"
|
|
1031
|
+
};
|
|
1032
|
+
state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };
|
|
984
1033
|
this.updateStreamingMessage();
|
|
985
1034
|
}
|
|
986
1035
|
} else {
|
|
@@ -989,9 +1038,11 @@ var OctavusChat = class {
|
|
|
989
1038
|
);
|
|
990
1039
|
if (toolPartIndex >= 0) {
|
|
991
1040
|
const part = state.parts[toolPartIndex];
|
|
992
|
-
|
|
993
|
-
|
|
994
|
-
|
|
1041
|
+
state.parts[toolPartIndex] = {
|
|
1042
|
+
...part,
|
|
1043
|
+
error: event.error,
|
|
1044
|
+
status: "error"
|
|
1045
|
+
};
|
|
995
1046
|
this.updateStreamingMessage();
|
|
996
1047
|
}
|
|
997
1048
|
}
|
|
@@ -1024,8 +1075,10 @@ var OctavusChat = class {
|
|
|
1024
1075
|
}
|
|
1025
1076
|
if (workerState) {
|
|
1026
1077
|
const workerPart = state.parts[workerState.partIndex];
|
|
1027
|
-
|
|
1028
|
-
|
|
1078
|
+
state.parts[workerState.partIndex] = {
|
|
1079
|
+
...workerPart,
|
|
1080
|
+
parts: [...workerPart.parts, sourcePart]
|
|
1081
|
+
};
|
|
1029
1082
|
} else {
|
|
1030
1083
|
state.parts.push(sourcePart);
|
|
1031
1084
|
}
|
|
@@ -1047,8 +1100,10 @@ var OctavusChat = class {
|
|
|
1047
1100
|
};
|
|
1048
1101
|
if (workerState) {
|
|
1049
1102
|
const workerPart = state.parts[workerState.partIndex];
|
|
1050
|
-
|
|
1051
|
-
|
|
1103
|
+
state.parts[workerState.partIndex] = {
|
|
1104
|
+
...workerPart,
|
|
1105
|
+
parts: [...workerPart.parts, filePart]
|
|
1106
|
+
};
|
|
1052
1107
|
} else {
|
|
1053
1108
|
state.parts.push(filePart);
|
|
1054
1109
|
}
|
|
@@ -1065,7 +1120,7 @@ var OctavusChat = class {
|
|
|
1065
1120
|
let partIndex;
|
|
1066
1121
|
if (existingIndex !== -1) {
|
|
1067
1122
|
const existingPart = state.parts[existingIndex];
|
|
1068
|
-
|
|
1123
|
+
state.parts[existingIndex] = { ...existingPart, status: "running" };
|
|
1069
1124
|
partIndex = existingIndex;
|
|
1070
1125
|
} else {
|
|
1071
1126
|
const workerPart = {
|
|
@@ -1095,21 +1150,23 @@ var OctavusChat = class {
|
|
|
1095
1150
|
const workerState = state.activeWorkers.get(event.workerId);
|
|
1096
1151
|
if (workerState !== void 0) {
|
|
1097
1152
|
const part = state.parts[workerState.partIndex];
|
|
1098
|
-
|
|
1099
|
-
|
|
1100
|
-
|
|
1101
|
-
|
|
1102
|
-
|
|
1103
|
-
|
|
1153
|
+
state.parts[workerState.partIndex] = {
|
|
1154
|
+
...part,
|
|
1155
|
+
output: event.output,
|
|
1156
|
+
error: event.error,
|
|
1157
|
+
status: event.error ? "error" : "done",
|
|
1158
|
+
parts: part.parts.map((p) => {
|
|
1159
|
+
if (p.type === "text" || p.type === "reasoning") {
|
|
1160
|
+
if (p.status === "streaming") {
|
|
1161
|
+
return { ...p, status: "done" };
|
|
1162
|
+
}
|
|
1163
|
+
}
|
|
1164
|
+
if (p.type === "object" && p.status === "streaming") {
|
|
1104
1165
|
return { ...p, status: "done" };
|
|
1105
1166
|
}
|
|
1106
|
-
|
|
1107
|
-
|
|
1108
|
-
|
|
1109
|
-
}
|
|
1110
|
-
return p;
|
|
1111
|
-
});
|
|
1112
|
-
state.parts[workerState.partIndex] = { ...part };
|
|
1167
|
+
return p;
|
|
1168
|
+
})
|
|
1169
|
+
};
|
|
1113
1170
|
state.activeWorkers.delete(event.workerId);
|
|
1114
1171
|
}
|
|
1115
1172
|
this.updateStreamingMessage();
|
|
@@ -1197,9 +1254,11 @@ var OctavusChat = class {
|
|
|
1197
1254
|
);
|
|
1198
1255
|
if (toolPartIndex >= 0) {
|
|
1199
1256
|
const part = state.parts[toolPartIndex];
|
|
1200
|
-
|
|
1201
|
-
|
|
1202
|
-
|
|
1257
|
+
state.parts[toolPartIndex] = {
|
|
1258
|
+
...part,
|
|
1259
|
+
result: output,
|
|
1260
|
+
status: "done"
|
|
1261
|
+
};
|
|
1203
1262
|
this.updateStreamingMessage();
|
|
1204
1263
|
}
|
|
1205
1264
|
}
|
|
@@ -1214,9 +1273,11 @@ var OctavusChat = class {
|
|
|
1214
1273
|
);
|
|
1215
1274
|
if (toolPartIndex >= 0) {
|
|
1216
1275
|
const part = state.parts[toolPartIndex];
|
|
1217
|
-
|
|
1218
|
-
|
|
1219
|
-
|
|
1276
|
+
state.parts[toolPartIndex] = {
|
|
1277
|
+
...part,
|
|
1278
|
+
error,
|
|
1279
|
+
status: "error"
|
|
1280
|
+
};
|
|
1220
1281
|
this.updateStreamingMessage();
|
|
1221
1282
|
}
|
|
1222
1283
|
}
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/chat.ts","../src/files.ts","../src/stream/reader.ts","../src/transports/types.ts","../src/transports/http.ts","../src/transports/socket.ts","../src/index.ts"],"sourcesContent":["import {\n generateId,\n threadForPart,\n isFileReferenceArray,\n OctavusError,\n type UIMessage,\n type UIMessagePart,\n type UITextPart,\n type UIReasoningPart,\n type UIToolCallPart,\n type UIOperationPart,\n type UISourcePart,\n type UIFilePart,\n type UIObjectPart,\n type UIWorkerPart,\n type DisplayMode,\n type StreamEvent,\n type FileReference,\n type PendingToolCall,\n type ToolResult,\n} from '@octavus/core';\nimport type { Transport } from './transports/types';\nimport { uploadFiles, type UploadFilesOptions } from './files';\n\n/** Block types that are internal operations (not LLM-driven) */\nconst OPERATION_BLOCK_TYPES = new Set(['set-resource', 'serialize-thread', 'generate-image']);\n\n// =============================================================================\n// Types\n// =============================================================================\n\nexport type ChatStatus = 'idle' | 'streaming' | 'error' | 'awaiting-input';\n\n/**\n * Context provided to client tool handlers.\n */\nexport interface ClientToolContext {\n /** Unique identifier for this tool call */\n toolCallId: string;\n /** Name of the tool being called */\n toolName: string;\n /** Signal for cancellation if user stops generation */\n signal: AbortSignal;\n /**\n * Register a file produced by this tool (e.g., a screenshot).\n * Files are sent to the platform alongside the tool result so the LLM\n * can see them as visual content rather than just a JSON URL.\n */\n addFile: (file: FileReference) => void;\n}\n\n/**\n * Handler function for client-side tool execution.\n * Can be:\n * - An async function that executes automatically and returns a result\n * - The string 'interactive' to indicate the tool requires user interaction\n */\nexport type ClientToolHandler =\n | ((args: Record<string, unknown>, ctx: ClientToolContext) => Promise<unknown>)\n | 'interactive';\n\n/**\n * Interactive tool call awaiting user interaction.\n * The `submit` and `cancel` methods are pre-bound to this tool call's ID.\n */\nexport interface InteractiveTool {\n /** Unique identifier for this tool call */\n toolCallId: string;\n /** Name of the tool being called */\n toolName: string;\n /** Arguments passed to the tool */\n args: Record<string, unknown>;\n /**\n * Submit a result for this tool call.\n * Call this when the user has provided input.\n *\n * @param result - The result from user interaction\n */\n submit: (result: unknown) => void;\n /**\n * Cancel this tool call with an optional reason.\n * Call this when the user dismisses the UI without providing input.\n *\n * @param reason - Optional reason for cancellation (default: 'User cancelled')\n */\n cancel: (reason?: string) => void;\n}\n\n/**\n * Internal pending tool state (before binding submit/cancel).\n */\ninterface PendingToolState {\n toolCallId: string;\n toolName: string;\n args: Record<string, unknown>;\n source?: 'llm' | 'block';\n outputVariable?: string;\n blockIndex?: number;\n thread?: string;\n /** Worker ID if this tool call is from a worker execution */\n workerId?: string;\n}\n\n/**\n * Input for creating a user message.\n * Supports text content, structured object content, and file attachments.\n */\nexport interface UserMessageInput {\n /**\n * Content of the message. Can be:\n * - string: Creates a text part\n * - object: Creates an object part (uses `type` field as typeName if present)\n */\n content?: string | Record<string, unknown>;\n /**\n * File attachments (shorthand). Can be:\n * - FileList: From file input element (will be uploaded via uploadFiles)\n * - File[]: Array of File objects (will be uploaded via uploadFiles)\n * - FileReference[]: Already uploaded files (used directly)\n */\n files?: FileList | File[] | FileReference[];\n}\n\nexport interface OctavusChatOptions {\n /**\n * Transport for streaming events.\n * Use `createHttpTransport` for HTTP/SSE or `createSocketTransport` for WebSocket/SockJS.\n */\n transport: Transport;\n\n /**\n * Function to request upload URLs from the platform.\n * Required if you want to use file uploads with FileList/File[].\n *\n * @example\n * ```typescript\n * requestUploadUrls: async (files) => {\n * const response = await fetch('/api/upload-urls', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, files }),\n * });\n * return response.json();\n * }\n * ```\n */\n requestUploadUrls?: UploadFilesOptions['requestUploadUrls'];\n\n /**\n * Client-side tool handlers.\n * Register handlers for tools that should execute in the browser.\n *\n * - If a tool has a handler function: executes automatically\n * - If a tool is marked as 'interactive': appears in `pendingClientTools` with bound `submit()`/`cancel()`\n *\n * @example Automatic client tool\n * ```typescript\n * clientTools: {\n * 'get-browser-location': async () => {\n * const pos = await new Promise((resolve, reject) => {\n * navigator.geolocation.getCurrentPosition(resolve, reject);\n * });\n * return { lat: pos.coords.latitude, lng: pos.coords.longitude };\n * },\n * }\n * ```\n *\n * @example Interactive client tool (user input required)\n * ```typescript\n * clientTools: {\n * 'request-feedback': 'interactive',\n * }\n * // Then render UI based on pendingClientTools['request-feedback']\n * // and call tool.submit(result) or tool.cancel()\n * ```\n */\n clientTools?: Record<string, ClientToolHandler>;\n\n /** Initial messages (for session refresh) */\n initialMessages?: UIMessage[];\n /**\n * Callback when an error occurs.\n * Receives an OctavusError with structured error information.\n *\n * @example\n * ```typescript\n * onError: (error) => {\n * console.error('Chat error:', {\n * type: error.errorType,\n * message: error.message,\n * retryable: error.retryable,\n * provider: error.provider,\n * });\n *\n * // Handle specific error types\n * if (isRateLimitError(error)) {\n * showRetryButton(error.retryAfter);\n * }\n * }\n * ```\n */\n onError?: (error: OctavusError) => void;\n /** Callback when streaming finishes successfully */\n onFinish?: () => void;\n /** Callback when streaming is stopped by user */\n onStop?: () => void;\n /** Callback when a resource is updated */\n onResourceUpdate?: (name: string, value: unknown) => void;\n /**\n * Callback when execution starts with the session/execution ID.\n * Useful for tracking the current execution for activity logs.\n *\n * @example\n * ```typescript\n * onStart: (sessionId) => {\n * setCurrentSessionId(sessionId);\n * }\n * ```\n */\n onStart?: (sessionId: string) => void;\n}\n\n// =============================================================================\n// Internal Types\n// =============================================================================\n\ninterface BlockState {\n blockId: string;\n blockName: string;\n blockType: string;\n display: DisplayMode;\n description?: string;\n outputToChat: boolean;\n thread?: string;\n reasoning: string;\n text: string;\n toolCalls: Map<string, UIToolCallPart>;\n}\n\n/** Tracks state for a worker part being populated */\ninterface WorkerPartState {\n partIndex: number;\n currentTextPartIndex: number | null;\n currentReasoningPartIndex: number | null;\n currentObjectPartIndex: number | null;\n accumulatedJson: string;\n /** Accumulated raw JSON text per tool call ID for progressive partial parsing */\n toolInputBuffers: Map<string, string>;\n}\n\ninterface StreamingState {\n messageId: string;\n parts: UIMessagePart[];\n activeBlock: BlockState | null;\n blocks: Map<string, BlockState>;\n currentTextPartIndex: number | null;\n currentReasoningPartIndex: number | null;\n currentObjectPartIndex: number | null;\n accumulatedJson: string;\n /** Active workers being populated: workerId -> worker state */\n activeWorkers: Map<string, WorkerPartState>;\n /** Accumulated raw JSON text per tool call ID for progressive partial parsing */\n toolInputBuffers: Map<string, string>;\n}\n\ntype Listener = () => void;\n\n// =============================================================================\n// Helpers\n// =============================================================================\n\n/**\n * Create a user message from input with optional file attachments.\n * Parts order: files first (for vision models), then content (text or object).\n */\nfunction createUserMessage(input: UserMessageInput, files?: FileReference[]): UIMessage {\n const parts: UIMessagePart[] = [];\n\n // Add file parts first (vision models expect images before text)\n if (files && files.length > 0) {\n for (const file of files) {\n parts.push({\n type: 'file',\n id: file.id,\n mediaType: file.mediaType,\n url: file.url,\n filename: file.filename,\n size: file.size,\n });\n }\n }\n\n // Add content part after files\n if (input.content !== undefined) {\n if (typeof input.content === 'string') {\n // String content → text part\n parts.push({ type: 'text', text: input.content, status: 'done' });\n } else {\n // Object content → object part\n // Use the object's `type` field as typeName if present, otherwise fallback to 'object'\n const typeName = (input.content as { type?: string }).type ?? 'object';\n parts.push({\n type: 'object',\n id: generateId(),\n typeName,\n object: input.content,\n status: 'done',\n });\n }\n }\n\n return {\n id: generateId(),\n role: 'user',\n parts,\n status: 'done',\n createdAt: new Date(),\n };\n}\n\n/**\n * Parse partial JSON by fixing incomplete structures (unclosed strings, brackets, braces).\n */\nfunction parsePartialJson(jsonText: string): unknown {\n if (!jsonText.trim()) {\n return undefined;\n }\n\n try {\n return JSON.parse(jsonText) as unknown;\n } catch {\n // Continue to fix incomplete JSON\n }\n\n let fixed = jsonText;\n\n // Count unclosed brackets/braces while tracking string boundaries\n let openBraces = 0;\n let openBrackets = 0;\n let inString = false;\n let escaped = false;\n\n for (const char of fixed) {\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === '\\\\') {\n escaped = true;\n continue;\n }\n\n if (char === '\"') {\n inString = !inString;\n continue;\n }\n\n if (!inString) {\n if (char === '{') openBraces += 1;\n else if (char === '}') openBraces -= 1;\n else if (char === '[') openBrackets += 1;\n else if (char === ']') openBrackets -= 1;\n }\n }\n\n // Close unclosed structures\n if (escaped) {\n // If input ends with a dangling backslash, complete the escape sequence.\n fixed += '\\\\';\n }\n if (inString) {\n fixed += '\"';\n }\n while (openBrackets > 0) {\n fixed += ']';\n openBrackets -= 1;\n }\n while (openBraces > 0) {\n fixed += '}';\n openBraces -= 1;\n }\n\n try {\n return JSON.parse(fixed) as unknown;\n } catch {\n return undefined;\n }\n}\n\nfunction createEmptyStreamingState(): StreamingState {\n return {\n messageId: generateId(),\n parts: [],\n activeBlock: null,\n blocks: new Map(),\n currentTextPartIndex: null,\n currentReasoningPartIndex: null,\n currentObjectPartIndex: null,\n accumulatedJson: '',\n activeWorkers: new Map(),\n toolInputBuffers: new Map(),\n };\n}\n\nfunction buildMessageFromState(state: StreamingState, status: 'streaming' | 'done'): UIMessage {\n return {\n id: state.messageId,\n role: 'assistant',\n parts: [...state.parts],\n status,\n createdAt: new Date(),\n };\n}\n\n/**\n * Finalize parts when stream is stopped or errors.\n * Marks streaming parts as done, pending/running tools as cancelled,\n * and recursively finalizes worker parts.\n */\nfunction finalizeParts(parts: UIMessagePart[], workerError?: string): UIMessagePart[] {\n return parts.map((part): UIMessagePart => {\n if (part.type === 'text' || part.type === 'reasoning') {\n if (part.status === 'streaming') {\n return { ...part, status: 'done' };\n }\n }\n if (part.type === 'object' && part.status === 'streaming') {\n return { ...part, status: 'done' };\n }\n if (part.type === 'tool-call') {\n if (part.status === 'pending' || part.status === 'running') {\n return { ...part, status: 'cancelled' };\n }\n }\n if (part.type === 'operation' && part.status === 'running') {\n return { ...part, status: 'cancelled' };\n }\n if (part.type === 'worker' && part.status === 'running') {\n return {\n ...part,\n status: 'error',\n error: workerError,\n parts: finalizeParts(part.parts), // Recursive for nested parts\n };\n }\n return part;\n });\n}\n\n// =============================================================================\n// OctavusChat Class\n// =============================================================================\n\n/**\n * Framework-agnostic chat client for Octavus agents.\n * Manages chat state and streaming, allowing reactive frameworks to subscribe to updates.\n *\n * @example HTTP transport (Next.js, etc.)\n * ```typescript\n * import { OctavusChat, createHttpTransport } from '@octavus/client-sdk';\n *\n * const chat = new OctavusChat({\n * transport: createHttpTransport({\n * request: (payload, options) =>\n * fetch('/api/trigger', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, ...payload }),\n * signal: options?.signal,\n * }),\n * }),\n * });\n * ```\n *\n * @example Socket transport (WebSocket, SockJS, Meteor)\n * ```typescript\n * import { OctavusChat, createSocketTransport } from '@octavus/client-sdk';\n *\n * const chat = new OctavusChat({\n * transport: createSocketTransport({\n * connect: () => new Promise((resolve, reject) => {\n * const ws = new WebSocket(`wss://api.octavus.ai/stream?sessionId=${sessionId}`);\n * ws.onopen = () => resolve(ws);\n * ws.onerror = () => reject(new Error('Connection failed'));\n * }),\n * }),\n * });\n * ```\n */\nexport class OctavusChat {\n // Private state\n private _messages: UIMessage[];\n private _status: ChatStatus = 'idle';\n private _error: OctavusError | null = null;\n private options: OctavusChatOptions;\n private transport: Transport;\n private streamingState: StreamingState | null = null;\n\n // Client tool state\n // Keyed by toolName -> array of pending tools for that name\n private _pendingToolsByName = new Map<string, PendingToolState[]>();\n // Keyed by toolCallId -> pending tool state (for internal lookup when submitting)\n private _pendingToolsByCallId = new Map<string, PendingToolState>();\n // Cache for React useSyncExternalStore compatibility\n private _pendingClientToolsCache: Record<string, InteractiveTool[]> = {};\n private _completedToolResults: ToolResult[] = [];\n private _clientToolAbortController: AbortController | null = null;\n // Server tool results from mixed server+client tools (for continuation)\n private _serverToolResults: ToolResult[] = [];\n // Execution ID for continuation (from client-tool-request event)\n private _pendingExecutionId: string | null = null;\n // Flag indicating automatic client tools have completed and are ready to continue\n // We wait for the finish event before actually continuing to avoid race conditions\n private _readyToContinue = false;\n // Flag indicating the finish event with client-tool-calls reason has been received\n // Used to handle the race condition where finish arrives before async tools complete\n private _finishEventReceived = false;\n\n // Listener sets for reactive frameworks\n private listeners = new Set<Listener>();\n\n constructor(options: OctavusChatOptions) {\n this.options = options;\n this._messages = options.initialMessages ?? [];\n this.transport = options.transport;\n }\n\n /**\n * Update mutable options (callbacks and tool handlers) without recreating the instance.\n * Used by the React hook to keep options fresh across renders, but can also be\n * called directly by non-React consumers.\n *\n * `transport` and `initialMessages` are excluded since they're only consumed at construction time.\n */\n updateOptions(updates: Partial<Omit<OctavusChatOptions, 'transport' | 'initialMessages'>>): void {\n this.options = { ...this.options, ...updates };\n }\n\n // =========================================================================\n // Public Getters\n // =========================================================================\n\n get messages(): UIMessage[] {\n return this._messages;\n }\n\n get status(): ChatStatus {\n return this._status;\n }\n\n /**\n * The current error, if any.\n * Contains structured error information including type, source, and retryability.\n */\n get error(): OctavusError | null {\n return this._error;\n }\n\n /**\n * Pending interactive tool calls keyed by tool name.\n * Each tool has bound `submit()` and `cancel()` methods.\n *\n * @example\n * ```tsx\n * const feedbackTools = pendingClientTools['request-feedback'] ?? [];\n *\n * {feedbackTools.map(tool => (\n * <FeedbackModal\n * key={tool.toolCallId}\n * {...tool.args}\n * onSubmit={(result) => tool.submit(result)}\n * onCancel={() => tool.cancel()}\n * />\n * ))}\n * ```\n */\n get pendingClientTools(): Record<string, InteractiveTool[]> {\n return this._pendingClientToolsCache;\n }\n\n // =========================================================================\n // Subscription Methods (for reactive frameworks)\n // =========================================================================\n\n /**\n * Subscribe to state changes. The callback is called whenever messages, status, or error changes.\n * @returns Unsubscribe function\n */\n subscribe(listener: Listener): () => void {\n this.listeners.add(listener);\n return () => this.listeners.delete(listener);\n }\n\n private notifyListeners(): void {\n this.listeners.forEach((l) => l());\n }\n\n // =========================================================================\n // Private Setters (notify listeners)\n // =========================================================================\n\n private setMessages(messages: UIMessage[]): void {\n this._messages = messages;\n this.notifyListeners();\n }\n\n private setStatus(status: ChatStatus): void {\n this._status = status;\n this.notifyListeners();\n }\n\n private setError(error: OctavusError | null): void {\n this._error = error;\n this.notifyListeners();\n }\n\n private updatePendingClientToolsCache(): void {\n const cache: Record<string, InteractiveTool[]> = {};\n for (const [toolName, tools] of this._pendingToolsByName.entries()) {\n cache[toolName] = tools.map((tool) => ({\n toolCallId: tool.toolCallId,\n toolName: tool.toolName,\n args: tool.args,\n submit: (result: unknown) => this.submitToolResult(tool.toolCallId, result),\n cancel: (reason?: string) =>\n this.submitToolResult(tool.toolCallId, undefined, reason ?? 'User cancelled'),\n }));\n }\n this._pendingClientToolsCache = cache;\n }\n\n // =========================================================================\n // Public Methods\n // =========================================================================\n\n /**\n * Trigger the agent and optionally add a user message to the chat.\n *\n * @param triggerName - The trigger name defined in the agent's protocol.yaml\n * @param input - Input parameters for the trigger (variable substitutions)\n * @param options.userMessage - If provided, adds a user message to the chat before triggering\n *\n * @example Send a text message\n * ```typescript\n * await chat.send('user-message',\n * { USER_MESSAGE: message },\n * { userMessage: { content: message } }\n * );\n * ```\n *\n * @example Send a message with file attachments\n * ```typescript\n * await chat.send('user-message',\n * { USER_MESSAGE: message, FILES: fileRefs },\n * { userMessage: { content: message, files: fileRefs } }\n * );\n * ```\n */\n async send(\n triggerName: string,\n input?: Record<string, unknown>,\n sendOptions?: { userMessage?: UserMessageInput },\n ): Promise<void> {\n this.transport.stop();\n\n let fileRefs: FileReference[] | undefined;\n if (sendOptions?.userMessage?.files) {\n const files = sendOptions.userMessage.files;\n if (isFileReferenceArray(files)) {\n fileRefs = files;\n } else if (this.options.requestUploadUrls) {\n fileRefs = await uploadFiles(files, {\n requestUploadUrls: this.options.requestUploadUrls,\n });\n } else {\n throw new Error(\n 'File upload requires requestUploadUrls option. Either provide FileReference[] or configure requestUploadUrls.',\n );\n }\n }\n\n // Auto-upload FILES in trigger input if needed\n let processedInput = input;\n if (input?.FILES !== undefined && !isFileReferenceArray(input.FILES)) {\n if (this.options.requestUploadUrls) {\n const inputFiles = input.FILES as FileList | File[];\n const uploadedRefs =\n fileRefs ??\n (await uploadFiles(inputFiles, {\n requestUploadUrls: this.options.requestUploadUrls,\n }));\n processedInput = { ...input, FILES: uploadedRefs };\n fileRefs = fileRefs ?? uploadedRefs;\n }\n }\n\n // Optimistic UI: add user message before server responds\n if (sendOptions?.userMessage !== undefined) {\n const userMsg = createUserMessage(sendOptions.userMessage, fileRefs);\n this.setMessages([...this._messages, userMsg]);\n }\n\n this.setStatus('streaming');\n this.setError(null);\n this.streamingState = createEmptyStreamingState();\n\n // Clear any previous client tool state\n this._pendingToolsByName.clear();\n this._pendingToolsByCallId.clear();\n this._completedToolResults = [];\n this._serverToolResults = [];\n this._pendingExecutionId = null;\n this._readyToContinue = false;\n this._finishEventReceived = false;\n this.updatePendingClientToolsCache();\n\n try {\n for await (const event of this.transport.trigger(triggerName, processedInput)) {\n if (this.streamingState === null) break;\n\n this.handleStreamEvent(event, this.streamingState);\n }\n } catch (err) {\n // Convert unknown errors to OctavusError\n const errorObj = OctavusError.isInstance(err)\n ? err\n : new OctavusError({\n errorType: 'internal_error',\n message: err instanceof Error ? err.message : 'Unknown error',\n source: 'client',\n retryable: false,\n cause: err,\n });\n\n // Finalize any streaming message before setting error state\n const state = this.streamingState;\n if (state !== null) {\n const messages = [...this._messages];\n const lastMsg = messages[messages.length - 1];\n\n if (state.parts.length > 0) {\n const finalParts = finalizeParts(state.parts, 'Stream error');\n\n const finalMessage: UIMessage = {\n id: state.messageId,\n role: 'assistant',\n parts: finalParts,\n status: 'done',\n createdAt: new Date(),\n };\n\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = finalMessage;\n } else {\n messages.push(finalMessage);\n }\n this.setMessages(messages);\n } else if (lastMsg?.id === state.messageId) {\n // No parts yet - remove the empty streaming message\n messages.pop();\n this.setMessages(messages);\n }\n }\n\n this.setError(errorObj);\n this.setStatus('error');\n this.streamingState = null;\n this.options.onError?.(errorObj);\n }\n }\n\n /**\n * Upload files directly without sending a message.\n * Useful for showing upload progress before sending.\n *\n * @param files - Files to upload\n * @param onProgress - Optional progress callback\n * @returns Array of file references\n *\n * @example\n * ```typescript\n * const fileRefs = await chat.uploadFiles(fileInput.files, (i, progress) => {\n * console.log(`File ${i}: ${progress}%`);\n * });\n * // Later...\n * await chat.send('user-message', { FILES: fileRefs }, { userMessage: { files: fileRefs } });\n * ```\n */\n async uploadFiles(\n files: FileList | File[],\n onProgress?: (fileIndex: number, progress: number) => void,\n ): Promise<FileReference[]> {\n if (!this.options.requestUploadUrls) {\n throw new Error('File upload requires requestUploadUrls option');\n }\n return await uploadFiles(files, {\n requestUploadUrls: this.options.requestUploadUrls,\n onProgress,\n });\n }\n\n /**\n * Internal: Submit a result for a pending tool.\n * Called by bound submit/cancel methods on InteractiveTool.\n */\n private submitToolResult(toolCallId: string, result?: unknown, error?: string): void {\n const pendingTool = this._pendingToolsByCallId.get(toolCallId);\n if (!pendingTool) {\n // Tool not found - may have been cancelled or already resolved\n return;\n }\n\n // Remove from both maps\n this._pendingToolsByCallId.delete(toolCallId);\n const toolsForName = this._pendingToolsByName.get(pendingTool.toolName);\n if (toolsForName) {\n const filtered = toolsForName.filter((t) => t.toolCallId !== toolCallId);\n if (filtered.length === 0) {\n this._pendingToolsByName.delete(pendingTool.toolName);\n } else {\n this._pendingToolsByName.set(pendingTool.toolName, filtered);\n }\n }\n this.updatePendingClientToolsCache();\n\n const toolResult: ToolResult = {\n toolCallId,\n toolName: pendingTool.toolName,\n result: error ? undefined : result,\n error,\n outputVariable: pendingTool.outputVariable,\n blockIndex: pendingTool.blockIndex,\n thread: pendingTool.thread,\n workerId: pendingTool.workerId,\n };\n this._completedToolResults.push(toolResult);\n\n if (error) {\n this.emitToolOutputError(toolCallId, error);\n } else {\n this.emitToolOutputAvailable(toolCallId, result);\n }\n\n if (this._pendingToolsByCallId.size === 0) {\n void this.continueWithClientToolResults();\n }\n\n this.notifyListeners();\n }\n\n /** Stop the current streaming and finalize any partial message */\n stop(): void {\n if (this._status !== 'streaming' && this._status !== 'awaiting-input') {\n return;\n }\n\n this._clientToolAbortController?.abort();\n this._clientToolAbortController = null;\n this._pendingToolsByName.clear();\n this._pendingToolsByCallId.clear();\n this._completedToolResults = [];\n this._serverToolResults = [];\n this._pendingExecutionId = null;\n this._readyToContinue = false;\n this._finishEventReceived = false;\n this.updatePendingClientToolsCache();\n\n this.transport.stop();\n\n const state = this.streamingState;\n if (state && state.parts.length > 0) {\n const finalParts = finalizeParts(state.parts, 'Stopped by user');\n\n const finalMessage: UIMessage = {\n id: state.messageId,\n role: 'assistant',\n parts: finalParts,\n status: 'done',\n createdAt: new Date(),\n };\n\n const messages = [...this._messages];\n const lastMsg = messages[messages.length - 1];\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = finalMessage;\n } else {\n messages.push(finalMessage);\n }\n this.setMessages(messages);\n }\n\n this.streamingState = null;\n this.setStatus('idle');\n this.options.onStop?.();\n }\n\n // =========================================================================\n // Private Helpers\n // =========================================================================\n\n private handleStreamEvent(event: StreamEvent, state: StreamingState): void {\n switch (event.type) {\n case 'start':\n // Call onStart callback with execution/session ID\n if (event.executionId) {\n this.options.onStart?.(event.executionId);\n }\n break;\n\n case 'block-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n const block: BlockState = {\n blockId: event.blockId,\n blockName: event.blockName,\n blockType: event.blockType,\n display: event.display,\n description: event.description,\n outputToChat: event.outputToChat ?? true,\n thread: event.thread,\n reasoning: '',\n text: '',\n toolCalls: new Map(),\n };\n state.blocks.set(event.blockId, block);\n state.activeBlock = block;\n\n const isOperation = OPERATION_BLOCK_TYPES.has(event.blockType);\n const isHidden = event.display === 'hidden';\n if (isOperation && !isHidden) {\n const thread = event.thread;\n const operationPart: UIOperationPart = {\n type: 'operation',\n operationId: event.blockId,\n name: event.description ?? event.blockName,\n operationType: event.blockType,\n status: 'running',\n thread: threadForPart(thread),\n };\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n workerPart.parts.push(operationPart);\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n state.parts.push(operationPart);\n }\n }\n\n state.currentTextPartIndex = null;\n state.currentReasoningPartIndex = null;\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'block-end': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Find operation in worker's nested parts\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const operationPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'operation' && p.operationId === event.blockId,\n );\n if (operationPartIndex >= 0) {\n const part = workerPart.parts[operationPartIndex] as UIOperationPart;\n workerPart.parts[operationPartIndex] = { ...part, status: 'done' };\n state.parts[workerState.partIndex] = { ...workerPart };\n }\n } else {\n const operationPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'operation' && p.operationId === event.blockId,\n );\n if (operationPartIndex >= 0) {\n const part = state.parts[operationPartIndex] as UIOperationPart;\n state.parts[operationPartIndex] = { ...part, status: 'done' };\n }\n }\n\n if (state.activeBlock?.blockId === event.blockId) {\n state.activeBlock = null;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'reasoning-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n const reasoningPart: UIReasoningPart = {\n type: 'reasoning',\n text: '',\n status: 'streaming',\n thread: threadForPart(state.activeBlock?.thread),\n };\n\n if (workerState) {\n // Add to worker's nested parts\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n workerPart.parts.push(reasoningPart);\n workerState.currentReasoningPartIndex = workerPart.parts.length - 1;\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n state.parts.push(reasoningPart);\n state.currentReasoningPartIndex = state.parts.length - 1;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'reasoning-delta': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Update worker's reasoning part\n if (workerState.currentReasoningPartIndex !== null) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const part = workerPart.parts[workerState.currentReasoningPartIndex] as UIReasoningPart;\n part.text += event.delta;\n workerPart.parts[workerState.currentReasoningPartIndex] = { ...part };\n state.parts[workerState.partIndex] = { ...workerPart };\n }\n } else {\n if (state.currentReasoningPartIndex !== null) {\n const part = state.parts[state.currentReasoningPartIndex] as UIReasoningPart;\n part.text += event.delta;\n state.parts[state.currentReasoningPartIndex] = { ...part };\n }\n\n if (state.activeBlock) {\n state.activeBlock.reasoning += event.delta;\n }\n }\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'reasoning-end': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Finalize worker's reasoning part\n if (workerState.currentReasoningPartIndex !== null) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const part = workerPart.parts[workerState.currentReasoningPartIndex] as UIReasoningPart;\n part.status = 'done';\n workerPart.parts[workerState.currentReasoningPartIndex] = { ...part };\n state.parts[workerState.partIndex] = { ...workerPart };\n workerState.currentReasoningPartIndex = null;\n }\n } else if (state.currentReasoningPartIndex !== null) {\n const part = state.parts[state.currentReasoningPartIndex] as UIReasoningPart;\n part.status = 'done';\n state.parts[state.currentReasoningPartIndex] = { ...part };\n state.currentReasoningPartIndex = null;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'text-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n const thread = threadForPart(state.activeBlock?.thread);\n const shouldAddPart = state.activeBlock?.outputToChat !== false || thread !== undefined;\n\n // For worker events, always add parts\n if (workerState || shouldAddPart) {\n // Structured output mode: accumulate JSON and parse progressively\n if (event.responseType) {\n const objectPart: UIObjectPart = {\n type: 'object',\n id: event.id,\n typeName: event.responseType,\n partial: undefined,\n object: undefined,\n status: 'streaming',\n thread,\n };\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n workerPart.parts.push(objectPart);\n workerState.currentObjectPartIndex = workerPart.parts.length - 1;\n workerState.accumulatedJson = '';\n workerState.currentTextPartIndex = null;\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n state.parts.push(objectPart);\n state.currentObjectPartIndex = state.parts.length - 1;\n state.accumulatedJson = '';\n state.currentTextPartIndex = null;\n }\n } else {\n const textPart: UITextPart = {\n type: 'text',\n text: '',\n status: 'streaming',\n thread,\n };\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n workerPart.parts.push(textPart);\n workerState.currentTextPartIndex = workerPart.parts.length - 1;\n workerState.currentObjectPartIndex = null;\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n state.parts.push(textPart);\n state.currentTextPartIndex = state.parts.length - 1;\n state.currentObjectPartIndex = null;\n }\n }\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'text-delta': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Update worker's text or object part\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n if (workerState.currentObjectPartIndex !== null) {\n workerState.accumulatedJson += event.delta;\n const part = workerPart.parts[workerState.currentObjectPartIndex] as UIObjectPart;\n const parsed = parsePartialJson(workerState.accumulatedJson);\n if (parsed !== undefined) {\n part.partial = parsed;\n workerPart.parts[workerState.currentObjectPartIndex] = { ...part };\n }\n } else if (workerState.currentTextPartIndex !== null) {\n const part = workerPart.parts[workerState.currentTextPartIndex] as UITextPart;\n part.text += event.delta;\n workerPart.parts[workerState.currentTextPartIndex] = { ...part };\n }\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n if (state.currentObjectPartIndex !== null) {\n state.accumulatedJson += event.delta;\n const part = state.parts[state.currentObjectPartIndex] as UIObjectPart;\n const parsed = parsePartialJson(state.accumulatedJson);\n if (parsed !== undefined) {\n part.partial = parsed;\n state.parts[state.currentObjectPartIndex] = { ...part };\n }\n } else if (state.currentTextPartIndex !== null) {\n const part = state.parts[state.currentTextPartIndex] as UITextPart;\n part.text += event.delta;\n state.parts[state.currentTextPartIndex] = { ...part };\n }\n\n if (state.activeBlock) {\n state.activeBlock.text += event.delta;\n }\n }\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'text-end': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Finalize worker's text or object part\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n if (workerState.currentObjectPartIndex !== null) {\n const part = workerPart.parts[workerState.currentObjectPartIndex] as UIObjectPart;\n try {\n const finalObject = JSON.parse(workerState.accumulatedJson) as unknown;\n part.object = finalObject;\n part.partial = finalObject;\n part.status = 'done';\n } catch {\n part.status = 'error';\n part.error = 'Failed to parse response as JSON';\n }\n workerPart.parts[workerState.currentObjectPartIndex] = { ...part };\n workerState.currentObjectPartIndex = null;\n workerState.accumulatedJson = '';\n } else if (workerState.currentTextPartIndex !== null) {\n const part = workerPart.parts[workerState.currentTextPartIndex] as UITextPart;\n part.status = 'done';\n workerPart.parts[workerState.currentTextPartIndex] = { ...part };\n workerState.currentTextPartIndex = null;\n }\n state.parts[workerState.partIndex] = { ...workerPart };\n } else if (state.currentObjectPartIndex !== null) {\n const part = state.parts[state.currentObjectPartIndex] as UIObjectPart;\n try {\n const finalObject = JSON.parse(state.accumulatedJson) as unknown;\n part.object = finalObject;\n part.partial = finalObject;\n part.status = 'done';\n } catch {\n part.status = 'error';\n part.error = 'Failed to parse response as JSON';\n }\n state.parts[state.currentObjectPartIndex] = { ...part };\n state.currentObjectPartIndex = null;\n state.accumulatedJson = '';\n } else if (state.currentTextPartIndex !== null) {\n const part = state.parts[state.currentTextPartIndex] as UITextPart;\n part.status = 'done';\n state.parts[state.currentTextPartIndex] = { ...part };\n state.currentTextPartIndex = null;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'tool-input-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n const toolPart: UIToolCallPart = {\n type: 'tool-call',\n toolCallId: event.toolCallId,\n toolName: event.toolName,\n displayName: event.title,\n args: {},\n status: 'pending',\n thread: threadForPart(state.activeBlock?.thread),\n };\n\n // Initialize the input buffer for this tool call\n if (workerState) {\n workerState.toolInputBuffers.set(event.toolCallId, '');\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n workerPart.parts.push(toolPart);\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n state.toolInputBuffers.set(event.toolCallId, '');\n state.parts.push(toolPart);\n\n if (state.activeBlock) {\n state.activeBlock.toolCalls.set(event.toolCallId, toolPart);\n }\n }\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'tool-input-delta': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Accumulate the delta into the worker's buffer\n const existing = workerState.toolInputBuffers.get(event.toolCallId) ?? '';\n const accumulated = existing + event.inputTextDelta;\n workerState.toolInputBuffers.set(event.toolCallId, accumulated);\n\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const toolPart = workerPart.parts[toolPartIndex] as UIToolCallPart;\n const parsed = parsePartialJson(accumulated);\n if (parsed !== undefined) {\n toolPart.args = parsed as Record<string, unknown>;\n workerPart.parts[toolPartIndex] = { ...toolPart };\n state.parts[workerState.partIndex] = { ...workerPart };\n this.updateStreamingMessage();\n }\n }\n } else {\n // Accumulate the delta into the top-level buffer\n const existing = state.toolInputBuffers.get(event.toolCallId) ?? '';\n const accumulated = existing + event.inputTextDelta;\n state.toolInputBuffers.set(event.toolCallId, accumulated);\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const toolPart = state.parts[toolPartIndex] as UIToolCallPart;\n const parsed = parsePartialJson(accumulated);\n if (parsed !== undefined) {\n toolPart.args = parsed as Record<string, unknown>;\n state.parts[toolPartIndex] = { ...toolPart };\n this.updateStreamingMessage();\n }\n }\n }\n break;\n }\n\n case 'tool-input-end':\n // Input streaming ended, wait for tool-input-available\n break;\n\n case 'tool-input-available': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Clean up the worker buffer\n workerState.toolInputBuffers.delete(event.toolCallId);\n\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = workerPart.parts[toolPartIndex] as UIToolCallPart;\n part.args = event.input as Record<string, unknown>;\n part.status = 'running';\n workerPart.parts[toolPartIndex] = { ...part };\n state.parts[workerState.partIndex] = { ...workerPart };\n this.updateStreamingMessage();\n }\n } else {\n // Clean up the top-level buffer\n state.toolInputBuffers.delete(event.toolCallId);\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n part.args = event.input as Record<string, unknown>;\n part.status = 'running';\n state.parts[toolPartIndex] = { ...part };\n this.updateStreamingMessage();\n }\n }\n break;\n }\n\n case 'tool-output-available': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = workerPart.parts[toolPartIndex] as UIToolCallPart;\n part.result = event.output;\n part.status = 'done';\n workerPart.parts[toolPartIndex] = { ...part };\n state.parts[workerState.partIndex] = { ...workerPart };\n this.updateStreamingMessage();\n }\n } else {\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n part.result = event.output;\n part.status = 'done';\n state.parts[toolPartIndex] = { ...part };\n this.updateStreamingMessage();\n }\n }\n break;\n }\n\n case 'tool-output-error': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = workerPart.parts[toolPartIndex] as UIToolCallPart;\n part.error = event.error;\n part.status = 'error';\n workerPart.parts[toolPartIndex] = { ...part };\n state.parts[workerState.partIndex] = { ...workerPart };\n this.updateStreamingMessage();\n }\n } else {\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n part.error = event.error;\n part.status = 'error';\n state.parts[toolPartIndex] = { ...part };\n this.updateStreamingMessage();\n }\n }\n break;\n }\n\n case 'source': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n const thread = threadForPart(state.activeBlock?.thread);\n\n let sourcePart: UISourcePart;\n if (event.sourceType === 'url') {\n sourcePart = {\n type: 'source',\n sourceType: 'url',\n id: event.id,\n url: event.url,\n title: event.title,\n thread,\n };\n } else {\n sourcePart = {\n type: 'source',\n sourceType: 'document',\n id: event.id,\n mediaType: event.mediaType,\n title: event.title,\n filename: event.filename,\n thread,\n };\n }\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n workerPart.parts.push(sourcePart);\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n state.parts.push(sourcePart);\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'file-available': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n // Add generated file as a part\n const filePart: UIFilePart = {\n type: 'file',\n id: event.id,\n mediaType: event.mediaType,\n url: event.url,\n filename: event.filename,\n size: event.size,\n toolCallId: event.toolCallId,\n thread: threadForPart(state.activeBlock?.thread),\n };\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n workerPart.parts.push(filePart);\n state.parts[workerState.partIndex] = { ...workerPart };\n } else {\n state.parts.push(filePart);\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'resource-update':\n this.options.onResourceUpdate?.(event.name, event.value);\n break;\n\n case 'worker-start': {\n // Check if worker with same workerId already exists (for continuations)\n const existingIndex = state.parts.findIndex(\n (p) => p.type === 'worker' && p.workerId === event.workerId,\n );\n\n let partIndex: number;\n if (existingIndex !== -1) {\n // Re-use existing worker part (continuation)\n const existingPart = state.parts[existingIndex] as UIWorkerPart;\n existingPart.status = 'running';\n partIndex = existingIndex;\n } else {\n // Create a new worker part\n const workerPart: UIWorkerPart = {\n type: 'worker',\n workerId: event.workerId,\n workerSlug: event.workerSlug,\n description: event.description,\n parts: [],\n status: 'running',\n };\n state.parts.push(workerPart);\n partIndex = state.parts.length - 1;\n }\n\n // Track the worker for event routing\n const workerState: WorkerPartState = {\n partIndex,\n currentTextPartIndex: null,\n currentReasoningPartIndex: null,\n currentObjectPartIndex: null,\n accumulatedJson: '',\n toolInputBuffers: new Map(),\n };\n state.activeWorkers.set(event.workerId, workerState);\n this.updateStreamingMessage();\n break;\n }\n\n case 'worker-result': {\n const workerState = state.activeWorkers.get(event.workerId);\n if (workerState !== undefined) {\n const part = state.parts[workerState.partIndex] as UIWorkerPart;\n part.output = event.output;\n part.error = event.error;\n part.status = event.error ? 'error' : 'done';\n\n // Finalize any streaming parts in the worker\n part.parts = part.parts.map((p): UIMessagePart => {\n if (p.type === 'text' || p.type === 'reasoning') {\n if (p.status === 'streaming') {\n return { ...p, status: 'done' };\n }\n }\n if (p.type === 'object' && p.status === 'streaming') {\n return { ...p, status: 'done' };\n }\n return p;\n });\n\n state.parts[workerState.partIndex] = { ...part };\n state.activeWorkers.delete(event.workerId);\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'finish': {\n // Handle client-tool-calls finish reason\n if (event.finishReason === 'client-tool-calls') {\n // Mark that finish event has been received (for async tools that complete later)\n this._finishEventReceived = true;\n // Don't finalize message - we're waiting for client tools\n if (this._pendingToolsByCallId.size > 0) {\n this.setStatus('awaiting-input');\n } else if (this._readyToContinue) {\n // Automatic tools completed before finish event arrived - continue now\n this._readyToContinue = false;\n this._finishEventReceived = false;\n void this.continueWithClientToolResults();\n }\n return;\n }\n\n const finalMessage = buildMessageFromState(state, 'done');\n\n finalMessage.parts = finalMessage.parts.map((part) => {\n if (part.type === 'text' || part.type === 'reasoning') {\n return { ...part, status: 'done' as const };\n }\n if (part.type === 'object' && part.status === 'streaming') {\n return { ...part, status: 'done' as const };\n }\n return part;\n });\n\n if (finalMessage.parts.length > 0) {\n const messages = [...this._messages];\n const lastMsg = messages[messages.length - 1];\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = finalMessage;\n } else {\n messages.push(finalMessage);\n }\n this.setMessages(messages);\n }\n\n this.setStatus('idle');\n this.streamingState = null;\n this.options.onFinish?.();\n break;\n }\n\n case 'error': {\n // Create structured error from the error event\n throw new OctavusError({\n errorType: event.errorType,\n message: event.message,\n source: event.source,\n retryable: event.retryable,\n retryAfter: event.retryAfter,\n code: event.code,\n provider: event.provider,\n tool: event.tool,\n });\n }\n\n case 'tool-request':\n // Handled by server-sdk, not relevant for UI\n break;\n\n case 'client-tool-request':\n // Store execution ID and server tool results for continuation\n this._pendingExecutionId = event.executionId;\n this._serverToolResults = event.serverToolResults ?? [];\n // Handle client-side tool execution\n void this.handleClientToolRequest(event.toolCalls, state);\n break;\n }\n }\n\n private updateStreamingMessage(): void {\n const state = this.streamingState;\n if (!state) return;\n\n const msg = buildMessageFromState(state, 'streaming');\n const messages = [...this._messages];\n\n const lastMsg = messages[messages.length - 1];\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = msg;\n } else {\n messages.push(msg);\n }\n\n this.setMessages(messages);\n }\n\n /**\n * Emit a tool-output-available event for a client tool result.\n */\n private emitToolOutputAvailable(toolCallId: string, output: unknown): void {\n const state = this.streamingState;\n if (!state) return;\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n part.result = output;\n part.status = 'done';\n state.parts[toolPartIndex] = { ...part };\n this.updateStreamingMessage();\n }\n }\n\n /**\n * Emit a tool-output-error event for a client tool result.\n */\n private emitToolOutputError(toolCallId: string, error: string): void {\n const state = this.streamingState;\n if (!state) return;\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n part.error = error;\n part.status = 'error';\n state.parts[toolPartIndex] = { ...part };\n this.updateStreamingMessage();\n }\n }\n\n /**\n * Continue execution with collected client tool results.\n */\n private async continueWithClientToolResults(): Promise<void> {\n if (this._completedToolResults.length === 0) return;\n\n if (this._pendingExecutionId === null) {\n // Context lost - this shouldn't happen, but handle gracefully\n const errorObj = new OctavusError({\n errorType: 'internal_error',\n message: 'Cannot continue execution: execution ID was lost.',\n source: 'client',\n retryable: false,\n });\n this.setError(errorObj);\n this.setStatus('error');\n this.options.onError?.(errorObj);\n return;\n }\n\n // Combine server results (from mixed tools scenario) with client results\n const allResults = [...this._serverToolResults, ...this._completedToolResults];\n const executionId = this._pendingExecutionId;\n this._serverToolResults = [];\n this._completedToolResults = [];\n this._pendingExecutionId = null;\n\n this.setStatus('streaming');\n\n try {\n // Use the transport's continuation method (works for both HTTP and Socket)\n for await (const event of this.transport.continueWithToolResults(executionId, allResults)) {\n if (this.streamingState === null) break;\n this.handleStreamEvent(event, this.streamingState);\n }\n } catch (err) {\n const errorObj = OctavusError.isInstance(err)\n ? err\n : new OctavusError({\n errorType: 'internal_error',\n message: err instanceof Error ? err.message : 'Unknown error',\n source: 'client',\n retryable: false,\n cause: err,\n });\n\n this.setError(errorObj);\n this.setStatus('error');\n this.streamingState = null;\n this.options.onError?.(errorObj);\n }\n }\n\n /**\n * Handle client tool request event.\n *\n * IMPORTANT: Interactive tools must be registered synchronously (before any await)\n * to avoid a race condition where the finish event is processed before tools are added.\n */\n private async handleClientToolRequest(\n toolCalls: PendingToolCall[],\n state: StreamingState,\n ): Promise<void> {\n this._clientToolAbortController = new AbortController();\n\n // FIRST PASS: Register all interactive tools synchronously (no await)\n // This ensures pending tools are populated before finish event is processed\n for (const tc of toolCalls) {\n const handler = this.options.clientTools?.[tc.toolName];\n if (handler === 'interactive') {\n const toolState: PendingToolState = {\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n args: tc.args,\n source: tc.source,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n };\n // Add to both maps\n this._pendingToolsByCallId.set(tc.toolCallId, toolState);\n const existing = this._pendingToolsByName.get(tc.toolName) ?? [];\n this._pendingToolsByName.set(tc.toolName, [...existing, toolState]);\n }\n }\n if (this._pendingToolsByCallId.size > 0) {\n this.updatePendingClientToolsCache();\n }\n\n // SECOND PASS: Execute automatic handlers and handle missing handlers\n for (const tc of toolCalls) {\n const handler = this.options.clientTools?.[tc.toolName];\n\n if (handler === 'interactive') {\n // Already registered above, just update UI state\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === tc.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n // Keep running status - user will see the tool is \"executing\" while modal is open\n state.parts[toolPartIndex] = { ...part };\n }\n } else if (handler) {\n try {\n const collectedFiles: FileReference[] = [];\n const result = await handler(tc.args, {\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n signal: this._clientToolAbortController.signal,\n addFile: (file) => collectedFiles.push(file),\n });\n\n this._completedToolResults.push({\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n result,\n files: collectedFiles.length > 0 ? collectedFiles : undefined,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n });\n\n this.emitToolOutputAvailable(tc.toolCallId, result);\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : 'Tool execution failed';\n this._completedToolResults.push({\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n error: errorMessage,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n });\n\n this.emitToolOutputError(tc.toolCallId, errorMessage);\n }\n } else {\n // No handler registered - treat as error\n const errorMessage = `No client handler for tool: ${tc.toolName}`;\n this._completedToolResults.push({\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n error: errorMessage,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n });\n\n this.emitToolOutputError(tc.toolCallId, errorMessage);\n }\n }\n\n // If no interactive tools, mark as ready to continue.\n // We wait for the finish event to arrive first to avoid a race condition where\n // the finish event gets delivered to the continuation's event resolver.\n if (this._pendingToolsByCallId.size === 0 && this._completedToolResults.length > 0) {\n this._readyToContinue = true;\n\n // If finish event already arrived while we were executing async tools,\n // trigger continuation now instead of waiting forever\n if (this._finishEventReceived) {\n this._readyToContinue = false;\n this._finishEventReceived = false;\n void this.continueWithClientToolResults();\n }\n }\n }\n}\n","import type { FileReference } from '@octavus/core';\n\n/**\n * Response from the upload URLs endpoint\n */\nexport interface UploadUrlsResponse {\n files: {\n id: string;\n uploadUrl: string;\n downloadUrl: string;\n }[];\n}\n\n/**\n * Options for uploading files\n */\nexport interface UploadFilesOptions {\n /**\n * Function to request upload URLs from the platform.\n * Consumer apps must implement this to authenticate with the platform.\n *\n * @param files - Array of file metadata to request URLs for\n * @returns Response with presigned upload and download URLs\n *\n * @example\n * ```typescript\n * requestUploadUrls: async (files) => {\n * const response = await fetch('/api/upload-urls', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, files }),\n * });\n * return response.json();\n * }\n * ```\n */\n requestUploadUrls: (\n files: { filename: string; mediaType: string; size: number }[],\n ) => Promise<UploadUrlsResponse>;\n\n /**\n * Callback for upload progress (0-100 per file).\n * Called multiple times during upload with real-time progress.\n *\n * @param fileIndex - Index of the file being uploaded\n * @param progress - Progress percentage (0-100)\n */\n onProgress?: (fileIndex: number, progress: number) => void;\n}\n\n/**\n * Upload a single file to S3 with progress tracking.\n * Uses XMLHttpRequest for upload progress events (fetch doesn't support this).\n */\nfunction uploadFileWithProgress(\n url: string,\n file: File,\n onProgress?: (progress: number) => void,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n const xhr = new XMLHttpRequest();\n\n xhr.upload.addEventListener('progress', (event) => {\n if (event.lengthComputable) {\n const progress = Math.round((event.loaded / event.total) * 100);\n onProgress?.(progress);\n }\n });\n\n xhr.addEventListener('load', () => {\n if (xhr.status >= 200 && xhr.status < 300) {\n resolve();\n } else {\n reject(new Error(`Upload failed with status ${xhr.status}`));\n }\n });\n\n xhr.addEventListener('error', () => {\n reject(new Error('Upload failed: network error'));\n });\n\n xhr.addEventListener('abort', () => {\n reject(new Error('Upload aborted'));\n });\n\n xhr.open('PUT', url);\n xhr.setRequestHeader('Content-Type', file.type || 'application/octet-stream');\n xhr.send(file);\n });\n}\n\n/**\n * Upload files to the Octavus platform.\n *\n * This function:\n * 1. Requests presigned upload URLs from the platform\n * 2. Uploads each file directly to S3 with progress tracking\n * 3. Returns file references that can be used in trigger input\n *\n * @param files - Files to upload (from file input or drag/drop)\n * @param options - Upload configuration\n * @returns Array of file references with download URLs\n *\n * @example\n * ```typescript\n * const fileRefs = await uploadFiles(fileInputRef.current.files, {\n * requestUploadUrls: async (files) => {\n * const response = await fetch('/api/upload-urls', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, files }),\n * });\n * return response.json();\n * },\n * onProgress: (fileIndex, progress) => {\n * console.log(`File ${fileIndex}: ${progress}%`);\n * },\n * });\n * ```\n */\nexport async function uploadFiles(\n files: FileList | File[],\n options: UploadFilesOptions,\n): Promise<FileReference[]> {\n const fileArray = Array.from(files);\n\n if (fileArray.length === 0) {\n return [];\n }\n\n const { files: uploadInfos } = await options.requestUploadUrls(\n fileArray.map((f) => ({\n filename: f.name,\n mediaType: f.type || 'application/octet-stream',\n size: f.size,\n })),\n );\n\n const references: FileReference[] = [];\n\n for (let i = 0; i < fileArray.length; i++) {\n const file = fileArray[i]!;\n const uploadInfo = uploadInfos[i]!;\n\n await uploadFileWithProgress(uploadInfo.uploadUrl, file, (progress) => {\n options.onProgress?.(i, progress);\n });\n\n references.push({\n id: uploadInfo.id,\n mediaType: file.type || 'application/octet-stream',\n url: uploadInfo.downloadUrl,\n filename: file.name,\n size: file.size,\n });\n }\n\n return references;\n}\n","import { safeParseStreamEvent, isAbortError, type StreamEvent } from '@octavus/core';\n\n/**\n * Parse SSE stream events.\n *\n * @param response - The HTTP response with SSE body\n * @param signal - Optional abort signal to cancel reading\n */\nexport async function* parseSSEStream(\n response: Response,\n signal?: AbortSignal,\n): AsyncGenerator<StreamEvent, void, unknown> {\n const reader = response.body?.getReader();\n if (!reader) {\n throw new Error('Response body is not readable');\n }\n\n const decoder = new TextDecoder();\n let buffer = '';\n\n try {\n let reading = true;\n while (reading) {\n // Check if aborted before reading\n if (signal?.aborted) {\n return;\n }\n\n let readResult: ReadableStreamReadResult<Uint8Array>;\n try {\n readResult = await reader.read();\n } catch (err) {\n // Handle abort errors gracefully - exit without throwing\n if (isAbortError(err)) {\n return;\n }\n throw err;\n }\n\n const { done, value } = readResult;\n\n if (done) {\n reading = false;\n continue;\n }\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() ?? '';\n\n for (const line of lines) {\n if (line.startsWith('data: ') && line !== 'data: [DONE]') {\n try {\n const parsed = safeParseStreamEvent(JSON.parse(line.slice(6)));\n if (parsed.success) {\n yield parsed.data;\n }\n // Skip malformed events silently\n } catch {\n // Skip malformed JSON - no logging in production\n }\n }\n }\n }\n } finally {\n reader.releaseLock();\n }\n}\n","import type { StreamEvent, ToolResult } from '@octavus/core';\n\n// =============================================================================\n// Base Transport Interface\n// =============================================================================\n\n/**\n * Transport interface for delivering events from server to client.\n *\n * Abstracts the connection mechanism (HTTP/SSE or WebSocket) behind a unified\n * async iterator interface. Use `createHttpTransport` or `createSocketTransport`\n * to create an implementation.\n */\nexport interface Transport {\n /**\n * Trigger the agent and stream events.\n * @param triggerName - The trigger name defined in the agent's protocol\n * @param input - Input parameters for variable substitution\n */\n trigger(triggerName: string, input?: Record<string, unknown>): AsyncIterable<StreamEvent>;\n\n /**\n * Continue execution with tool results after client-side tool handling.\n *\n * @param executionId - The execution ID from the client-tool-request event\n * @param results - All tool results (server + client) to send\n */\n continueWithToolResults(executionId: string, results: ToolResult[]): AsyncIterable<StreamEvent>;\n\n /** Stop the current stream. Safe to call when no stream is active. */\n stop(): void;\n}\n\n// =============================================================================\n// Socket Transport (extends Transport with connection management)\n// =============================================================================\n\n/**\n * Connection states for socket transport.\n *\n * - `disconnected`: Not connected (initial state or after disconnect)\n * - `connecting`: Connection attempt in progress\n * - `connected`: Successfully connected and ready\n * - `error`: Connection failed (check error in listener callback)\n */\nexport type ConnectionState = 'disconnected' | 'connecting' | 'connected' | 'error';\n\n/**\n * Callback for connection state changes.\n */\nexport type ConnectionStateListener = (state: ConnectionState, error?: Error) => void;\n\n/**\n * Socket transport with connection management capabilities.\n *\n * Extends the base Transport interface with methods for managing persistent\n * WebSocket/SockJS connections. Use this when you need:\n * - Eager connection (connect before first message)\n * - Connection status UI indicators\n * - Manual connection lifecycle control\n *\n * Created via `createSocketTransport()`.\n */\nexport interface SocketTransport extends Transport {\n /**\n * Current connection state.\n *\n * - `disconnected`: Not connected (initial state)\n * - `connecting`: Connection in progress\n * - `connected`: Ready to send/receive\n * - `error`: Connection failed\n */\n readonly connectionState: ConnectionState;\n\n /**\n * Subscribe to connection state changes.\n *\n * The listener is called immediately with the current state, then again\n * whenever the state changes.\n *\n * @param listener - Callback invoked on state changes\n * @returns Unsubscribe function\n *\n * @example\n * ```typescript\n * const unsubscribe = transport.onConnectionStateChange((state, error) => {\n * setConnectionState(state);\n * if (error) setConnectionError(error);\n * });\n * ```\n */\n onConnectionStateChange(listener: ConnectionStateListener): () => void;\n\n /**\n * Eagerly establish the connection.\n *\n * By default, socket transport connects lazily on first `trigger()`. Call\n * this method to establish the connection early (e.g., on component mount):\n * - Faster first message response\n * - Show accurate connection status in UI\n * - Handle connection errors before user interaction\n *\n * Safe to call multiple times - resolves immediately if already connected.\n *\n * @example\n * ```tsx\n * useEffect(() => {\n * transport.connect()\n * .then(() => console.log('Connected'))\n * .catch((err) => console.error('Failed:', err));\n *\n * return () => transport.disconnect();\n * }, [transport]);\n * ```\n */\n connect(): Promise<void>;\n\n /**\n * Close the connection and clean up resources.\n *\n * The transport will reconnect automatically on next `trigger()` call.\n */\n disconnect(): void;\n}\n\n// =============================================================================\n// Type Guards\n// =============================================================================\n\n/**\n * Check if a transport is a SocketTransport with connection management.\n *\n * @example\n * ```typescript\n * if (isSocketTransport(transport)) {\n * transport.connect(); // TypeScript knows this is available\n * }\n * ```\n */\nexport function isSocketTransport(transport: Transport): transport is SocketTransport {\n return (\n 'connect' in transport &&\n 'disconnect' in transport &&\n 'connectionState' in transport &&\n 'onConnectionStateChange' in transport\n );\n}\n","import { isAbortError, type ToolResult } from '@octavus/core';\nimport { parseSSEStream } from '@/stream/reader';\nimport type { Transport } from './types';\n\n// =============================================================================\n// Request Types\n// =============================================================================\n\n/** Start a new trigger execution */\nexport interface TriggerRequest {\n type: 'trigger';\n triggerName: string;\n input?: Record<string, unknown>;\n}\n\n/** Continue execution after client-side tool handling */\nexport interface ContinueRequest {\n type: 'continue';\n executionId: string;\n toolResults: ToolResult[];\n}\n\n/** All request types supported by the HTTP transport */\nexport type HttpRequest = TriggerRequest | ContinueRequest;\n\n// =============================================================================\n// Transport Options\n// =============================================================================\n\n/** Request options passed to the request callback */\nexport interface HttpRequestOptions {\n /** Abort signal to cancel the request */\n signal?: AbortSignal;\n}\n\n/**\n * Options for creating an HTTP transport.\n */\nexport interface HttpTransportOptions {\n /**\n * Function to make requests to your backend.\n * Receives a discriminated union with `type` to identify the request kind.\n *\n * @param request - The request payload (check `request.type` for the kind)\n * @param options - Request options including abort signal\n * @returns Response with SSE stream body\n *\n * @example\n * ```typescript\n * request: (payload, options) =>\n * fetch('/api/trigger', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, ...payload }),\n * signal: options?.signal,\n * })\n * ```\n */\n request: (request: HttpRequest, options?: HttpRequestOptions) => Promise<Response>;\n}\n\n// =============================================================================\n// Transport Implementation\n// =============================================================================\n\n/**\n * Create an HTTP transport using native fetch() and SSE parsing.\n * This is the default transport for Next.js and other HTTP-based applications.\n *\n * @example\n * ```typescript\n * const transport = createHttpTransport({\n * request: (payload, options) =>\n * fetch('/api/trigger', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, ...payload }),\n * signal: options?.signal,\n * }),\n * });\n * ```\n */\nexport function createHttpTransport(options: HttpTransportOptions): Transport {\n let abortController: AbortController | null = null;\n\n async function* streamResponse(responsePromise: Promise<Response>) {\n try {\n const response = await responsePromise;\n\n if (!response.ok) {\n const errorText = await response.text().catch(() => `Request failed: ${response.status}`);\n throw new Error(errorText);\n }\n\n if (!response.body) {\n throw new Error('Response body is empty');\n }\n\n for await (const event of parseSSEStream(response, abortController!.signal)) {\n if (abortController?.signal.aborted) {\n break;\n }\n yield event;\n }\n } catch (err) {\n if (isAbortError(err)) {\n return;\n }\n throw err;\n }\n }\n\n return {\n async *trigger(triggerName, input) {\n abortController = new AbortController();\n const response = options.request(\n { type: 'trigger', triggerName, input },\n { signal: abortController.signal },\n );\n yield* streamResponse(response);\n },\n\n async *continueWithToolResults(executionId, toolResults) {\n abortController = new AbortController();\n const response = options.request(\n { type: 'continue', executionId, toolResults },\n { signal: abortController.signal },\n );\n yield* streamResponse(response);\n },\n\n stop() {\n abortController?.abort();\n abortController = null;\n },\n };\n}\n","import { safeParseStreamEvent, type StreamEvent, type ToolResult } from '@octavus/core';\nimport type { SocketTransport, ConnectionState, ConnectionStateListener } from './types';\n\n// =============================================================================\n// Types\n// =============================================================================\n\n/**\n * Socket interface compatible with both WebSocket and SockJS.\n *\n * Uses MessageEvent for the message handler, which both WebSocket and SockJS support.\n * The `| undefined` union accommodates SockJS's optional property typing.\n */\nexport interface SocketLike {\n send(data: string): void;\n close(): void;\n readyState: number;\n onmessage: ((event: MessageEvent) => void) | null | undefined;\n onclose: ((event: CloseEvent) => void) | null | undefined;\n}\n\n/** WebSocket readyState constants */\nconst SOCKET_OPEN = 1;\n\n// =============================================================================\n// Transport Options\n// =============================================================================\n\n/**\n * Options for creating a socket transport.\n */\nexport interface SocketTransportOptions {\n /**\n * Function to create and connect the socket.\n * Works directly with WebSocket and SockJS - no wrappers needed.\n *\n * @example Native WebSocket\n * ```typescript\n * connect: () => new Promise((resolve, reject) => {\n * const ws = new WebSocket('wss://api.example.com/stream?sessionId=xxx');\n * ws.onopen = () => resolve(ws);\n * ws.onerror = () => reject(new Error('Connection failed'));\n * })\n * ```\n *\n * @example SockJS\n * ```typescript\n * connect: () => new Promise((resolve, reject) => {\n * const sock = new SockJS('/chat-service');\n * sock.onopen = () => resolve(sock);\n * sock.onerror = () => reject(new Error('Connection failed'));\n * })\n * ```\n */\n connect: () => Promise<SocketLike>;\n\n /**\n * Called for every message received (parsed as JSON).\n * Use this to handle custom (non-Octavus) events.\n * Octavus StreamEvents are handled automatically by the transport.\n *\n * @example\n * ```typescript\n * onMessage: (data) => {\n * const msg = data as { type: string };\n * if (msg.type === 'typing-indicator') {\n * setIsTyping(true);\n * }\n * if (msg.type === 'presence-update') {\n * updatePresence(msg.users);\n * }\n * }\n * ```\n */\n onMessage?: (data: unknown) => void;\n\n /**\n * Called when the socket connection closes.\n * Use this for cleanup or reconnection logic.\n */\n onClose?: () => void;\n}\n\n/**\n * Create a socket transport that works with any WebSocket-like connection.\n * Supports native WebSocket, SockJS, or any compatible socket implementation.\n *\n * The server should send StreamEvent format (same as SSE) over the socket.\n * Unknown events are safely ignored using Zod validation.\n *\n * ## Connection Lifecycle\n *\n * By default, the socket connects **lazily** on the first `send()` call.\n * Use `connect()` to establish the connection eagerly (e.g., on component mount):\n *\n * ```typescript\n * // Eager connection for UI status indicators\n * useEffect(() => {\n * transport.connect()\n * .then(() => console.log('Connected'))\n * .catch((err) => console.error('Failed:', err));\n *\n * return () => transport.disconnect();\n * }, [transport]);\n * ```\n *\n * @example Basic usage with WebSocket\n * ```typescript\n * const transport = createSocketTransport({\n * connect: () => new Promise((resolve, reject) => {\n * const ws = new WebSocket(`wss://api.octavus.ai/stream?sessionId=${sessionId}`);\n * ws.onopen = () => resolve(ws);\n * ws.onerror = () => reject(new Error('Connection failed'));\n * }),\n * });\n * ```\n *\n * @example With SockJS and connection state\n * ```typescript\n * const transport = createSocketTransport({\n * connect: () => new Promise((resolve, reject) => {\n * const sock = new SockJS('/octavus-stream');\n * sock.onopen = () => resolve(sock);\n * sock.onerror = () => reject(new Error('Connection failed'));\n * }),\n * });\n *\n * // Subscribe to connection state changes\n * transport.onConnectionStateChange((state, error) => {\n * setConnectionState(state);\n * if (error) setConnectionError(error);\n * });\n * ```\n */\nexport function createSocketTransport(options: SocketTransportOptions): SocketTransport {\n let socket: SocketLike | null = null;\n let eventQueue: StreamEvent[] = [];\n let eventResolver: ((event: StreamEvent | null) => void) | null = null;\n let isStreaming = false;\n\n let connectionState: ConnectionState = 'disconnected';\n let connectionError: Error | undefined;\n let connectionPromise: Promise<void> | null = null;\n const connectionListeners = new Set<ConnectionStateListener>();\n\n function setConnectionState(state: ConnectionState, error?: Error) {\n connectionState = state;\n connectionError = error;\n connectionListeners.forEach((listener) => listener(state, error));\n }\n\n function enqueueEvent(event: StreamEvent) {\n if (eventResolver) {\n eventResolver(event);\n eventResolver = null;\n } else {\n eventQueue.push(event);\n }\n }\n\n function nextEvent(): Promise<StreamEvent | null> {\n if (eventQueue.length > 0) {\n return Promise.resolve(eventQueue.shift()!);\n }\n if (!isStreaming) {\n return Promise.resolve(null);\n }\n return new Promise((resolve) => {\n eventResolver = resolve;\n });\n }\n\n function setupSocketHandlers(sock: SocketLike): void {\n sock.onmessage = (e: MessageEvent) => {\n try {\n const data: unknown = typeof e.data === 'string' ? JSON.parse(e.data) : e.data;\n\n options.onMessage?.(data);\n\n const result = safeParseStreamEvent(data);\n if (result.success) {\n const event = result.data;\n enqueueEvent(event);\n\n if (event.type === 'finish' || event.type === 'error') {\n isStreaming = false;\n }\n }\n } catch {\n // Malformed JSON, skip\n }\n };\n\n sock.onclose = () => {\n socket = null;\n connectionPromise = null;\n setConnectionState('disconnected');\n options.onClose?.();\n\n isStreaming = false;\n if (eventResolver) {\n eventResolver(null);\n eventResolver = null;\n }\n };\n }\n\n async function ensureConnected(): Promise<void> {\n // Already connected\n if (socket?.readyState === SOCKET_OPEN) {\n return;\n }\n\n // Connection in progress - wait for it\n if (connectionPromise) {\n await connectionPromise;\n return;\n }\n\n // Start new connection\n setConnectionState('connecting');\n\n connectionPromise = (async () => {\n try {\n const sock = await options.connect();\n socket = sock;\n setupSocketHandlers(sock);\n setConnectionState('connected');\n } catch (err) {\n socket = null;\n connectionPromise = null;\n const error = err instanceof Error ? err : new Error('Connection failed');\n setConnectionState('error', error);\n throw error;\n }\n })();\n\n await connectionPromise;\n }\n\n return {\n // =========================================================================\n // Connection Management\n // =========================================================================\n\n get connectionState(): ConnectionState {\n return connectionState;\n },\n\n onConnectionStateChange(listener: ConnectionStateListener): () => void {\n connectionListeners.add(listener);\n // Immediately notify with current state\n listener(connectionState, connectionError);\n return () => connectionListeners.delete(listener);\n },\n\n async connect(): Promise<void> {\n await ensureConnected();\n },\n\n disconnect(): void {\n if (socket) {\n socket.close();\n socket = null;\n }\n connectionPromise = null;\n isStreaming = false;\n if (eventResolver) {\n eventResolver(null);\n eventResolver = null;\n }\n setConnectionState('disconnected');\n },\n\n // =========================================================================\n // Streaming\n // =========================================================================\n\n async *trigger(triggerName, input) {\n await ensureConnected();\n\n eventQueue = [];\n eventResolver = null; // Clear any pending resolver\n isStreaming = true;\n\n // Note: clientToolResults not sent here - socket uses sendClientToolResults() for continuation\n socket!.send(\n JSON.stringify({\n type: 'trigger',\n triggerName,\n input,\n }),\n );\n\n while (true) {\n const event = await nextEvent();\n if (event === null) break;\n yield event;\n if (event.type === 'finish' || event.type === 'error') break;\n }\n },\n\n stop() {\n if (socket?.readyState === SOCKET_OPEN) {\n socket.send(JSON.stringify({ type: 'stop' }));\n }\n isStreaming = false;\n if (eventResolver) {\n eventResolver(null);\n eventResolver = null;\n }\n },\n\n /**\n * Continue execution with tool results after client-side tool handling.\n * @param executionId - The execution ID from the client-tool-request event\n * @param toolResults - All tool results (server + client) to send\n */\n async *continueWithToolResults(executionId: string, toolResults: ToolResult[]) {\n await ensureConnected();\n\n eventQueue = [];\n eventResolver = null; // Clear any pending resolver from previous operation\n isStreaming = true;\n\n socket!.send(\n JSON.stringify({\n type: 'continue',\n executionId,\n toolResults,\n }),\n );\n\n while (true) {\n const event = await nextEvent();\n if (event === null) break;\n yield event;\n if (event.type === 'finish' || event.type === 'error') break;\n }\n },\n };\n}\n","export {\n OctavusChat,\n type OctavusChatOptions,\n type ChatStatus,\n type UserMessageInput,\n type ClientToolContext,\n type ClientToolHandler,\n type InteractiveTool,\n} from './chat';\n\nexport { uploadFiles, type UploadFilesOptions, type UploadUrlsResponse } from './files';\n\nexport { parseSSEStream } from './stream/reader';\n\n// Transport exports\nexport {\n createHttpTransport,\n createSocketTransport,\n isSocketTransport,\n type Transport,\n type SocketTransport,\n type ConnectionState,\n type ConnectionStateListener,\n type HttpTransportOptions,\n type HttpRequestOptions,\n type HttpRequest,\n type TriggerRequest,\n type ContinueRequest,\n type SocketLike,\n type SocketTransportOptions,\n} from './transports';\n\nexport type * from '@octavus/core';\nexport {\n // Error classes\n AppError,\n NotFoundError,\n ValidationError,\n ConflictError,\n ForbiddenError,\n OctavusError,\n // Error type guards\n isRateLimitError,\n isAuthenticationError,\n isProviderError,\n isToolError,\n isRetryableError,\n isValidationError,\n // Error event helpers\n createErrorEvent,\n errorToStreamEvent,\n createInternalErrorEvent,\n createApiErrorEvent,\n // Utilities\n generateId,\n isAbortError,\n // Thread helpers\n MAIN_THREAD,\n resolveThread,\n isMainThread,\n threadForPart,\n isOtherThread,\n // Type guards\n isFileReference,\n isFileReferenceArray,\n // Safe parse helpers\n safeParseStreamEvent,\n safeParseUIMessage,\n safeParseUIMessages,\n // Skills\n OCTAVUS_SKILL_TOOLS,\n isOctavusSkillTool,\n getSkillSlugFromToolCall,\n} from '@octavus/core';\n"],"mappings":";AAAA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OAgBK;;;ACkCP,SAAS,uBACP,KACA,MACA,YACe;AACf,SAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,UAAM,MAAM,IAAI,eAAe;AAE/B,QAAI,OAAO,iBAAiB,YAAY,CAAC,UAAU;AACjD,UAAI,MAAM,kBAAkB;AAC1B,cAAM,WAAW,KAAK,MAAO,MAAM,SAAS,MAAM,QAAS,GAAG;AAC9D,qBAAa,QAAQ;AAAA,MACvB;AAAA,IACF,CAAC;AAED,QAAI,iBAAiB,QAAQ,MAAM;AACjC,UAAI,IAAI,UAAU,OAAO,IAAI,SAAS,KAAK;AACzC,gBAAQ;AAAA,MACV,OAAO;AACL,eAAO,IAAI,MAAM,6BAA6B,IAAI,MAAM,EAAE,CAAC;AAAA,MAC7D;AAAA,IACF,CAAC;AAED,QAAI,iBAAiB,SAAS,MAAM;AAClC,aAAO,IAAI,MAAM,8BAA8B,CAAC;AAAA,IAClD,CAAC;AAED,QAAI,iBAAiB,SAAS,MAAM;AAClC,aAAO,IAAI,MAAM,gBAAgB,CAAC;AAAA,IACpC,CAAC;AAED,QAAI,KAAK,OAAO,GAAG;AACnB,QAAI,iBAAiB,gBAAgB,KAAK,QAAQ,0BAA0B;AAC5E,QAAI,KAAK,IAAI;AAAA,EACf,CAAC;AACH;AA+BA,eAAsB,YACpB,OACA,SAC0B;AAC1B,QAAM,YAAY,MAAM,KAAK,KAAK;AAElC,MAAI,UAAU,WAAW,GAAG;AAC1B,WAAO,CAAC;AAAA,EACV;AAEA,QAAM,EAAE,OAAO,YAAY,IAAI,MAAM,QAAQ;AAAA,IAC3C,UAAU,IAAI,CAAC,OAAO;AAAA,MACpB,UAAU,EAAE;AAAA,MACZ,WAAW,EAAE,QAAQ;AAAA,MACrB,MAAM,EAAE;AAAA,IACV,EAAE;AAAA,EACJ;AAEA,QAAM,aAA8B,CAAC;AAErC,WAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,UAAM,OAAO,UAAU,CAAC;AACxB,UAAM,aAAa,YAAY,CAAC;AAEhC,UAAM,uBAAuB,WAAW,WAAW,MAAM,CAAC,aAAa;AACrE,cAAQ,aAAa,GAAG,QAAQ;AAAA,IAClC,CAAC;AAED,eAAW,KAAK;AAAA,MACd,IAAI,WAAW;AAAA,MACf,WAAW,KAAK,QAAQ;AAAA,MACxB,KAAK,WAAW;AAAA,MAChB,UAAU,KAAK;AAAA,MACf,MAAM,KAAK;AAAA,IACb,CAAC;AAAA,EACH;AAEA,SAAO;AACT;;;ADrIA,IAAM,wBAAwB,oBAAI,IAAI,CAAC,gBAAgB,oBAAoB,gBAAgB,CAAC;AA0P5F,SAAS,kBAAkB,OAAyB,OAAoC;AACtF,QAAM,QAAyB,CAAC;AAGhC,MAAI,SAAS,MAAM,SAAS,GAAG;AAC7B,eAAW,QAAQ,OAAO;AACxB,YAAM,KAAK;AAAA,QACT,MAAM;AAAA,QACN,IAAI,KAAK;AAAA,QACT,WAAW,KAAK;AAAA,QAChB,KAAK,KAAK;AAAA,QACV,UAAU,KAAK;AAAA,QACf,MAAM,KAAK;AAAA,MACb,CAAC;AAAA,IACH;AAAA,EACF;AAGA,MAAI,MAAM,YAAY,QAAW;AAC/B,QAAI,OAAO,MAAM,YAAY,UAAU;AAErC,YAAM,KAAK,EAAE,MAAM,QAAQ,MAAM,MAAM,SAAS,QAAQ,OAAO,CAAC;AAAA,IAClE,OAAO;AAGL,YAAM,WAAY,MAAM,QAA8B,QAAQ;AAC9D,YAAM,KAAK;AAAA,QACT,MAAM;AAAA,QACN,IAAI,WAAW;AAAA,QACf;AAAA,QACA,QAAQ,MAAM;AAAA,QACd,QAAQ;AAAA,MACV,CAAC;AAAA,IACH;AAAA,EACF;AAEA,SAAO;AAAA,IACL,IAAI,WAAW;AAAA,IACf,MAAM;AAAA,IACN;AAAA,IACA,QAAQ;AAAA,IACR,WAAW,oBAAI,KAAK;AAAA,EACtB;AACF;AAKA,SAAS,iBAAiB,UAA2B;AACnD,MAAI,CAAC,SAAS,KAAK,GAAG;AACpB,WAAO;AAAA,EACT;AAEA,MAAI;AACF,WAAO,KAAK,MAAM,QAAQ;AAAA,EAC5B,QAAQ;AAAA,EAER;AAEA,MAAI,QAAQ;AAGZ,MAAI,aAAa;AACjB,MAAI,eAAe;AACnB,MAAI,WAAW;AACf,MAAI,UAAU;AAEd,aAAW,QAAQ,OAAO;AACxB,QAAI,SAAS;AACX,gBAAU;AACV;AAAA,IACF;AAEA,QAAI,SAAS,MAAM;AACjB,gBAAU;AACV;AAAA,IACF;AAEA,QAAI,SAAS,KAAK;AAChB,iBAAW,CAAC;AACZ;AAAA,IACF;AAEA,QAAI,CAAC,UAAU;AACb,UAAI,SAAS,IAAK,eAAc;AAAA,eACvB,SAAS,IAAK,eAAc;AAAA,eAC5B,SAAS,IAAK,iBAAgB;AAAA,eAC9B,SAAS,IAAK,iBAAgB;AAAA,IACzC;AAAA,EACF;AAGA,MAAI,SAAS;AAEX,aAAS;AAAA,EACX;AACA,MAAI,UAAU;AACZ,aAAS;AAAA,EACX;AACA,SAAO,eAAe,GAAG;AACvB,aAAS;AACT,oBAAgB;AAAA,EAClB;AACA,SAAO,aAAa,GAAG;AACrB,aAAS;AACT,kBAAc;AAAA,EAChB;AAEA,MAAI;AACF,WAAO,KAAK,MAAM,KAAK;AAAA,EACzB,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEA,SAAS,4BAA4C;AACnD,SAAO;AAAA,IACL,WAAW,WAAW;AAAA,IACtB,OAAO,CAAC;AAAA,IACR,aAAa;AAAA,IACb,QAAQ,oBAAI,IAAI;AAAA,IAChB,sBAAsB;AAAA,IACtB,2BAA2B;AAAA,IAC3B,wBAAwB;AAAA,IACxB,iBAAiB;AAAA,IACjB,eAAe,oBAAI,IAAI;AAAA,IACvB,kBAAkB,oBAAI,IAAI;AAAA,EAC5B;AACF;AAEA,SAAS,sBAAsB,OAAuB,QAAyC;AAC7F,SAAO;AAAA,IACL,IAAI,MAAM;AAAA,IACV,MAAM;AAAA,IACN,OAAO,CAAC,GAAG,MAAM,KAAK;AAAA,IACtB;AAAA,IACA,WAAW,oBAAI,KAAK;AAAA,EACtB;AACF;AAOA,SAAS,cAAc,OAAwB,aAAuC;AACpF,SAAO,MAAM,IAAI,CAAC,SAAwB;AACxC,QAAI,KAAK,SAAS,UAAU,KAAK,SAAS,aAAa;AACrD,UAAI,KAAK,WAAW,aAAa;AAC/B,eAAO,EAAE,GAAG,MAAM,QAAQ,OAAO;AAAA,MACnC;AAAA,IACF;AACA,QAAI,KAAK,SAAS,YAAY,KAAK,WAAW,aAAa;AACzD,aAAO,EAAE,GAAG,MAAM,QAAQ,OAAO;AAAA,IACnC;AACA,QAAI,KAAK,SAAS,aAAa;AAC7B,UAAI,KAAK,WAAW,aAAa,KAAK,WAAW,WAAW;AAC1D,eAAO,EAAE,GAAG,MAAM,QAAQ,YAAY;AAAA,MACxC;AAAA,IACF;AACA,QAAI,KAAK,SAAS,eAAe,KAAK,WAAW,WAAW;AAC1D,aAAO,EAAE,GAAG,MAAM,QAAQ,YAAY;AAAA,IACxC;AACA,QAAI,KAAK,SAAS,YAAY,KAAK,WAAW,WAAW;AACvD,aAAO;AAAA,QACL,GAAG;AAAA,QACH,QAAQ;AAAA,QACR,OAAO;AAAA,QACP,OAAO,cAAc,KAAK,KAAK;AAAA;AAAA,MACjC;AAAA,IACF;AACA,WAAO;AAAA,EACT,CAAC;AACH;AA0CO,IAAM,cAAN,MAAkB;AAAA;AAAA,EAEf;AAAA,EACA,UAAsB;AAAA,EACtB,SAA8B;AAAA,EAC9B;AAAA,EACA;AAAA,EACA,iBAAwC;AAAA;AAAA;AAAA,EAIxC,sBAAsB,oBAAI,IAAgC;AAAA;AAAA,EAE1D,wBAAwB,oBAAI,IAA8B;AAAA;AAAA,EAE1D,2BAA8D,CAAC;AAAA,EAC/D,wBAAsC,CAAC;AAAA,EACvC,6BAAqD;AAAA;AAAA,EAErD,qBAAmC,CAAC;AAAA;AAAA,EAEpC,sBAAqC;AAAA;AAAA;AAAA,EAGrC,mBAAmB;AAAA;AAAA;AAAA,EAGnB,uBAAuB;AAAA;AAAA,EAGvB,YAAY,oBAAI,IAAc;AAAA,EAEtC,YAAY,SAA6B;AACvC,SAAK,UAAU;AACf,SAAK,YAAY,QAAQ,mBAAmB,CAAC;AAC7C,SAAK,YAAY,QAAQ;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,cAAc,SAAmF;AAC/F,SAAK,UAAU,EAAE,GAAG,KAAK,SAAS,GAAG,QAAQ;AAAA,EAC/C;AAAA;AAAA;AAAA;AAAA,EAMA,IAAI,WAAwB;AAC1B,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,IAAI,SAAqB;AACvB,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,IAAI,QAA6B;AAC/B,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAoBA,IAAI,qBAAwD;AAC1D,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,UAAU,UAAgC;AACxC,SAAK,UAAU,IAAI,QAAQ;AAC3B,WAAO,MAAM,KAAK,UAAU,OAAO,QAAQ;AAAA,EAC7C;AAAA,EAEQ,kBAAwB;AAC9B,SAAK,UAAU,QAAQ,CAAC,MAAM,EAAE,CAAC;AAAA,EACnC;AAAA;AAAA;AAAA;AAAA,EAMQ,YAAY,UAA6B;AAC/C,SAAK,YAAY;AACjB,SAAK,gBAAgB;AAAA,EACvB;AAAA,EAEQ,UAAU,QAA0B;AAC1C,SAAK,UAAU;AACf,SAAK,gBAAgB;AAAA,EACvB;AAAA,EAEQ,SAAS,OAAkC;AACjD,SAAK,SAAS;AACd,SAAK,gBAAgB;AAAA,EACvB;AAAA,EAEQ,gCAAsC;AAC5C,UAAM,QAA2C,CAAC;AAClD,eAAW,CAAC,UAAU,KAAK,KAAK,KAAK,oBAAoB,QAAQ,GAAG;AAClE,YAAM,QAAQ,IAAI,MAAM,IAAI,CAAC,UAAU;AAAA,QACrC,YAAY,KAAK;AAAA,QACjB,UAAU,KAAK;AAAA,QACf,MAAM,KAAK;AAAA,QACX,QAAQ,CAAC,WAAoB,KAAK,iBAAiB,KAAK,YAAY,MAAM;AAAA,QAC1E,QAAQ,CAAC,WACP,KAAK,iBAAiB,KAAK,YAAY,QAAW,UAAU,gBAAgB;AAAA,MAChF,EAAE;AAAA,IACJ;AACA,SAAK,2BAA2B;AAAA,EAClC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA6BA,MAAM,KACJ,aACA,OACA,aACe;AACf,SAAK,UAAU,KAAK;AAEpB,QAAI;AACJ,QAAI,aAAa,aAAa,OAAO;AACnC,YAAM,QAAQ,YAAY,YAAY;AACtC,UAAI,qBAAqB,KAAK,GAAG;AAC/B,mBAAW;AAAA,MACb,WAAW,KAAK,QAAQ,mBAAmB;AACzC,mBAAW,MAAM,YAAY,OAAO;AAAA,UAClC,mBAAmB,KAAK,QAAQ;AAAA,QAClC,CAAC;AAAA,MACH,OAAO;AACL,cAAM,IAAI;AAAA,UACR;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAGA,QAAI,iBAAiB;AACrB,QAAI,OAAO,UAAU,UAAa,CAAC,qBAAqB,MAAM,KAAK,GAAG;AACpE,UAAI,KAAK,QAAQ,mBAAmB;AAClC,cAAM,aAAa,MAAM;AACzB,cAAM,eACJ,YACC,MAAM,YAAY,YAAY;AAAA,UAC7B,mBAAmB,KAAK,QAAQ;AAAA,QAClC,CAAC;AACH,yBAAiB,EAAE,GAAG,OAAO,OAAO,aAAa;AACjD,mBAAW,YAAY;AAAA,MACzB;AAAA,IACF;AAGA,QAAI,aAAa,gBAAgB,QAAW;AAC1C,YAAM,UAAU,kBAAkB,YAAY,aAAa,QAAQ;AACnE,WAAK,YAAY,CAAC,GAAG,KAAK,WAAW,OAAO,CAAC;AAAA,IAC/C;AAEA,SAAK,UAAU,WAAW;AAC1B,SAAK,SAAS,IAAI;AAClB,SAAK,iBAAiB,0BAA0B;AAGhD,SAAK,oBAAoB,MAAM;AAC/B,SAAK,sBAAsB,MAAM;AACjC,SAAK,wBAAwB,CAAC;AAC9B,SAAK,qBAAqB,CAAC;AAC3B,SAAK,sBAAsB;AAC3B,SAAK,mBAAmB;AACxB,SAAK,uBAAuB;AAC5B,SAAK,8BAA8B;AAEnC,QAAI;AACF,uBAAiB,SAAS,KAAK,UAAU,QAAQ,aAAa,cAAc,GAAG;AAC7E,YAAI,KAAK,mBAAmB,KAAM;AAElC,aAAK,kBAAkB,OAAO,KAAK,cAAc;AAAA,MACnD;AAAA,IACF,SAAS,KAAK;AAEZ,YAAM,WAAW,aAAa,WAAW,GAAG,IACxC,MACA,IAAI,aAAa;AAAA,QACf,WAAW;AAAA,QACX,SAAS,eAAe,QAAQ,IAAI,UAAU;AAAA,QAC9C,QAAQ;AAAA,QACR,WAAW;AAAA,QACX,OAAO;AAAA,MACT,CAAC;AAGL,YAAM,QAAQ,KAAK;AACnB,UAAI,UAAU,MAAM;AAClB,cAAM,WAAW,CAAC,GAAG,KAAK,SAAS;AACnC,cAAM,UAAU,SAAS,SAAS,SAAS,CAAC;AAE5C,YAAI,MAAM,MAAM,SAAS,GAAG;AAC1B,gBAAM,aAAa,cAAc,MAAM,OAAO,cAAc;AAE5D,gBAAM,eAA0B;AAAA,YAC9B,IAAI,MAAM;AAAA,YACV,MAAM;AAAA,YACN,OAAO;AAAA,YACP,QAAQ;AAAA,YACR,WAAW,oBAAI,KAAK;AAAA,UACtB;AAEA,cAAI,SAAS,OAAO,MAAM,WAAW;AACnC,qBAAS,SAAS,SAAS,CAAC,IAAI;AAAA,UAClC,OAAO;AACL,qBAAS,KAAK,YAAY;AAAA,UAC5B;AACA,eAAK,YAAY,QAAQ;AAAA,QAC3B,WAAW,SAAS,OAAO,MAAM,WAAW;AAE1C,mBAAS,IAAI;AACb,eAAK,YAAY,QAAQ;AAAA,QAC3B;AAAA,MACF;AAEA,WAAK,SAAS,QAAQ;AACtB,WAAK,UAAU,OAAO;AACtB,WAAK,iBAAiB;AACtB,WAAK,QAAQ,UAAU,QAAQ;AAAA,IACjC;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmBA,MAAM,YACJ,OACA,YAC0B;AAC1B,QAAI,CAAC,KAAK,QAAQ,mBAAmB;AACnC,YAAM,IAAI,MAAM,+CAA+C;AAAA,IACjE;AACA,WAAO,MAAM,YAAY,OAAO;AAAA,MAC9B,mBAAmB,KAAK,QAAQ;AAAA,MAChC;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA;AAAA,EAMQ,iBAAiB,YAAoB,QAAkB,OAAsB;AACnF,UAAM,cAAc,KAAK,sBAAsB,IAAI,UAAU;AAC7D,QAAI,CAAC,aAAa;AAEhB;AAAA,IACF;AAGA,SAAK,sBAAsB,OAAO,UAAU;AAC5C,UAAM,eAAe,KAAK,oBAAoB,IAAI,YAAY,QAAQ;AACtE,QAAI,cAAc;AAChB,YAAM,WAAW,aAAa,OAAO,CAAC,MAAM,EAAE,eAAe,UAAU;AACvE,UAAI,SAAS,WAAW,GAAG;AACzB,aAAK,oBAAoB,OAAO,YAAY,QAAQ;AAAA,MACtD,OAAO;AACL,aAAK,oBAAoB,IAAI,YAAY,UAAU,QAAQ;AAAA,MAC7D;AAAA,IACF;AACA,SAAK,8BAA8B;AAEnC,UAAM,aAAyB;AAAA,MAC7B;AAAA,MACA,UAAU,YAAY;AAAA,MACtB,QAAQ,QAAQ,SAAY;AAAA,MAC5B;AAAA,MACA,gBAAgB,YAAY;AAAA,MAC5B,YAAY,YAAY;AAAA,MACxB,QAAQ,YAAY;AAAA,MACpB,UAAU,YAAY;AAAA,IACxB;AACA,SAAK,sBAAsB,KAAK,UAAU;AAE1C,QAAI,OAAO;AACT,WAAK,oBAAoB,YAAY,KAAK;AAAA,IAC5C,OAAO;AACL,WAAK,wBAAwB,YAAY,MAAM;AAAA,IACjD;AAEA,QAAI,KAAK,sBAAsB,SAAS,GAAG;AACzC,WAAK,KAAK,8BAA8B;AAAA,IAC1C;AAEA,SAAK,gBAAgB;AAAA,EACvB;AAAA;AAAA,EAGA,OAAa;AACX,QAAI,KAAK,YAAY,eAAe,KAAK,YAAY,kBAAkB;AACrE;AAAA,IACF;AAEA,SAAK,4BAA4B,MAAM;AACvC,SAAK,6BAA6B;AAClC,SAAK,oBAAoB,MAAM;AAC/B,SAAK,sBAAsB,MAAM;AACjC,SAAK,wBAAwB,CAAC;AAC9B,SAAK,qBAAqB,CAAC;AAC3B,SAAK,sBAAsB;AAC3B,SAAK,mBAAmB;AACxB,SAAK,uBAAuB;AAC5B,SAAK,8BAA8B;AAEnC,SAAK,UAAU,KAAK;AAEpB,UAAM,QAAQ,KAAK;AACnB,QAAI,SAAS,MAAM,MAAM,SAAS,GAAG;AACnC,YAAM,aAAa,cAAc,MAAM,OAAO,iBAAiB;AAE/D,YAAM,eAA0B;AAAA,QAC9B,IAAI,MAAM;AAAA,QACV,MAAM;AAAA,QACN,OAAO;AAAA,QACP,QAAQ;AAAA,QACR,WAAW,oBAAI,KAAK;AAAA,MACtB;AAEA,YAAM,WAAW,CAAC,GAAG,KAAK,SAAS;AACnC,YAAM,UAAU,SAAS,SAAS,SAAS,CAAC;AAC5C,UAAI,SAAS,OAAO,MAAM,WAAW;AACnC,iBAAS,SAAS,SAAS,CAAC,IAAI;AAAA,MAClC,OAAO;AACL,iBAAS,KAAK,YAAY;AAAA,MAC5B;AACA,WAAK,YAAY,QAAQ;AAAA,IAC3B;AAEA,SAAK,iBAAiB;AACtB,SAAK,UAAU,MAAM;AACrB,SAAK,QAAQ,SAAS;AAAA,EACxB;AAAA;AAAA;AAAA;AAAA,EAMQ,kBAAkB,OAAoB,OAA6B;AACzE,YAAQ,MAAM,MAAM;AAAA,MAClB,KAAK;AAEH,YAAI,MAAM,aAAa;AACrB,eAAK,QAAQ,UAAU,MAAM,WAAW;AAAA,QAC1C;AACA;AAAA,MAEF,KAAK,eAAe;AAClB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,cAAM,QAAoB;AAAA,UACxB,SAAS,MAAM;AAAA,UACf,WAAW,MAAM;AAAA,UACjB,WAAW,MAAM;AAAA,UACjB,SAAS,MAAM;AAAA,UACf,aAAa,MAAM;AAAA,UACnB,cAAc,MAAM,gBAAgB;AAAA,UACpC,QAAQ,MAAM;AAAA,UACd,WAAW;AAAA,UACX,MAAM;AAAA,UACN,WAAW,oBAAI,IAAI;AAAA,QACrB;AACA,cAAM,OAAO,IAAI,MAAM,SAAS,KAAK;AACrC,cAAM,cAAc;AAEpB,cAAM,cAAc,sBAAsB,IAAI,MAAM,SAAS;AAC7D,cAAM,WAAW,MAAM,YAAY;AACnC,YAAI,eAAe,CAAC,UAAU;AAC5B,gBAAM,SAAS,MAAM;AACrB,gBAAM,gBAAiC;AAAA,YACrC,MAAM;AAAA,YACN,aAAa,MAAM;AAAA,YACnB,MAAM,MAAM,eAAe,MAAM;AAAA,YACjC,eAAe,MAAM;AAAA,YACrB,QAAQ;AAAA,YACR,QAAQ,cAAc,MAAM;AAAA,UAC9B;AAEA,cAAI,aAAa;AACf,kBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,uBAAW,MAAM,KAAK,aAAa;AACnC,kBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,UACvD,OAAO;AACL,kBAAM,MAAM,KAAK,aAAa;AAAA,UAChC;AAAA,QACF;AAEA,cAAM,uBAAuB;AAC7B,cAAM,4BAA4B;AAElC,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,aAAa;AAChB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AAEf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,gBAAM,qBAAqB,WAAW,MAAM;AAAA,YAC1C,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,gBAAgB,MAAM;AAAA,UAC1E;AACA,cAAI,sBAAsB,GAAG;AAC3B,kBAAM,OAAO,WAAW,MAAM,kBAAkB;AAChD,uBAAW,MAAM,kBAAkB,IAAI,EAAE,GAAG,MAAM,QAAQ,OAAO;AACjE,kBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,UACvD;AAAA,QACF,OAAO;AACL,gBAAM,qBAAqB,MAAM,MAAM;AAAA,YACrC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,gBAAgB,MAAM;AAAA,UAC1E;AACA,cAAI,sBAAsB,GAAG;AAC3B,kBAAM,OAAO,MAAM,MAAM,kBAAkB;AAC3C,kBAAM,MAAM,kBAAkB,IAAI,EAAE,GAAG,MAAM,QAAQ,OAAO;AAAA,UAC9D;AAAA,QACF;AAEA,YAAI,MAAM,aAAa,YAAY,MAAM,SAAS;AAChD,gBAAM,cAAc;AAAA,QACtB;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,mBAAmB;AACtB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,cAAM,gBAAiC;AAAA,UACrC,MAAM;AAAA,UACN,MAAM;AAAA,UACN,QAAQ;AAAA,UACR,QAAQ,cAAc,MAAM,aAAa,MAAM;AAAA,QACjD;AAEA,YAAI,aAAa;AAEf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,qBAAW,MAAM,KAAK,aAAa;AACnC,sBAAY,4BAA4B,WAAW,MAAM,SAAS;AAClE,gBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,QACvD,OAAO;AACL,gBAAM,MAAM,KAAK,aAAa;AAC9B,gBAAM,4BAA4B,MAAM,MAAM,SAAS;AAAA,QACzD;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,mBAAmB;AACtB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AAEf,cAAI,YAAY,8BAA8B,MAAM;AAClD,kBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,kBAAM,OAAO,WAAW,MAAM,YAAY,yBAAyB;AACnE,iBAAK,QAAQ,MAAM;AACnB,uBAAW,MAAM,YAAY,yBAAyB,IAAI,EAAE,GAAG,KAAK;AACpE,kBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,UACvD;AAAA,QACF,OAAO;AACL,cAAI,MAAM,8BAA8B,MAAM;AAC5C,kBAAM,OAAO,MAAM,MAAM,MAAM,yBAAyB;AACxD,iBAAK,QAAQ,MAAM;AACnB,kBAAM,MAAM,MAAM,yBAAyB,IAAI,EAAE,GAAG,KAAK;AAAA,UAC3D;AAEA,cAAI,MAAM,aAAa;AACrB,kBAAM,YAAY,aAAa,MAAM;AAAA,UACvC;AAAA,QACF;AAEA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,iBAAiB;AACpB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AAEf,cAAI,YAAY,8BAA8B,MAAM;AAClD,kBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,kBAAM,OAAO,WAAW,MAAM,YAAY,yBAAyB;AACnE,iBAAK,SAAS;AACd,uBAAW,MAAM,YAAY,yBAAyB,IAAI,EAAE,GAAG,KAAK;AACpE,kBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AACrD,wBAAY,4BAA4B;AAAA,UAC1C;AAAA,QACF,WAAW,MAAM,8BAA8B,MAAM;AACnD,gBAAM,OAAO,MAAM,MAAM,MAAM,yBAAyB;AACxD,eAAK,SAAS;AACd,gBAAM,MAAM,MAAM,yBAAyB,IAAI,EAAE,GAAG,KAAK;AACzD,gBAAM,4BAA4B;AAAA,QACpC;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,cAAc;AACjB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AACnE,cAAM,SAAS,cAAc,MAAM,aAAa,MAAM;AACtD,cAAM,gBAAgB,MAAM,aAAa,iBAAiB,SAAS,WAAW;AAG9E,YAAI,eAAe,eAAe;AAEhC,cAAI,MAAM,cAAc;AACtB,kBAAM,aAA2B;AAAA,cAC/B,MAAM;AAAA,cACN,IAAI,MAAM;AAAA,cACV,UAAU,MAAM;AAAA,cAChB,SAAS;AAAA,cACT,QAAQ;AAAA,cACR,QAAQ;AAAA,cACR;AAAA,YACF;AACA,gBAAI,aAAa;AACf,oBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,yBAAW,MAAM,KAAK,UAAU;AAChC,0BAAY,yBAAyB,WAAW,MAAM,SAAS;AAC/D,0BAAY,kBAAkB;AAC9B,0BAAY,uBAAuB;AACnC,oBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,YACvD,OAAO;AACL,oBAAM,MAAM,KAAK,UAAU;AAC3B,oBAAM,yBAAyB,MAAM,MAAM,SAAS;AACpD,oBAAM,kBAAkB;AACxB,oBAAM,uBAAuB;AAAA,YAC/B;AAAA,UACF,OAAO;AACL,kBAAM,WAAuB;AAAA,cAC3B,MAAM;AAAA,cACN,MAAM;AAAA,cACN,QAAQ;AAAA,cACR;AAAA,YACF;AACA,gBAAI,aAAa;AACf,oBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,yBAAW,MAAM,KAAK,QAAQ;AAC9B,0BAAY,uBAAuB,WAAW,MAAM,SAAS;AAC7D,0BAAY,yBAAyB;AACrC,oBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,YACvD,OAAO;AACL,oBAAM,MAAM,KAAK,QAAQ;AACzB,oBAAM,uBAAuB,MAAM,MAAM,SAAS;AAClD,oBAAM,yBAAyB;AAAA,YACjC;AAAA,UACF;AAAA,QACF;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,cAAc;AACjB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AAEf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,cAAI,YAAY,2BAA2B,MAAM;AAC/C,wBAAY,mBAAmB,MAAM;AACrC,kBAAM,OAAO,WAAW,MAAM,YAAY,sBAAsB;AAChE,kBAAM,SAAS,iBAAiB,YAAY,eAAe;AAC3D,gBAAI,WAAW,QAAW;AACxB,mBAAK,UAAU;AACf,yBAAW,MAAM,YAAY,sBAAsB,IAAI,EAAE,GAAG,KAAK;AAAA,YACnE;AAAA,UACF,WAAW,YAAY,yBAAyB,MAAM;AACpD,kBAAM,OAAO,WAAW,MAAM,YAAY,oBAAoB;AAC9D,iBAAK,QAAQ,MAAM;AACnB,uBAAW,MAAM,YAAY,oBAAoB,IAAI,EAAE,GAAG,KAAK;AAAA,UACjE;AACA,gBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,QACvD,OAAO;AACL,cAAI,MAAM,2BAA2B,MAAM;AACzC,kBAAM,mBAAmB,MAAM;AAC/B,kBAAM,OAAO,MAAM,MAAM,MAAM,sBAAsB;AACrD,kBAAM,SAAS,iBAAiB,MAAM,eAAe;AACrD,gBAAI,WAAW,QAAW;AACxB,mBAAK,UAAU;AACf,oBAAM,MAAM,MAAM,sBAAsB,IAAI,EAAE,GAAG,KAAK;AAAA,YACxD;AAAA,UACF,WAAW,MAAM,yBAAyB,MAAM;AAC9C,kBAAM,OAAO,MAAM,MAAM,MAAM,oBAAoB;AACnD,iBAAK,QAAQ,MAAM;AACnB,kBAAM,MAAM,MAAM,oBAAoB,IAAI,EAAE,GAAG,KAAK;AAAA,UACtD;AAEA,cAAI,MAAM,aAAa;AACrB,kBAAM,YAAY,QAAQ,MAAM;AAAA,UAClC;AAAA,QACF;AAEA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,YAAY;AACf,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AAEf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,cAAI,YAAY,2BAA2B,MAAM;AAC/C,kBAAM,OAAO,WAAW,MAAM,YAAY,sBAAsB;AAChE,gBAAI;AACF,oBAAM,cAAc,KAAK,MAAM,YAAY,eAAe;AAC1D,mBAAK,SAAS;AACd,mBAAK,UAAU;AACf,mBAAK,SAAS;AAAA,YAChB,QAAQ;AACN,mBAAK,SAAS;AACd,mBAAK,QAAQ;AAAA,YACf;AACA,uBAAW,MAAM,YAAY,sBAAsB,IAAI,EAAE,GAAG,KAAK;AACjE,wBAAY,yBAAyB;AACrC,wBAAY,kBAAkB;AAAA,UAChC,WAAW,YAAY,yBAAyB,MAAM;AACpD,kBAAM,OAAO,WAAW,MAAM,YAAY,oBAAoB;AAC9D,iBAAK,SAAS;AACd,uBAAW,MAAM,YAAY,oBAAoB,IAAI,EAAE,GAAG,KAAK;AAC/D,wBAAY,uBAAuB;AAAA,UACrC;AACA,gBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,QACvD,WAAW,MAAM,2BAA2B,MAAM;AAChD,gBAAM,OAAO,MAAM,MAAM,MAAM,sBAAsB;AACrD,cAAI;AACF,kBAAM,cAAc,KAAK,MAAM,MAAM,eAAe;AACpD,iBAAK,SAAS;AACd,iBAAK,UAAU;AACf,iBAAK,SAAS;AAAA,UAChB,QAAQ;AACN,iBAAK,SAAS;AACd,iBAAK,QAAQ;AAAA,UACf;AACA,gBAAM,MAAM,MAAM,sBAAsB,IAAI,EAAE,GAAG,KAAK;AACtD,gBAAM,yBAAyB;AAC/B,gBAAM,kBAAkB;AAAA,QAC1B,WAAW,MAAM,yBAAyB,MAAM;AAC9C,gBAAM,OAAO,MAAM,MAAM,MAAM,oBAAoB;AACnD,eAAK,SAAS;AACd,gBAAM,MAAM,MAAM,oBAAoB,IAAI,EAAE,GAAG,KAAK;AACpD,gBAAM,uBAAuB;AAAA,QAC/B;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,oBAAoB;AACvB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,cAAM,WAA2B;AAAA,UAC/B,MAAM;AAAA,UACN,YAAY,MAAM;AAAA,UAClB,UAAU,MAAM;AAAA,UAChB,aAAa,MAAM;AAAA,UACnB,MAAM,CAAC;AAAA,UACP,QAAQ;AAAA,UACR,QAAQ,cAAc,MAAM,aAAa,MAAM;AAAA,QACjD;AAGA,YAAI,aAAa;AACf,sBAAY,iBAAiB,IAAI,MAAM,YAAY,EAAE;AACrD,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,qBAAW,MAAM,KAAK,QAAQ;AAC9B,gBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,QACvD,OAAO;AACL,gBAAM,iBAAiB,IAAI,MAAM,YAAY,EAAE;AAC/C,gBAAM,MAAM,KAAK,QAAQ;AAEzB,cAAI,MAAM,aAAa;AACrB,kBAAM,YAAY,UAAU,IAAI,MAAM,YAAY,QAAQ;AAAA,UAC5D;AAAA,QACF;AAEA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,oBAAoB;AACvB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AAEf,gBAAM,WAAW,YAAY,iBAAiB,IAAI,MAAM,UAAU,KAAK;AACvE,gBAAM,cAAc,WAAW,MAAM;AACrC,sBAAY,iBAAiB,IAAI,MAAM,YAAY,WAAW;AAE9D,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,gBAAM,gBAAgB,WAAW,MAAM;AAAA,YACrC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,WAAW,WAAW,MAAM,aAAa;AAC/C,kBAAM,SAAS,iBAAiB,WAAW;AAC3C,gBAAI,WAAW,QAAW;AACxB,uBAAS,OAAO;AAChB,yBAAW,MAAM,aAAa,IAAI,EAAE,GAAG,SAAS;AAChD,oBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AACrD,mBAAK,uBAAuB;AAAA,YAC9B;AAAA,UACF;AAAA,QACF,OAAO;AAEL,gBAAM,WAAW,MAAM,iBAAiB,IAAI,MAAM,UAAU,KAAK;AACjE,gBAAM,cAAc,WAAW,MAAM;AACrC,gBAAM,iBAAiB,IAAI,MAAM,YAAY,WAAW;AAExD,gBAAM,gBAAgB,MAAM,MAAM;AAAA,YAChC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,WAAW,MAAM,MAAM,aAAa;AAC1C,kBAAM,SAAS,iBAAiB,WAAW;AAC3C,gBAAI,WAAW,QAAW;AACxB,uBAAS,OAAO;AAChB,oBAAM,MAAM,aAAa,IAAI,EAAE,GAAG,SAAS;AAC3C,mBAAK,uBAAuB;AAAA,YAC9B;AAAA,UACF;AAAA,QACF;AACA;AAAA,MACF;AAAA,MAEA,KAAK;AAEH;AAAA,MAEF,KAAK,wBAAwB;AAC3B,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AAEf,sBAAY,iBAAiB,OAAO,MAAM,UAAU;AAEpD,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,gBAAM,gBAAgB,WAAW,MAAM;AAAA,YACrC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,OAAO,WAAW,MAAM,aAAa;AAC3C,iBAAK,OAAO,MAAM;AAClB,iBAAK,SAAS;AACd,uBAAW,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AAC5C,kBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AACrD,iBAAK,uBAAuB;AAAA,UAC9B;AAAA,QACF,OAAO;AAEL,gBAAM,iBAAiB,OAAO,MAAM,UAAU;AAE9C,gBAAM,gBAAgB,MAAM,MAAM;AAAA,YAChC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,OAAO,MAAM,MAAM,aAAa;AACtC,iBAAK,OAAO,MAAM;AAClB,iBAAK,SAAS;AACd,kBAAM,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AACvC,iBAAK,uBAAuB;AAAA,UAC9B;AAAA,QACF;AACA;AAAA,MACF;AAAA,MAEA,KAAK,yBAAyB;AAC5B,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AACf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,gBAAM,gBAAgB,WAAW,MAAM;AAAA,YACrC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,OAAO,WAAW,MAAM,aAAa;AAC3C,iBAAK,SAAS,MAAM;AACpB,iBAAK,SAAS;AACd,uBAAW,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AAC5C,kBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AACrD,iBAAK,uBAAuB;AAAA,UAC9B;AAAA,QACF,OAAO;AACL,gBAAM,gBAAgB,MAAM,MAAM;AAAA,YAChC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,OAAO,MAAM,MAAM,aAAa;AACtC,iBAAK,SAAS,MAAM;AACpB,iBAAK,SAAS;AACd,kBAAM,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AACvC,iBAAK,uBAAuB;AAAA,UAC9B;AAAA,QACF;AACA;AAAA,MACF;AAAA,MAEA,KAAK,qBAAqB;AACxB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAEnE,YAAI,aAAa;AACf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,gBAAM,gBAAgB,WAAW,MAAM;AAAA,YACrC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,OAAO,WAAW,MAAM,aAAa;AAC3C,iBAAK,QAAQ,MAAM;AACnB,iBAAK,SAAS;AACd,uBAAW,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AAC5C,kBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AACrD,iBAAK,uBAAuB;AAAA,UAC9B;AAAA,QACF,OAAO;AACL,gBAAM,gBAAgB,MAAM,MAAM;AAAA,YAChC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,MAAM;AAAA,UACzE;AACA,cAAI,iBAAiB,GAAG;AACtB,kBAAM,OAAO,MAAM,MAAM,aAAa;AACtC,iBAAK,QAAQ,MAAM;AACnB,iBAAK,SAAS;AACd,kBAAM,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AACvC,iBAAK,uBAAuB;AAAA,UAC9B;AAAA,QACF;AACA;AAAA,MACF;AAAA,MAEA,KAAK,UAAU;AACb,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AACnE,cAAM,SAAS,cAAc,MAAM,aAAa,MAAM;AAEtD,YAAI;AACJ,YAAI,MAAM,eAAe,OAAO;AAC9B,uBAAa;AAAA,YACX,MAAM;AAAA,YACN,YAAY;AAAA,YACZ,IAAI,MAAM;AAAA,YACV,KAAK,MAAM;AAAA,YACX,OAAO,MAAM;AAAA,YACb;AAAA,UACF;AAAA,QACF,OAAO;AACL,uBAAa;AAAA,YACX,MAAM;AAAA,YACN,YAAY;AAAA,YACZ,IAAI,MAAM;AAAA,YACV,WAAW,MAAM;AAAA,YACjB,OAAO,MAAM;AAAA,YACb,UAAU,MAAM;AAAA,YAChB;AAAA,UACF;AAAA,QACF;AAEA,YAAI,aAAa;AACf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,qBAAW,MAAM,KAAK,UAAU;AAChC,gBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,QACvD,OAAO;AACL,gBAAM,MAAM,KAAK,UAAU;AAAA,QAC7B;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,kBAAkB;AACrB,cAAM,WAAW,MAAM;AACvB,cAAM,cAAc,WAAW,MAAM,cAAc,IAAI,QAAQ,IAAI;AAGnE,cAAM,WAAuB;AAAA,UAC3B,MAAM;AAAA,UACN,IAAI,MAAM;AAAA,UACV,WAAW,MAAM;AAAA,UACjB,KAAK,MAAM;AAAA,UACX,UAAU,MAAM;AAAA,UAChB,MAAM,MAAM;AAAA,UACZ,YAAY,MAAM;AAAA,UAClB,QAAQ,cAAc,MAAM,aAAa,MAAM;AAAA,QACjD;AAEA,YAAI,aAAa;AACf,gBAAM,aAAa,MAAM,MAAM,YAAY,SAAS;AACpD,qBAAW,MAAM,KAAK,QAAQ;AAC9B,gBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,WAAW;AAAA,QACvD,OAAO;AACL,gBAAM,MAAM,KAAK,QAAQ;AAAA,QAC3B;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK;AACH,aAAK,QAAQ,mBAAmB,MAAM,MAAM,MAAM,KAAK;AACvD;AAAA,MAEF,KAAK,gBAAgB;AAEnB,cAAM,gBAAgB,MAAM,MAAM;AAAA,UAChC,CAAC,MAAM,EAAE,SAAS,YAAY,EAAE,aAAa,MAAM;AAAA,QACrD;AAEA,YAAI;AACJ,YAAI,kBAAkB,IAAI;AAExB,gBAAM,eAAe,MAAM,MAAM,aAAa;AAC9C,uBAAa,SAAS;AACtB,sBAAY;AAAA,QACd,OAAO;AAEL,gBAAM,aAA2B;AAAA,YAC/B,MAAM;AAAA,YACN,UAAU,MAAM;AAAA,YAChB,YAAY,MAAM;AAAA,YAClB,aAAa,MAAM;AAAA,YACnB,OAAO,CAAC;AAAA,YACR,QAAQ;AAAA,UACV;AACA,gBAAM,MAAM,KAAK,UAAU;AAC3B,sBAAY,MAAM,MAAM,SAAS;AAAA,QACnC;AAGA,cAAM,cAA+B;AAAA,UACnC;AAAA,UACA,sBAAsB;AAAA,UACtB,2BAA2B;AAAA,UAC3B,wBAAwB;AAAA,UACxB,iBAAiB;AAAA,UACjB,kBAAkB,oBAAI,IAAI;AAAA,QAC5B;AACA,cAAM,cAAc,IAAI,MAAM,UAAU,WAAW;AACnD,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,iBAAiB;AACpB,cAAM,cAAc,MAAM,cAAc,IAAI,MAAM,QAAQ;AAC1D,YAAI,gBAAgB,QAAW;AAC7B,gBAAM,OAAO,MAAM,MAAM,YAAY,SAAS;AAC9C,eAAK,SAAS,MAAM;AACpB,eAAK,QAAQ,MAAM;AACnB,eAAK,SAAS,MAAM,QAAQ,UAAU;AAGtC,eAAK,QAAQ,KAAK,MAAM,IAAI,CAAC,MAAqB;AAChD,gBAAI,EAAE,SAAS,UAAU,EAAE,SAAS,aAAa;AAC/C,kBAAI,EAAE,WAAW,aAAa;AAC5B,uBAAO,EAAE,GAAG,GAAG,QAAQ,OAAO;AAAA,cAChC;AAAA,YACF;AACA,gBAAI,EAAE,SAAS,YAAY,EAAE,WAAW,aAAa;AACnD,qBAAO,EAAE,GAAG,GAAG,QAAQ,OAAO;AAAA,YAChC;AACA,mBAAO;AAAA,UACT,CAAC;AAED,gBAAM,MAAM,YAAY,SAAS,IAAI,EAAE,GAAG,KAAK;AAC/C,gBAAM,cAAc,OAAO,MAAM,QAAQ;AAAA,QAC3C;AACA,aAAK,uBAAuB;AAC5B;AAAA,MACF;AAAA,MAEA,KAAK,UAAU;AAEb,YAAI,MAAM,iBAAiB,qBAAqB;AAE9C,eAAK,uBAAuB;AAE5B,cAAI,KAAK,sBAAsB,OAAO,GAAG;AACvC,iBAAK,UAAU,gBAAgB;AAAA,UACjC,WAAW,KAAK,kBAAkB;AAEhC,iBAAK,mBAAmB;AACxB,iBAAK,uBAAuB;AAC5B,iBAAK,KAAK,8BAA8B;AAAA,UAC1C;AACA;AAAA,QACF;AAEA,cAAM,eAAe,sBAAsB,OAAO,MAAM;AAExD,qBAAa,QAAQ,aAAa,MAAM,IAAI,CAAC,SAAS;AACpD,cAAI,KAAK,SAAS,UAAU,KAAK,SAAS,aAAa;AACrD,mBAAO,EAAE,GAAG,MAAM,QAAQ,OAAgB;AAAA,UAC5C;AACA,cAAI,KAAK,SAAS,YAAY,KAAK,WAAW,aAAa;AACzD,mBAAO,EAAE,GAAG,MAAM,QAAQ,OAAgB;AAAA,UAC5C;AACA,iBAAO;AAAA,QACT,CAAC;AAED,YAAI,aAAa,MAAM,SAAS,GAAG;AACjC,gBAAM,WAAW,CAAC,GAAG,KAAK,SAAS;AACnC,gBAAM,UAAU,SAAS,SAAS,SAAS,CAAC;AAC5C,cAAI,SAAS,OAAO,MAAM,WAAW;AACnC,qBAAS,SAAS,SAAS,CAAC,IAAI;AAAA,UAClC,OAAO;AACL,qBAAS,KAAK,YAAY;AAAA,UAC5B;AACA,eAAK,YAAY,QAAQ;AAAA,QAC3B;AAEA,aAAK,UAAU,MAAM;AACrB,aAAK,iBAAiB;AACtB,aAAK,QAAQ,WAAW;AACxB;AAAA,MACF;AAAA,MAEA,KAAK,SAAS;AAEZ,cAAM,IAAI,aAAa;AAAA,UACrB,WAAW,MAAM;AAAA,UACjB,SAAS,MAAM;AAAA,UACf,QAAQ,MAAM;AAAA,UACd,WAAW,MAAM;AAAA,UACjB,YAAY,MAAM;AAAA,UAClB,MAAM,MAAM;AAAA,UACZ,UAAU,MAAM;AAAA,UAChB,MAAM,MAAM;AAAA,QACd,CAAC;AAAA,MACH;AAAA,MAEA,KAAK;AAEH;AAAA,MAEF,KAAK;AAEH,aAAK,sBAAsB,MAAM;AACjC,aAAK,qBAAqB,MAAM,qBAAqB,CAAC;AAEtD,aAAK,KAAK,wBAAwB,MAAM,WAAW,KAAK;AACxD;AAAA,IACJ;AAAA,EACF;AAAA,EAEQ,yBAA+B;AACrC,UAAM,QAAQ,KAAK;AACnB,QAAI,CAAC,MAAO;AAEZ,UAAM,MAAM,sBAAsB,OAAO,WAAW;AACpD,UAAM,WAAW,CAAC,GAAG,KAAK,SAAS;AAEnC,UAAM,UAAU,SAAS,SAAS,SAAS,CAAC;AAC5C,QAAI,SAAS,OAAO,MAAM,WAAW;AACnC,eAAS,SAAS,SAAS,CAAC,IAAI;AAAA,IAClC,OAAO;AACL,eAAS,KAAK,GAAG;AAAA,IACnB;AAEA,SAAK,YAAY,QAAQ;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA,EAKQ,wBAAwB,YAAoB,QAAuB;AACzE,UAAM,QAAQ,KAAK;AACnB,QAAI,CAAC,MAAO;AAEZ,UAAM,gBAAgB,MAAM,MAAM;AAAA,MAChC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe;AAAA,IACnE;AACA,QAAI,iBAAiB,GAAG;AACtB,YAAM,OAAO,MAAM,MAAM,aAAa;AACtC,WAAK,SAAS;AACd,WAAK,SAAS;AACd,YAAM,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AACvC,WAAK,uBAAuB;AAAA,IAC9B;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAAoB,YAAoB,OAAqB;AACnE,UAAM,QAAQ,KAAK;AACnB,QAAI,CAAC,MAAO;AAEZ,UAAM,gBAAgB,MAAM,MAAM;AAAA,MAChC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe;AAAA,IACnE;AACA,QAAI,iBAAiB,GAAG;AACtB,YAAM,OAAO,MAAM,MAAM,aAAa;AACtC,WAAK,QAAQ;AACb,WAAK,SAAS;AACd,YAAM,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AACvC,WAAK,uBAAuB;AAAA,IAC9B;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,gCAA+C;AAC3D,QAAI,KAAK,sBAAsB,WAAW,EAAG;AAE7C,QAAI,KAAK,wBAAwB,MAAM;AAErC,YAAM,WAAW,IAAI,aAAa;AAAA,QAChC,WAAW;AAAA,QACX,SAAS;AAAA,QACT,QAAQ;AAAA,QACR,WAAW;AAAA,MACb,CAAC;AACD,WAAK,SAAS,QAAQ;AACtB,WAAK,UAAU,OAAO;AACtB,WAAK,QAAQ,UAAU,QAAQ;AAC/B;AAAA,IACF;AAGA,UAAM,aAAa,CAAC,GAAG,KAAK,oBAAoB,GAAG,KAAK,qBAAqB;AAC7E,UAAM,cAAc,KAAK;AACzB,SAAK,qBAAqB,CAAC;AAC3B,SAAK,wBAAwB,CAAC;AAC9B,SAAK,sBAAsB;AAE3B,SAAK,UAAU,WAAW;AAE1B,QAAI;AAEF,uBAAiB,SAAS,KAAK,UAAU,wBAAwB,aAAa,UAAU,GAAG;AACzF,YAAI,KAAK,mBAAmB,KAAM;AAClC,aAAK,kBAAkB,OAAO,KAAK,cAAc;AAAA,MACnD;AAAA,IACF,SAAS,KAAK;AACZ,YAAM,WAAW,aAAa,WAAW,GAAG,IACxC,MACA,IAAI,aAAa;AAAA,QACf,WAAW;AAAA,QACX,SAAS,eAAe,QAAQ,IAAI,UAAU;AAAA,QAC9C,QAAQ;AAAA,QACR,WAAW;AAAA,QACX,OAAO;AAAA,MACT,CAAC;AAEL,WAAK,SAAS,QAAQ;AACtB,WAAK,UAAU,OAAO;AACtB,WAAK,iBAAiB;AACtB,WAAK,QAAQ,UAAU,QAAQ;AAAA,IACjC;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAc,wBACZ,WACA,OACe;AACf,SAAK,6BAA6B,IAAI,gBAAgB;AAItD,eAAW,MAAM,WAAW;AAC1B,YAAM,UAAU,KAAK,QAAQ,cAAc,GAAG,QAAQ;AACtD,UAAI,YAAY,eAAe;AAC7B,cAAM,YAA8B;AAAA,UAClC,YAAY,GAAG;AAAA,UACf,UAAU,GAAG;AAAA,UACb,MAAM,GAAG;AAAA,UACT,QAAQ,GAAG;AAAA,UACX,gBAAgB,GAAG;AAAA,UACnB,YAAY,GAAG;AAAA,UACf,QAAQ,GAAG;AAAA,UACX,UAAU,GAAG;AAAA,QACf;AAEA,aAAK,sBAAsB,IAAI,GAAG,YAAY,SAAS;AACvD,cAAM,WAAW,KAAK,oBAAoB,IAAI,GAAG,QAAQ,KAAK,CAAC;AAC/D,aAAK,oBAAoB,IAAI,GAAG,UAAU,CAAC,GAAG,UAAU,SAAS,CAAC;AAAA,MACpE;AAAA,IACF;AACA,QAAI,KAAK,sBAAsB,OAAO,GAAG;AACvC,WAAK,8BAA8B;AAAA,IACrC;AAGA,eAAW,MAAM,WAAW;AAC1B,YAAM,UAAU,KAAK,QAAQ,cAAc,GAAG,QAAQ;AAEtD,UAAI,YAAY,eAAe;AAE7B,cAAM,gBAAgB,MAAM,MAAM;AAAA,UAChC,CAAC,MAAqB,EAAE,SAAS,eAAe,EAAE,eAAe,GAAG;AAAA,QACtE;AACA,YAAI,iBAAiB,GAAG;AACtB,gBAAM,OAAO,MAAM,MAAM,aAAa;AAEtC,gBAAM,MAAM,aAAa,IAAI,EAAE,GAAG,KAAK;AAAA,QACzC;AAAA,MACF,WAAW,SAAS;AAClB,YAAI;AACF,gBAAM,iBAAkC,CAAC;AACzC,gBAAM,SAAS,MAAM,QAAQ,GAAG,MAAM;AAAA,YACpC,YAAY,GAAG;AAAA,YACf,UAAU,GAAG;AAAA,YACb,QAAQ,KAAK,2BAA2B;AAAA,YACxC,SAAS,CAAC,SAAS,eAAe,KAAK,IAAI;AAAA,UAC7C,CAAC;AAED,eAAK,sBAAsB,KAAK;AAAA,YAC9B,YAAY,GAAG;AAAA,YACf,UAAU,GAAG;AAAA,YACb;AAAA,YACA,OAAO,eAAe,SAAS,IAAI,iBAAiB;AAAA,YACpD,gBAAgB,GAAG;AAAA,YACnB,YAAY,GAAG;AAAA,YACf,QAAQ,GAAG;AAAA,YACX,UAAU,GAAG;AAAA,UACf,CAAC;AAED,eAAK,wBAAwB,GAAG,YAAY,MAAM;AAAA,QACpD,SAAS,KAAK;AACZ,gBAAM,eAAe,eAAe,QAAQ,IAAI,UAAU;AAC1D,eAAK,sBAAsB,KAAK;AAAA,YAC9B,YAAY,GAAG;AAAA,YACf,UAAU,GAAG;AAAA,YACb,OAAO;AAAA,YACP,gBAAgB,GAAG;AAAA,YACnB,YAAY,GAAG;AAAA,YACf,QAAQ,GAAG;AAAA,YACX,UAAU,GAAG;AAAA,UACf,CAAC;AAED,eAAK,oBAAoB,GAAG,YAAY,YAAY;AAAA,QACtD;AAAA,MACF,OAAO;AAEL,cAAM,eAAe,+BAA+B,GAAG,QAAQ;AAC/D,aAAK,sBAAsB,KAAK;AAAA,UAC9B,YAAY,GAAG;AAAA,UACf,UAAU,GAAG;AAAA,UACb,OAAO;AAAA,UACP,gBAAgB,GAAG;AAAA,UACnB,YAAY,GAAG;AAAA,UACf,QAAQ,GAAG;AAAA,UACX,UAAU,GAAG;AAAA,QACf,CAAC;AAED,aAAK,oBAAoB,GAAG,YAAY,YAAY;AAAA,MACtD;AAAA,IACF;AAKA,QAAI,KAAK,sBAAsB,SAAS,KAAK,KAAK,sBAAsB,SAAS,GAAG;AAClF,WAAK,mBAAmB;AAIxB,UAAI,KAAK,sBAAsB;AAC7B,aAAK,mBAAmB;AACxB,aAAK,uBAAuB;AAC5B,aAAK,KAAK,8BAA8B;AAAA,MAC1C;AAAA,IACF;AAAA,EACF;AACF;;;AEpzDA,SAAS,sBAAsB,oBAAsC;AAQrE,gBAAuB,eACrB,UACA,QAC4C;AAC5C,QAAM,SAAS,SAAS,MAAM,UAAU;AACxC,MAAI,CAAC,QAAQ;AACX,UAAM,IAAI,MAAM,+BAA+B;AAAA,EACjD;AAEA,QAAM,UAAU,IAAI,YAAY;AAChC,MAAI,SAAS;AAEb,MAAI;AACF,QAAI,UAAU;AACd,WAAO,SAAS;AAEd,UAAI,QAAQ,SAAS;AACnB;AAAA,MACF;AAEA,UAAI;AACJ,UAAI;AACF,qBAAa,MAAM,OAAO,KAAK;AAAA,MACjC,SAAS,KAAK;AAEZ,YAAI,aAAa,GAAG,GAAG;AACrB;AAAA,QACF;AACA,cAAM;AAAA,MACR;AAEA,YAAM,EAAE,MAAM,MAAM,IAAI;AAExB,UAAI,MAAM;AACR,kBAAU;AACV;AAAA,MACF;AAEA,gBAAU,QAAQ,OAAO,OAAO,EAAE,QAAQ,KAAK,CAAC;AAChD,YAAM,QAAQ,OAAO,MAAM,IAAI;AAC/B,eAAS,MAAM,IAAI,KAAK;AAExB,iBAAW,QAAQ,OAAO;AACxB,YAAI,KAAK,WAAW,QAAQ,KAAK,SAAS,gBAAgB;AACxD,cAAI;AACF,kBAAM,SAAS,qBAAqB,KAAK,MAAM,KAAK,MAAM,CAAC,CAAC,CAAC;AAC7D,gBAAI,OAAO,SAAS;AAClB,oBAAM,OAAO;AAAA,YACf;AAAA,UAEF,QAAQ;AAAA,UAER;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF,UAAE;AACA,WAAO,YAAY;AAAA,EACrB;AACF;;;ACwEO,SAAS,kBAAkB,WAAoD;AACpF,SACE,aAAa,aACb,gBAAgB,aAChB,qBAAqB,aACrB,6BAA6B;AAEjC;;;AClJA,SAAS,gBAAAA,qBAAqC;AAkFvC,SAAS,oBAAoB,SAA0C;AAC5E,MAAI,kBAA0C;AAE9C,kBAAgB,eAAe,iBAAoC;AACjE,QAAI;AACF,YAAM,WAAW,MAAM;AAEvB,UAAI,CAAC,SAAS,IAAI;AAChB,cAAM,YAAY,MAAM,SAAS,KAAK,EAAE,MAAM,MAAM,mBAAmB,SAAS,MAAM,EAAE;AACxF,cAAM,IAAI,MAAM,SAAS;AAAA,MAC3B;AAEA,UAAI,CAAC,SAAS,MAAM;AAClB,cAAM,IAAI,MAAM,wBAAwB;AAAA,MAC1C;AAEA,uBAAiB,SAAS,eAAe,UAAU,gBAAiB,MAAM,GAAG;AAC3E,YAAI,iBAAiB,OAAO,SAAS;AACnC;AAAA,QACF;AACA,cAAM;AAAA,MACR;AAAA,IACF,SAAS,KAAK;AACZ,UAAIC,cAAa,GAAG,GAAG;AACrB;AAAA,MACF;AACA,YAAM;AAAA,IACR;AAAA,EACF;AAEA,SAAO;AAAA,IACL,OAAO,QAAQ,aAAa,OAAO;AACjC,wBAAkB,IAAI,gBAAgB;AACtC,YAAM,WAAW,QAAQ;AAAA,QACvB,EAAE,MAAM,WAAW,aAAa,MAAM;AAAA,QACtC,EAAE,QAAQ,gBAAgB,OAAO;AAAA,MACnC;AACA,aAAO,eAAe,QAAQ;AAAA,IAChC;AAAA,IAEA,OAAO,wBAAwB,aAAa,aAAa;AACvD,wBAAkB,IAAI,gBAAgB;AACtC,YAAM,WAAW,QAAQ;AAAA,QACvB,EAAE,MAAM,YAAY,aAAa,YAAY;AAAA,QAC7C,EAAE,QAAQ,gBAAgB,OAAO;AAAA,MACnC;AACA,aAAO,eAAe,QAAQ;AAAA,IAChC;AAAA,IAEA,OAAO;AACL,uBAAiB,MAAM;AACvB,wBAAkB;AAAA,IACpB;AAAA,EACF;AACF;;;ACxIA,SAAS,wBAAAC,6BAA+D;AAsBxE,IAAM,cAAc;AAgHb,SAAS,sBAAsB,SAAkD;AACtF,MAAI,SAA4B;AAChC,MAAI,aAA4B,CAAC;AACjC,MAAI,gBAA8D;AAClE,MAAI,cAAc;AAElB,MAAI,kBAAmC;AACvC,MAAI;AACJ,MAAI,oBAA0C;AAC9C,QAAM,sBAAsB,oBAAI,IAA6B;AAE7D,WAAS,mBAAmB,OAAwB,OAAe;AACjE,sBAAkB;AAClB,sBAAkB;AAClB,wBAAoB,QAAQ,CAAC,aAAa,SAAS,OAAO,KAAK,CAAC;AAAA,EAClE;AAEA,WAAS,aAAa,OAAoB;AACxC,QAAI,eAAe;AACjB,oBAAc,KAAK;AACnB,sBAAgB;AAAA,IAClB,OAAO;AACL,iBAAW,KAAK,KAAK;AAAA,IACvB;AAAA,EACF;AAEA,WAAS,YAAyC;AAChD,QAAI,WAAW,SAAS,GAAG;AACzB,aAAO,QAAQ,QAAQ,WAAW,MAAM,CAAE;AAAA,IAC5C;AACA,QAAI,CAAC,aAAa;AAChB,aAAO,QAAQ,QAAQ,IAAI;AAAA,IAC7B;AACA,WAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,sBAAgB;AAAA,IAClB,CAAC;AAAA,EACH;AAEA,WAAS,oBAAoB,MAAwB;AACnD,SAAK,YAAY,CAAC,MAAoB;AACpC,UAAI;AACF,cAAM,OAAgB,OAAO,EAAE,SAAS,WAAW,KAAK,MAAM,EAAE,IAAI,IAAI,EAAE;AAE1E,gBAAQ,YAAY,IAAI;AAExB,cAAM,SAASA,sBAAqB,IAAI;AACxC,YAAI,OAAO,SAAS;AAClB,gBAAM,QAAQ,OAAO;AACrB,uBAAa,KAAK;AAElB,cAAI,MAAM,SAAS,YAAY,MAAM,SAAS,SAAS;AACrD,0BAAc;AAAA,UAChB;AAAA,QACF;AAAA,MACF,QAAQ;AAAA,MAER;AAAA,IACF;AAEA,SAAK,UAAU,MAAM;AACnB,eAAS;AACT,0BAAoB;AACpB,yBAAmB,cAAc;AACjC,cAAQ,UAAU;AAElB,oBAAc;AACd,UAAI,eAAe;AACjB,sBAAc,IAAI;AAClB,wBAAgB;AAAA,MAClB;AAAA,IACF;AAAA,EACF;AAEA,iBAAe,kBAAiC;AAE9C,QAAI,QAAQ,eAAe,aAAa;AACtC;AAAA,IACF;AAGA,QAAI,mBAAmB;AACrB,YAAM;AACN;AAAA,IACF;AAGA,uBAAmB,YAAY;AAE/B,yBAAqB,YAAY;AAC/B,UAAI;AACF,cAAM,OAAO,MAAM,QAAQ,QAAQ;AACnC,iBAAS;AACT,4BAAoB,IAAI;AACxB,2BAAmB,WAAW;AAAA,MAChC,SAAS,KAAK;AACZ,iBAAS;AACT,4BAAoB;AACpB,cAAM,QAAQ,eAAe,QAAQ,MAAM,IAAI,MAAM,mBAAmB;AACxE,2BAAmB,SAAS,KAAK;AACjC,cAAM;AAAA,MACR;AAAA,IACF,GAAG;AAEH,UAAM;AAAA,EACR;AAEA,SAAO;AAAA;AAAA;AAAA;AAAA,IAKL,IAAI,kBAAmC;AACrC,aAAO;AAAA,IACT;AAAA,IAEA,wBAAwB,UAA+C;AACrE,0BAAoB,IAAI,QAAQ;AAEhC,eAAS,iBAAiB,eAAe;AACzC,aAAO,MAAM,oBAAoB,OAAO,QAAQ;AAAA,IAClD;AAAA,IAEA,MAAM,UAAyB;AAC7B,YAAM,gBAAgB;AAAA,IACxB;AAAA,IAEA,aAAmB;AACjB,UAAI,QAAQ;AACV,eAAO,MAAM;AACb,iBAAS;AAAA,MACX;AACA,0BAAoB;AACpB,oBAAc;AACd,UAAI,eAAe;AACjB,sBAAc,IAAI;AAClB,wBAAgB;AAAA,MAClB;AACA,yBAAmB,cAAc;AAAA,IACnC;AAAA;AAAA;AAAA;AAAA,IAMA,OAAO,QAAQ,aAAa,OAAO;AACjC,YAAM,gBAAgB;AAEtB,mBAAa,CAAC;AACd,sBAAgB;AAChB,oBAAc;AAGd,aAAQ;AAAA,QACN,KAAK,UAAU;AAAA,UACb,MAAM;AAAA,UACN;AAAA,UACA;AAAA,QACF,CAAC;AAAA,MACH;AAEA,aAAO,MAAM;AACX,cAAM,QAAQ,MAAM,UAAU;AAC9B,YAAI,UAAU,KAAM;AACpB,cAAM;AACN,YAAI,MAAM,SAAS,YAAY,MAAM,SAAS,QAAS;AAAA,MACzD;AAAA,IACF;AAAA,IAEA,OAAO;AACL,UAAI,QAAQ,eAAe,aAAa;AACtC,eAAO,KAAK,KAAK,UAAU,EAAE,MAAM,OAAO,CAAC,CAAC;AAAA,MAC9C;AACA,oBAAc;AACd,UAAI,eAAe;AACjB,sBAAc,IAAI;AAClB,wBAAgB;AAAA,MAClB;AAAA,IACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAOA,OAAO,wBAAwB,aAAqB,aAA2B;AAC7E,YAAM,gBAAgB;AAEtB,mBAAa,CAAC;AACd,sBAAgB;AAChB,oBAAc;AAEd,aAAQ;AAAA,QACN,KAAK,UAAU;AAAA,UACb,MAAM;AAAA,UACN;AAAA,UACA;AAAA,QACF,CAAC;AAAA,MACH;AAEA,aAAO,MAAM;AACX,cAAM,QAAQ,MAAM,UAAU;AAC9B,YAAI,UAAU,KAAM;AACpB,cAAM;AACN,YAAI,MAAM,SAAS,YAAY,MAAM,SAAS,QAAS;AAAA,MACzD;AAAA,IACF;AAAA,EACF;AACF;;;ACpTA;AAAA,EAEE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA,gBAAAC;AAAA,EAEA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAEA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAEA,cAAAC;AAAA,EACA,gBAAAC;AAAA,EAEA;AAAA,EACA;AAAA,EACA;AAAA,EACA,iBAAAC;AAAA,EACA;AAAA,EAEA;AAAA,EACA,wBAAAC;AAAA,EAEA,wBAAAC;AAAA,EACA;AAAA,EACA;AAAA,EAEA;AAAA,EACA;AAAA,EACA;AAAA,OACK;","names":["isAbortError","isAbortError","safeParseStreamEvent","OctavusError","generateId","isAbortError","threadForPart","isFileReferenceArray","safeParseStreamEvent"]}
|
|
1
|
+
{"version":3,"sources":["../src/chat.ts","../src/files.ts","../src/stream/reader.ts","../src/transports/types.ts","../src/transports/http.ts","../src/transports/socket.ts","../src/index.ts"],"sourcesContent":["import {\n generateId,\n threadForPart,\n isFileReferenceArray,\n OctavusError,\n type UIMessage,\n type UIMessagePart,\n type UITextPart,\n type UIReasoningPart,\n type UIToolCallPart,\n type UIOperationPart,\n type UISourcePart,\n type UIFilePart,\n type UIObjectPart,\n type UIWorkerPart,\n type DisplayMode,\n type StreamEvent,\n type FileReference,\n type PendingToolCall,\n type ToolResult,\n} from '@octavus/core';\nimport type { Transport } from './transports/types';\nimport { uploadFiles, type UploadFilesOptions } from './files';\n\n/** Block types that are internal operations (not LLM-driven) */\nconst OPERATION_BLOCK_TYPES = new Set(['set-resource', 'serialize-thread', 'generate-image']);\n\n// =============================================================================\n// Types\n// =============================================================================\n\nexport type ChatStatus = 'idle' | 'streaming' | 'error' | 'awaiting-input';\n\n/**\n * Context provided to client tool handlers.\n */\nexport interface ClientToolContext {\n /** Unique identifier for this tool call */\n toolCallId: string;\n /** Name of the tool being called */\n toolName: string;\n /** Signal for cancellation if user stops generation */\n signal: AbortSignal;\n /**\n * Register a file produced by this tool (e.g., a screenshot).\n * Files are sent to the platform alongside the tool result so the LLM\n * can see them as visual content rather than just a JSON URL.\n */\n addFile: (file: FileReference) => void;\n}\n\n/**\n * Handler function for client-side tool execution.\n * Can be:\n * - An async function that executes automatically and returns a result\n * - The string 'interactive' to indicate the tool requires user interaction\n */\nexport type ClientToolHandler =\n | ((args: Record<string, unknown>, ctx: ClientToolContext) => Promise<unknown>)\n | 'interactive';\n\n/**\n * Interactive tool call awaiting user interaction.\n * The `submit` and `cancel` methods are pre-bound to this tool call's ID.\n */\nexport interface InteractiveTool {\n /** Unique identifier for this tool call */\n toolCallId: string;\n /** Name of the tool being called */\n toolName: string;\n /** Arguments passed to the tool */\n args: Record<string, unknown>;\n /**\n * Submit a result for this tool call.\n * Call this when the user has provided input.\n *\n * @param result - The result from user interaction\n */\n submit: (result: unknown) => void;\n /**\n * Cancel this tool call with an optional reason.\n * Call this when the user dismisses the UI without providing input.\n *\n * @param reason - Optional reason for cancellation (default: 'User cancelled')\n */\n cancel: (reason?: string) => void;\n}\n\n/**\n * Internal pending tool state (before binding submit/cancel).\n */\ninterface PendingToolState {\n toolCallId: string;\n toolName: string;\n args: Record<string, unknown>;\n source?: 'llm' | 'block';\n outputVariable?: string;\n blockIndex?: number;\n thread?: string;\n /** Worker ID if this tool call is from a worker execution */\n workerId?: string;\n}\n\n/**\n * Input for creating a user message.\n * Supports text content, structured object content, and file attachments.\n */\nexport interface UserMessageInput {\n /**\n * Content of the message. Can be:\n * - string: Creates a text part\n * - object: Creates an object part (uses `type` field as typeName if present)\n */\n content?: string | Record<string, unknown>;\n /**\n * File attachments (shorthand). Can be:\n * - FileList: From file input element (will be uploaded via uploadFiles)\n * - File[]: Array of File objects (will be uploaded via uploadFiles)\n * - FileReference[]: Already uploaded files (used directly)\n */\n files?: FileList | File[] | FileReference[];\n}\n\nexport interface OctavusChatOptions {\n /**\n * Transport for streaming events.\n * Use `createHttpTransport` for HTTP/SSE or `createSocketTransport` for WebSocket/SockJS.\n */\n transport: Transport;\n\n /**\n * Function to request upload URLs from the platform.\n * Required if you want to use file uploads with FileList/File[].\n *\n * @example\n * ```typescript\n * requestUploadUrls: async (files) => {\n * const response = await fetch('/api/upload-urls', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, files }),\n * });\n * return response.json();\n * }\n * ```\n */\n requestUploadUrls?: UploadFilesOptions['requestUploadUrls'];\n\n /**\n * Client-side tool handlers.\n * Register handlers for tools that should execute in the browser.\n *\n * - If a tool has a handler function: executes automatically\n * - If a tool is marked as 'interactive': appears in `pendingClientTools` with bound `submit()`/`cancel()`\n *\n * @example Automatic client tool\n * ```typescript\n * clientTools: {\n * 'get-browser-location': async () => {\n * const pos = await new Promise((resolve, reject) => {\n * navigator.geolocation.getCurrentPosition(resolve, reject);\n * });\n * return { lat: pos.coords.latitude, lng: pos.coords.longitude };\n * },\n * }\n * ```\n *\n * @example Interactive client tool (user input required)\n * ```typescript\n * clientTools: {\n * 'request-feedback': 'interactive',\n * }\n * // Then render UI based on pendingClientTools['request-feedback']\n * // and call tool.submit(result) or tool.cancel()\n * ```\n */\n clientTools?: Record<string, ClientToolHandler>;\n\n /** Initial messages (for session refresh) */\n initialMessages?: UIMessage[];\n /**\n * Callback when an error occurs.\n * Receives an OctavusError with structured error information.\n *\n * @example\n * ```typescript\n * onError: (error) => {\n * console.error('Chat error:', {\n * type: error.errorType,\n * message: error.message,\n * retryable: error.retryable,\n * provider: error.provider,\n * });\n *\n * // Handle specific error types\n * if (isRateLimitError(error)) {\n * showRetryButton(error.retryAfter);\n * }\n * }\n * ```\n */\n onError?: (error: OctavusError) => void;\n /** Callback when streaming finishes successfully */\n onFinish?: () => void;\n /** Callback when streaming is stopped by user */\n onStop?: () => void;\n /** Callback when a resource is updated */\n onResourceUpdate?: (name: string, value: unknown) => void;\n /**\n * Callback when execution starts with the session/execution ID.\n * Useful for tracking the current execution for activity logs.\n *\n * @example\n * ```typescript\n * onStart: (sessionId) => {\n * setCurrentSessionId(sessionId);\n * }\n * ```\n */\n onStart?: (sessionId: string) => void;\n}\n\n// =============================================================================\n// Internal Types\n// =============================================================================\n\ninterface BlockState {\n blockId: string;\n blockName: string;\n blockType: string;\n display: DisplayMode;\n description?: string;\n outputToChat: boolean;\n thread?: string;\n reasoning: string;\n text: string;\n toolCalls: Map<string, UIToolCallPart>;\n}\n\n/** Tracks state for a worker part being populated */\ninterface WorkerPartState {\n partIndex: number;\n currentTextPartIndex: number | null;\n currentReasoningPartIndex: number | null;\n currentObjectPartIndex: number | null;\n accumulatedJson: string;\n /** Accumulated raw JSON text per tool call ID for progressive partial parsing */\n toolInputBuffers: Map<string, string>;\n}\n\ninterface StreamingState {\n messageId: string;\n parts: UIMessagePart[];\n activeBlock: BlockState | null;\n blocks: Map<string, BlockState>;\n currentTextPartIndex: number | null;\n currentReasoningPartIndex: number | null;\n currentObjectPartIndex: number | null;\n accumulatedJson: string;\n /** Active workers being populated: workerId -> worker state */\n activeWorkers: Map<string, WorkerPartState>;\n /** Accumulated raw JSON text per tool call ID for progressive partial parsing */\n toolInputBuffers: Map<string, string>;\n}\n\ntype Listener = () => void;\n\n// =============================================================================\n// Helpers\n// =============================================================================\n\n/**\n * Create a user message from input with optional file attachments.\n * Parts order: files first (for vision models), then content (text or object).\n */\nfunction createUserMessage(input: UserMessageInput, files?: FileReference[]): UIMessage {\n const parts: UIMessagePart[] = [];\n\n // Add file parts first (vision models expect images before text)\n if (files && files.length > 0) {\n for (const file of files) {\n parts.push({\n type: 'file',\n id: file.id,\n mediaType: file.mediaType,\n url: file.url,\n filename: file.filename,\n size: file.size,\n });\n }\n }\n\n // Add content part after files\n if (input.content !== undefined) {\n if (typeof input.content === 'string') {\n // String content → text part\n parts.push({ type: 'text', text: input.content, status: 'done' });\n } else {\n // Object content → object part\n // Use the object's `type` field as typeName if present, otherwise fallback to 'object'\n const typeName = (input.content as { type?: string }).type ?? 'object';\n parts.push({\n type: 'object',\n id: generateId(),\n typeName,\n object: input.content,\n status: 'done',\n });\n }\n }\n\n return {\n id: generateId(),\n role: 'user',\n parts,\n status: 'done',\n createdAt: new Date(),\n };\n}\n\n/**\n * Parse partial JSON by fixing incomplete structures (unclosed strings, brackets, braces).\n */\nfunction parsePartialJson(jsonText: string): unknown {\n if (!jsonText.trim()) {\n return undefined;\n }\n\n try {\n return JSON.parse(jsonText) as unknown;\n } catch {\n // Continue to fix incomplete JSON\n }\n\n let fixed = jsonText;\n\n // Count unclosed brackets/braces while tracking string boundaries\n let openBraces = 0;\n let openBrackets = 0;\n let inString = false;\n let escaped = false;\n\n for (const char of fixed) {\n if (escaped) {\n escaped = false;\n continue;\n }\n\n if (char === '\\\\') {\n escaped = true;\n continue;\n }\n\n if (char === '\"') {\n inString = !inString;\n continue;\n }\n\n if (!inString) {\n if (char === '{') openBraces += 1;\n else if (char === '}') openBraces -= 1;\n else if (char === '[') openBrackets += 1;\n else if (char === ']') openBrackets -= 1;\n }\n }\n\n // Close unclosed structures\n if (escaped) {\n // If input ends with a dangling backslash, complete the escape sequence.\n fixed += '\\\\';\n }\n if (inString) {\n fixed += '\"';\n }\n while (openBrackets > 0) {\n fixed += ']';\n openBrackets -= 1;\n }\n while (openBraces > 0) {\n fixed += '}';\n openBraces -= 1;\n }\n\n try {\n return JSON.parse(fixed) as unknown;\n } catch {\n return undefined;\n }\n}\n\nfunction createEmptyStreamingState(): StreamingState {\n return {\n messageId: generateId(),\n parts: [],\n activeBlock: null,\n blocks: new Map(),\n currentTextPartIndex: null,\n currentReasoningPartIndex: null,\n currentObjectPartIndex: null,\n accumulatedJson: '',\n activeWorkers: new Map(),\n toolInputBuffers: new Map(),\n };\n}\n\nfunction buildMessageFromState(state: StreamingState, status: 'streaming' | 'done'): UIMessage {\n return {\n id: state.messageId,\n role: 'assistant',\n parts: [...state.parts],\n status,\n createdAt: new Date(),\n };\n}\n\n/**\n * Finalize parts when stream is stopped or errors.\n * Marks streaming parts as done, pending/running tools as cancelled,\n * and recursively finalizes worker parts.\n */\nfunction finalizeParts(parts: UIMessagePart[], workerError?: string): UIMessagePart[] {\n return parts.map((part): UIMessagePart => {\n if (part.type === 'text' || part.type === 'reasoning') {\n if (part.status === 'streaming') {\n return { ...part, status: 'done' };\n }\n }\n if (part.type === 'object' && part.status === 'streaming') {\n return { ...part, status: 'done' };\n }\n if (part.type === 'tool-call') {\n if (part.status === 'pending' || part.status === 'running') {\n return { ...part, status: 'cancelled' };\n }\n }\n if (part.type === 'operation' && part.status === 'running') {\n return { ...part, status: 'cancelled' };\n }\n if (part.type === 'worker' && part.status === 'running') {\n return {\n ...part,\n status: 'error',\n error: workerError,\n parts: finalizeParts(part.parts), // Recursive for nested parts\n };\n }\n return part;\n });\n}\n\n// =============================================================================\n// OctavusChat Class\n// =============================================================================\n\n/**\n * Framework-agnostic chat client for Octavus agents.\n * Manages chat state and streaming, allowing reactive frameworks to subscribe to updates.\n *\n * @example HTTP transport (Next.js, etc.)\n * ```typescript\n * import { OctavusChat, createHttpTransport } from '@octavus/client-sdk';\n *\n * const chat = new OctavusChat({\n * transport: createHttpTransport({\n * request: (payload, options) =>\n * fetch('/api/trigger', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, ...payload }),\n * signal: options?.signal,\n * }),\n * }),\n * });\n * ```\n *\n * @example Socket transport (WebSocket, SockJS, Meteor)\n * ```typescript\n * import { OctavusChat, createSocketTransport } from '@octavus/client-sdk';\n *\n * const chat = new OctavusChat({\n * transport: createSocketTransport({\n * connect: () => new Promise((resolve, reject) => {\n * const ws = new WebSocket(`wss://api.octavus.ai/stream?sessionId=${sessionId}`);\n * ws.onopen = () => resolve(ws);\n * ws.onerror = () => reject(new Error('Connection failed'));\n * }),\n * }),\n * });\n * ```\n */\nexport class OctavusChat {\n // Private state\n private _messages: UIMessage[];\n private _status: ChatStatus = 'idle';\n private _error: OctavusError | null = null;\n private options: OctavusChatOptions;\n private transport: Transport;\n private streamingState: StreamingState | null = null;\n\n // Client tool state\n // Keyed by toolName -> array of pending tools for that name\n private _pendingToolsByName = new Map<string, PendingToolState[]>();\n // Keyed by toolCallId -> pending tool state (for internal lookup when submitting)\n private _pendingToolsByCallId = new Map<string, PendingToolState>();\n // Cache for React useSyncExternalStore compatibility\n private _pendingClientToolsCache: Record<string, InteractiveTool[]> = {};\n private _completedToolResults: ToolResult[] = [];\n private _clientToolAbortController: AbortController | null = null;\n // Server tool results from mixed server+client tools (for continuation)\n private _serverToolResults: ToolResult[] = [];\n // Execution ID for continuation (from client-tool-request event)\n private _pendingExecutionId: string | null = null;\n // Flag indicating automatic client tools have completed and are ready to continue\n // We wait for the finish event before actually continuing to avoid race conditions\n private _readyToContinue = false;\n // Flag indicating the finish event with client-tool-calls reason has been received\n // Used to handle the race condition where finish arrives before async tools complete\n private _finishEventReceived = false;\n\n // Listener sets for reactive frameworks\n private listeners = new Set<Listener>();\n\n constructor(options: OctavusChatOptions) {\n this.options = options;\n this._messages = options.initialMessages ?? [];\n this.transport = options.transport;\n }\n\n /**\n * Update mutable options (callbacks and tool handlers) without recreating the instance.\n * Used by the React hook to keep options fresh across renders, but can also be\n * called directly by non-React consumers.\n *\n * `transport` and `initialMessages` are excluded since they're only consumed at construction time.\n */\n updateOptions(updates: Partial<Omit<OctavusChatOptions, 'transport' | 'initialMessages'>>): void {\n this.options = { ...this.options, ...updates };\n }\n\n // =========================================================================\n // Public Getters\n // =========================================================================\n\n get messages(): UIMessage[] {\n return this._messages;\n }\n\n get status(): ChatStatus {\n return this._status;\n }\n\n /**\n * The current error, if any.\n * Contains structured error information including type, source, and retryability.\n */\n get error(): OctavusError | null {\n return this._error;\n }\n\n /**\n * Pending interactive tool calls keyed by tool name.\n * Each tool has bound `submit()` and `cancel()` methods.\n *\n * @example\n * ```tsx\n * const feedbackTools = pendingClientTools['request-feedback'] ?? [];\n *\n * {feedbackTools.map(tool => (\n * <FeedbackModal\n * key={tool.toolCallId}\n * {...tool.args}\n * onSubmit={(result) => tool.submit(result)}\n * onCancel={() => tool.cancel()}\n * />\n * ))}\n * ```\n */\n get pendingClientTools(): Record<string, InteractiveTool[]> {\n return this._pendingClientToolsCache;\n }\n\n // =========================================================================\n // Subscription Methods (for reactive frameworks)\n // =========================================================================\n\n /**\n * Subscribe to state changes. The callback is called whenever messages, status, or error changes.\n * @returns Unsubscribe function\n */\n subscribe(listener: Listener): () => void {\n this.listeners.add(listener);\n return () => this.listeners.delete(listener);\n }\n\n private notifyListeners(): void {\n this.listeners.forEach((l) => l());\n }\n\n // =========================================================================\n // Private Setters (notify listeners)\n // =========================================================================\n\n private setMessages(messages: UIMessage[]): void {\n this._messages = messages;\n this.notifyListeners();\n }\n\n private setStatus(status: ChatStatus): void {\n this._status = status;\n this.notifyListeners();\n }\n\n private setError(error: OctavusError | null): void {\n this._error = error;\n this.notifyListeners();\n }\n\n private updatePendingClientToolsCache(): void {\n const cache: Record<string, InteractiveTool[]> = {};\n for (const [toolName, tools] of this._pendingToolsByName.entries()) {\n cache[toolName] = tools.map((tool) => ({\n toolCallId: tool.toolCallId,\n toolName: tool.toolName,\n args: tool.args,\n submit: (result: unknown) => this.submitToolResult(tool.toolCallId, result),\n cancel: (reason?: string) =>\n this.submitToolResult(tool.toolCallId, undefined, reason ?? 'User cancelled'),\n }));\n }\n this._pendingClientToolsCache = cache;\n }\n\n // =========================================================================\n // Public Methods\n // =========================================================================\n\n /**\n * Trigger the agent and optionally add a user message to the chat.\n *\n * @param triggerName - The trigger name defined in the agent's protocol.yaml\n * @param input - Input parameters for the trigger (variable substitutions)\n * @param options.userMessage - If provided, adds a user message to the chat before triggering\n *\n * @example Send a text message\n * ```typescript\n * await chat.send('user-message',\n * { USER_MESSAGE: message },\n * { userMessage: { content: message } }\n * );\n * ```\n *\n * @example Send a message with file attachments\n * ```typescript\n * await chat.send('user-message',\n * { USER_MESSAGE: message, FILES: fileRefs },\n * { userMessage: { content: message, files: fileRefs } }\n * );\n * ```\n */\n async send(\n triggerName: string,\n input?: Record<string, unknown>,\n sendOptions?: { userMessage?: UserMessageInput },\n ): Promise<void> {\n this.transport.stop();\n\n let fileRefs: FileReference[] | undefined;\n if (sendOptions?.userMessage?.files) {\n const files = sendOptions.userMessage.files;\n if (isFileReferenceArray(files)) {\n fileRefs = files;\n } else if (this.options.requestUploadUrls) {\n fileRefs = await uploadFiles(files, {\n requestUploadUrls: this.options.requestUploadUrls,\n });\n } else {\n throw new Error(\n 'File upload requires requestUploadUrls option. Either provide FileReference[] or configure requestUploadUrls.',\n );\n }\n }\n\n // Auto-upload FILES in trigger input if needed\n let processedInput = input;\n if (input?.FILES !== undefined && !isFileReferenceArray(input.FILES)) {\n if (this.options.requestUploadUrls) {\n const inputFiles = input.FILES as FileList | File[];\n const uploadedRefs =\n fileRefs ??\n (await uploadFiles(inputFiles, {\n requestUploadUrls: this.options.requestUploadUrls,\n }));\n processedInput = { ...input, FILES: uploadedRefs };\n fileRefs = fileRefs ?? uploadedRefs;\n }\n }\n\n // Optimistic UI: add user message before server responds\n if (sendOptions?.userMessage !== undefined) {\n const userMsg = createUserMessage(sendOptions.userMessage, fileRefs);\n this.setMessages([...this._messages, userMsg]);\n }\n\n this.setStatus('streaming');\n this.setError(null);\n this.streamingState = createEmptyStreamingState();\n\n // Clear any previous client tool state\n this._pendingToolsByName.clear();\n this._pendingToolsByCallId.clear();\n this._completedToolResults = [];\n this._serverToolResults = [];\n this._pendingExecutionId = null;\n this._readyToContinue = false;\n this._finishEventReceived = false;\n this.updatePendingClientToolsCache();\n\n try {\n for await (const event of this.transport.trigger(triggerName, processedInput)) {\n if (this.streamingState === null) break;\n\n this.handleStreamEvent(event, this.streamingState);\n }\n } catch (err) {\n // Convert unknown errors to OctavusError\n const errorObj = OctavusError.isInstance(err)\n ? err\n : new OctavusError({\n errorType: 'internal_error',\n message: err instanceof Error ? err.message : 'Unknown error',\n source: 'client',\n retryable: false,\n cause: err,\n });\n\n // Finalize any streaming message before setting error state\n const state = this.streamingState;\n if (state !== null) {\n const messages = [...this._messages];\n const lastMsg = messages[messages.length - 1];\n\n if (state.parts.length > 0) {\n const finalParts = finalizeParts(state.parts, 'Stream error');\n\n const finalMessage: UIMessage = {\n id: state.messageId,\n role: 'assistant',\n parts: finalParts,\n status: 'done',\n createdAt: new Date(),\n };\n\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = finalMessage;\n } else {\n messages.push(finalMessage);\n }\n this.setMessages(messages);\n } else if (lastMsg?.id === state.messageId) {\n // No parts yet - remove the empty streaming message\n messages.pop();\n this.setMessages(messages);\n }\n }\n\n this.setError(errorObj);\n this.setStatus('error');\n this.streamingState = null;\n this.options.onError?.(errorObj);\n }\n }\n\n /**\n * Upload files directly without sending a message.\n * Useful for showing upload progress before sending.\n *\n * @param files - Files to upload\n * @param onProgress - Optional progress callback\n * @returns Array of file references\n *\n * @example\n * ```typescript\n * const fileRefs = await chat.uploadFiles(fileInput.files, (i, progress) => {\n * console.log(`File ${i}: ${progress}%`);\n * });\n * // Later...\n * await chat.send('user-message', { FILES: fileRefs }, { userMessage: { files: fileRefs } });\n * ```\n */\n async uploadFiles(\n files: FileList | File[],\n onProgress?: (fileIndex: number, progress: number) => void,\n ): Promise<FileReference[]> {\n if (!this.options.requestUploadUrls) {\n throw new Error('File upload requires requestUploadUrls option');\n }\n return await uploadFiles(files, {\n requestUploadUrls: this.options.requestUploadUrls,\n onProgress,\n });\n }\n\n /**\n * Internal: Submit a result for a pending tool.\n * Called by bound submit/cancel methods on InteractiveTool.\n */\n private submitToolResult(toolCallId: string, result?: unknown, error?: string): void {\n const pendingTool = this._pendingToolsByCallId.get(toolCallId);\n if (!pendingTool) {\n // Tool not found - may have been cancelled or already resolved\n return;\n }\n\n // Remove from both maps\n this._pendingToolsByCallId.delete(toolCallId);\n const toolsForName = this._pendingToolsByName.get(pendingTool.toolName);\n if (toolsForName) {\n const filtered = toolsForName.filter((t) => t.toolCallId !== toolCallId);\n if (filtered.length === 0) {\n this._pendingToolsByName.delete(pendingTool.toolName);\n } else {\n this._pendingToolsByName.set(pendingTool.toolName, filtered);\n }\n }\n this.updatePendingClientToolsCache();\n\n const toolResult: ToolResult = {\n toolCallId,\n toolName: pendingTool.toolName,\n result: error ? undefined : result,\n error,\n outputVariable: pendingTool.outputVariable,\n blockIndex: pendingTool.blockIndex,\n thread: pendingTool.thread,\n workerId: pendingTool.workerId,\n };\n this._completedToolResults.push(toolResult);\n\n if (error) {\n this.emitToolOutputError(toolCallId, error);\n } else {\n this.emitToolOutputAvailable(toolCallId, result);\n }\n\n if (this._pendingToolsByCallId.size === 0) {\n void this.continueWithClientToolResults();\n }\n\n this.notifyListeners();\n }\n\n /** Stop the current streaming and finalize any partial message */\n stop(): void {\n if (this._status !== 'streaming' && this._status !== 'awaiting-input') {\n return;\n }\n\n this._clientToolAbortController?.abort();\n this._clientToolAbortController = null;\n this._pendingToolsByName.clear();\n this._pendingToolsByCallId.clear();\n this._completedToolResults = [];\n this._serverToolResults = [];\n this._pendingExecutionId = null;\n this._readyToContinue = false;\n this._finishEventReceived = false;\n this.updatePendingClientToolsCache();\n\n this.transport.stop();\n\n const state = this.streamingState;\n if (state && state.parts.length > 0) {\n const finalParts = finalizeParts(state.parts, 'Stopped by user');\n\n const finalMessage: UIMessage = {\n id: state.messageId,\n role: 'assistant',\n parts: finalParts,\n status: 'done',\n createdAt: new Date(),\n };\n\n const messages = [...this._messages];\n const lastMsg = messages[messages.length - 1];\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = finalMessage;\n } else {\n messages.push(finalMessage);\n }\n this.setMessages(messages);\n }\n\n this.streamingState = null;\n this.setStatus('idle');\n this.options.onStop?.();\n }\n\n // =========================================================================\n // Private Helpers\n // =========================================================================\n\n private handleStreamEvent(event: StreamEvent, state: StreamingState): void {\n switch (event.type) {\n case 'start':\n // Call onStart callback with execution/session ID\n if (event.executionId) {\n this.options.onStart?.(event.executionId);\n }\n break;\n\n case 'block-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n const block: BlockState = {\n blockId: event.blockId,\n blockName: event.blockName,\n blockType: event.blockType,\n display: event.display,\n description: event.description,\n outputToChat: event.outputToChat ?? true,\n thread: event.thread,\n reasoning: '',\n text: '',\n toolCalls: new Map(),\n };\n state.blocks.set(event.blockId, block);\n state.activeBlock = block;\n\n const isOperation = OPERATION_BLOCK_TYPES.has(event.blockType);\n const isHidden = event.display === 'hidden';\n if (isOperation && !isHidden) {\n const thread = event.thread;\n const operationPart: UIOperationPart = {\n type: 'operation',\n operationId: event.blockId,\n name: event.description ?? event.blockName,\n operationType: event.blockType,\n status: 'running',\n thread: threadForPart(thread),\n };\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n state.parts[workerState.partIndex] = {\n ...workerPart,\n parts: [...workerPart.parts, operationPart],\n };\n } else {\n state.parts.push(operationPart);\n }\n }\n\n state.currentTextPartIndex = null;\n state.currentReasoningPartIndex = null;\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'block-end': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Find operation in worker's nested parts\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const operationPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'operation' && p.operationId === event.blockId,\n );\n if (operationPartIndex >= 0) {\n const part = workerPart.parts[operationPartIndex] as UIOperationPart;\n const updatedParts = [...workerPart.parts];\n updatedParts[operationPartIndex] = { ...part, status: 'done' };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n }\n } else {\n const operationPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'operation' && p.operationId === event.blockId,\n );\n if (operationPartIndex >= 0) {\n const part = state.parts[operationPartIndex] as UIOperationPart;\n state.parts[operationPartIndex] = { ...part, status: 'done' };\n }\n }\n\n if (state.activeBlock?.blockId === event.blockId) {\n state.activeBlock = null;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'reasoning-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n const reasoningPart: UIReasoningPart = {\n type: 'reasoning',\n text: '',\n status: 'streaming',\n thread: threadForPart(state.activeBlock?.thread),\n };\n\n if (workerState) {\n // Add to worker's nested parts\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const newParts = [...workerPart.parts, reasoningPart];\n workerState.currentReasoningPartIndex = newParts.length - 1;\n state.parts[workerState.partIndex] = { ...workerPart, parts: newParts };\n } else {\n state.parts.push(reasoningPart);\n state.currentReasoningPartIndex = state.parts.length - 1;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'reasoning-delta': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Update worker's reasoning part\n if (workerState.currentReasoningPartIndex !== null) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const part = workerPart.parts[workerState.currentReasoningPartIndex] as UIReasoningPart;\n const updatedParts = [...workerPart.parts];\n updatedParts[workerState.currentReasoningPartIndex] = {\n ...part,\n text: part.text + event.delta,\n };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n }\n } else {\n if (state.currentReasoningPartIndex !== null) {\n const part = state.parts[state.currentReasoningPartIndex] as UIReasoningPart;\n state.parts[state.currentReasoningPartIndex] = {\n ...part,\n text: part.text + event.delta,\n };\n }\n\n if (state.activeBlock) {\n state.activeBlock.reasoning += event.delta;\n }\n }\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'reasoning-end': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Finalize worker's reasoning part\n if (workerState.currentReasoningPartIndex !== null) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const part = workerPart.parts[workerState.currentReasoningPartIndex] as UIReasoningPart;\n const updatedParts = [...workerPart.parts];\n updatedParts[workerState.currentReasoningPartIndex] = {\n ...part,\n status: 'done',\n };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n workerState.currentReasoningPartIndex = null;\n }\n } else if (state.currentReasoningPartIndex !== null) {\n const part = state.parts[state.currentReasoningPartIndex] as UIReasoningPart;\n state.parts[state.currentReasoningPartIndex] = { ...part, status: 'done' };\n state.currentReasoningPartIndex = null;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'text-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n const thread = threadForPart(state.activeBlock?.thread);\n const shouldAddPart = state.activeBlock?.outputToChat !== false || thread !== undefined;\n\n // For worker events, always add parts\n if (workerState || shouldAddPart) {\n // Structured output mode: accumulate JSON and parse progressively\n if (event.responseType) {\n const objectPart: UIObjectPart = {\n type: 'object',\n id: event.id,\n typeName: event.responseType,\n partial: undefined,\n object: undefined,\n status: 'streaming',\n thread,\n };\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const newParts = [...workerPart.parts, objectPart];\n workerState.currentObjectPartIndex = newParts.length - 1;\n workerState.accumulatedJson = '';\n workerState.currentTextPartIndex = null;\n state.parts[workerState.partIndex] = { ...workerPart, parts: newParts };\n } else {\n state.parts.push(objectPart);\n state.currentObjectPartIndex = state.parts.length - 1;\n state.accumulatedJson = '';\n state.currentTextPartIndex = null;\n }\n } else {\n const textPart: UITextPart = {\n type: 'text',\n text: '',\n status: 'streaming',\n thread,\n };\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const newParts = [...workerPart.parts, textPart];\n workerState.currentTextPartIndex = newParts.length - 1;\n workerState.currentObjectPartIndex = null;\n state.parts[workerState.partIndex] = { ...workerPart, parts: newParts };\n } else {\n state.parts.push(textPart);\n state.currentTextPartIndex = state.parts.length - 1;\n state.currentObjectPartIndex = null;\n }\n }\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'text-delta': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Update worker's text or object part\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n if (workerState.currentObjectPartIndex !== null) {\n workerState.accumulatedJson += event.delta;\n const part = workerPart.parts[workerState.currentObjectPartIndex] as UIObjectPart;\n const parsed = parsePartialJson(workerState.accumulatedJson);\n if (parsed !== undefined) {\n const updatedParts = [...workerPart.parts];\n updatedParts[workerState.currentObjectPartIndex] = { ...part, partial: parsed };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n }\n } else if (workerState.currentTextPartIndex !== null) {\n const part = workerPart.parts[workerState.currentTextPartIndex] as UITextPart;\n const updatedParts = [...workerPart.parts];\n updatedParts[workerState.currentTextPartIndex] = {\n ...part,\n text: part.text + event.delta,\n };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n }\n } else {\n if (state.currentObjectPartIndex !== null) {\n state.accumulatedJson += event.delta;\n const part = state.parts[state.currentObjectPartIndex] as UIObjectPart;\n const parsed = parsePartialJson(state.accumulatedJson);\n if (parsed !== undefined) {\n state.parts[state.currentObjectPartIndex] = { ...part, partial: parsed };\n }\n } else if (state.currentTextPartIndex !== null) {\n const part = state.parts[state.currentTextPartIndex] as UITextPart;\n state.parts[state.currentTextPartIndex] = {\n ...part,\n text: part.text + event.delta,\n };\n }\n\n if (state.activeBlock) {\n state.activeBlock.text += event.delta;\n }\n }\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'text-end': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Finalize worker's text or object part\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const updatedParts = [...workerPart.parts];\n if (workerState.currentObjectPartIndex !== null) {\n const part = workerPart.parts[workerState.currentObjectPartIndex] as UIObjectPart;\n try {\n const finalObject = JSON.parse(workerState.accumulatedJson) as unknown;\n updatedParts[workerState.currentObjectPartIndex] = {\n ...part,\n object: finalObject,\n partial: finalObject,\n status: 'done',\n };\n } catch {\n updatedParts[workerState.currentObjectPartIndex] = {\n ...part,\n status: 'error',\n error: 'Failed to parse response as JSON',\n };\n }\n workerState.currentObjectPartIndex = null;\n workerState.accumulatedJson = '';\n } else if (workerState.currentTextPartIndex !== null) {\n const part = workerPart.parts[workerState.currentTextPartIndex] as UITextPart;\n updatedParts[workerState.currentTextPartIndex] = {\n ...part,\n status: 'done',\n };\n workerState.currentTextPartIndex = null;\n }\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n } else if (state.currentObjectPartIndex !== null) {\n const part = state.parts[state.currentObjectPartIndex] as UIObjectPart;\n try {\n const finalObject = JSON.parse(state.accumulatedJson) as unknown;\n state.parts[state.currentObjectPartIndex] = {\n ...part,\n object: finalObject,\n partial: finalObject,\n status: 'done',\n };\n } catch {\n state.parts[state.currentObjectPartIndex] = {\n ...part,\n status: 'error',\n error: 'Failed to parse response as JSON',\n };\n }\n state.currentObjectPartIndex = null;\n state.accumulatedJson = '';\n } else if (state.currentTextPartIndex !== null) {\n const part = state.parts[state.currentTextPartIndex] as UITextPart;\n state.parts[state.currentTextPartIndex] = { ...part, status: 'done' };\n state.currentTextPartIndex = null;\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'tool-input-start': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n const toolPart: UIToolCallPart = {\n type: 'tool-call',\n toolCallId: event.toolCallId,\n toolName: event.toolName,\n displayName: event.title,\n args: {},\n result: undefined,\n error: undefined,\n status: 'pending',\n thread: threadForPart(state.activeBlock?.thread),\n };\n\n // Initialize the input buffer for this tool call\n if (workerState) {\n workerState.toolInputBuffers.set(event.toolCallId, '');\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n state.parts[workerState.partIndex] = {\n ...workerPart,\n parts: [...workerPart.parts, toolPart],\n };\n } else {\n state.toolInputBuffers.set(event.toolCallId, '');\n state.parts.push(toolPart);\n\n if (state.activeBlock) {\n state.activeBlock.toolCalls.set(event.toolCallId, toolPart);\n }\n }\n\n this.updateStreamingMessage();\n break;\n }\n\n case 'tool-input-delta': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Accumulate the delta into the worker's buffer\n const existing = workerState.toolInputBuffers.get(event.toolCallId) ?? '';\n const accumulated = existing + event.inputTextDelta;\n workerState.toolInputBuffers.set(event.toolCallId, accumulated);\n\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const toolPart = workerPart.parts[toolPartIndex] as UIToolCallPart;\n const parsed = parsePartialJson(accumulated);\n if (parsed !== undefined) {\n const updatedParts = [...workerPart.parts];\n updatedParts[toolPartIndex] = {\n ...toolPart,\n args: parsed as Record<string, unknown>,\n };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n this.updateStreamingMessage();\n }\n }\n } else {\n // Accumulate the delta into the top-level buffer\n const existing = state.toolInputBuffers.get(event.toolCallId) ?? '';\n const accumulated = existing + event.inputTextDelta;\n state.toolInputBuffers.set(event.toolCallId, accumulated);\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const toolPart = state.parts[toolPartIndex] as UIToolCallPart;\n const parsed = parsePartialJson(accumulated);\n if (parsed !== undefined) {\n state.parts[toolPartIndex] = {\n ...toolPart,\n args: parsed as Record<string, unknown>,\n };\n this.updateStreamingMessage();\n }\n }\n }\n break;\n }\n\n case 'tool-input-end':\n // Input streaming ended, wait for tool-input-available\n break;\n\n case 'tool-input-available': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n // Clean up the worker buffer\n workerState.toolInputBuffers.delete(event.toolCallId);\n\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = workerPart.parts[toolPartIndex] as UIToolCallPart;\n const updatedParts = [...workerPart.parts];\n updatedParts[toolPartIndex] = {\n ...part,\n args: event.input as Record<string, unknown>,\n status: 'running',\n };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n this.updateStreamingMessage();\n }\n } else {\n // Clean up the top-level buffer\n state.toolInputBuffers.delete(event.toolCallId);\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n state.parts[toolPartIndex] = {\n ...part,\n args: event.input as Record<string, unknown>,\n status: 'running',\n };\n this.updateStreamingMessage();\n }\n }\n break;\n }\n\n case 'tool-output-available': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = workerPart.parts[toolPartIndex] as UIToolCallPart;\n const updatedParts = [...workerPart.parts];\n updatedParts[toolPartIndex] = {\n ...part,\n result: event.output,\n status: 'done',\n };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n this.updateStreamingMessage();\n }\n } else {\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n state.parts[toolPartIndex] = {\n ...part,\n result: event.output,\n status: 'done',\n };\n this.updateStreamingMessage();\n }\n }\n break;\n }\n\n case 'tool-output-error': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n const toolPartIndex = workerPart.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = workerPart.parts[toolPartIndex] as UIToolCallPart;\n const updatedParts = [...workerPart.parts];\n updatedParts[toolPartIndex] = {\n ...part,\n error: event.error,\n status: 'error',\n };\n state.parts[workerState.partIndex] = { ...workerPart, parts: updatedParts };\n this.updateStreamingMessage();\n }\n } else {\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === event.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n state.parts[toolPartIndex] = {\n ...part,\n error: event.error,\n status: 'error',\n };\n this.updateStreamingMessage();\n }\n }\n break;\n }\n\n case 'source': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n const thread = threadForPart(state.activeBlock?.thread);\n\n let sourcePart: UISourcePart;\n if (event.sourceType === 'url') {\n sourcePart = {\n type: 'source',\n sourceType: 'url',\n id: event.id,\n url: event.url,\n title: event.title,\n thread,\n };\n } else {\n sourcePart = {\n type: 'source',\n sourceType: 'document',\n id: event.id,\n mediaType: event.mediaType,\n title: event.title,\n filename: event.filename,\n thread,\n };\n }\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n state.parts[workerState.partIndex] = {\n ...workerPart,\n parts: [...workerPart.parts, sourcePart],\n };\n } else {\n state.parts.push(sourcePart);\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'file-available': {\n const workerId = event.workerId;\n const workerState = workerId ? state.activeWorkers.get(workerId) : undefined;\n\n // Add generated file as a part\n const filePart: UIFilePart = {\n type: 'file',\n id: event.id,\n mediaType: event.mediaType,\n url: event.url,\n filename: event.filename,\n size: event.size,\n toolCallId: event.toolCallId,\n thread: threadForPart(state.activeBlock?.thread),\n };\n\n if (workerState) {\n const workerPart = state.parts[workerState.partIndex] as UIWorkerPart;\n state.parts[workerState.partIndex] = {\n ...workerPart,\n parts: [...workerPart.parts, filePart],\n };\n } else {\n state.parts.push(filePart);\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'resource-update':\n this.options.onResourceUpdate?.(event.name, event.value);\n break;\n\n case 'worker-start': {\n // Check if worker with same workerId already exists (for continuations)\n const existingIndex = state.parts.findIndex(\n (p) => p.type === 'worker' && p.workerId === event.workerId,\n );\n\n let partIndex: number;\n if (existingIndex !== -1) {\n // Re-use existing worker part (continuation)\n const existingPart = state.parts[existingIndex] as UIWorkerPart;\n state.parts[existingIndex] = { ...existingPart, status: 'running' };\n partIndex = existingIndex;\n } else {\n // Create a new worker part\n const workerPart: UIWorkerPart = {\n type: 'worker',\n workerId: event.workerId,\n workerSlug: event.workerSlug,\n description: event.description,\n parts: [],\n status: 'running',\n };\n state.parts.push(workerPart);\n partIndex = state.parts.length - 1;\n }\n\n // Track the worker for event routing\n const workerState: WorkerPartState = {\n partIndex,\n currentTextPartIndex: null,\n currentReasoningPartIndex: null,\n currentObjectPartIndex: null,\n accumulatedJson: '',\n toolInputBuffers: new Map(),\n };\n state.activeWorkers.set(event.workerId, workerState);\n this.updateStreamingMessage();\n break;\n }\n\n case 'worker-result': {\n const workerState = state.activeWorkers.get(event.workerId);\n if (workerState !== undefined) {\n const part = state.parts[workerState.partIndex] as UIWorkerPart;\n state.parts[workerState.partIndex] = {\n ...part,\n output: event.output,\n error: event.error,\n status: event.error ? 'error' : 'done',\n parts: part.parts.map((p): UIMessagePart => {\n if (p.type === 'text' || p.type === 'reasoning') {\n if (p.status === 'streaming') {\n return { ...p, status: 'done' };\n }\n }\n if (p.type === 'object' && p.status === 'streaming') {\n return { ...p, status: 'done' };\n }\n return p;\n }),\n };\n state.activeWorkers.delete(event.workerId);\n }\n this.updateStreamingMessage();\n break;\n }\n\n case 'finish': {\n // Handle client-tool-calls finish reason\n if (event.finishReason === 'client-tool-calls') {\n // Mark that finish event has been received (for async tools that complete later)\n this._finishEventReceived = true;\n // Don't finalize message - we're waiting for client tools\n if (this._pendingToolsByCallId.size > 0) {\n this.setStatus('awaiting-input');\n } else if (this._readyToContinue) {\n // Automatic tools completed before finish event arrived - continue now\n this._readyToContinue = false;\n this._finishEventReceived = false;\n void this.continueWithClientToolResults();\n }\n return;\n }\n\n const finalMessage = buildMessageFromState(state, 'done');\n\n finalMessage.parts = finalMessage.parts.map((part) => {\n if (part.type === 'text' || part.type === 'reasoning') {\n return { ...part, status: 'done' as const };\n }\n if (part.type === 'object' && part.status === 'streaming') {\n return { ...part, status: 'done' as const };\n }\n return part;\n });\n\n if (finalMessage.parts.length > 0) {\n const messages = [...this._messages];\n const lastMsg = messages[messages.length - 1];\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = finalMessage;\n } else {\n messages.push(finalMessage);\n }\n this.setMessages(messages);\n }\n\n this.setStatus('idle');\n this.streamingState = null;\n this.options.onFinish?.();\n break;\n }\n\n case 'error': {\n // Create structured error from the error event\n throw new OctavusError({\n errorType: event.errorType,\n message: event.message,\n source: event.source,\n retryable: event.retryable,\n retryAfter: event.retryAfter,\n code: event.code,\n provider: event.provider,\n tool: event.tool,\n });\n }\n\n case 'tool-request':\n // Handled by server-sdk, not relevant for UI\n break;\n\n case 'client-tool-request':\n // Store execution ID and server tool results for continuation\n this._pendingExecutionId = event.executionId;\n this._serverToolResults = event.serverToolResults ?? [];\n // Handle client-side tool execution\n void this.handleClientToolRequest(event.toolCalls, state);\n break;\n }\n }\n\n private updateStreamingMessage(): void {\n const state = this.streamingState;\n if (!state) return;\n\n const msg = buildMessageFromState(state, 'streaming');\n const messages = [...this._messages];\n\n const lastMsg = messages[messages.length - 1];\n if (lastMsg?.id === state.messageId) {\n messages[messages.length - 1] = msg;\n } else {\n messages.push(msg);\n }\n\n this.setMessages(messages);\n }\n\n /**\n * Emit a tool-output-available event for a client tool result.\n */\n private emitToolOutputAvailable(toolCallId: string, output: unknown): void {\n const state = this.streamingState;\n if (!state) return;\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n state.parts[toolPartIndex] = {\n ...part,\n result: output,\n status: 'done',\n };\n this.updateStreamingMessage();\n }\n }\n\n /**\n * Emit a tool-output-error event for a client tool result.\n */\n private emitToolOutputError(toolCallId: string, error: string): void {\n const state = this.streamingState;\n if (!state) return;\n\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n state.parts[toolPartIndex] = {\n ...part,\n error,\n status: 'error',\n };\n this.updateStreamingMessage();\n }\n }\n\n /**\n * Continue execution with collected client tool results.\n */\n private async continueWithClientToolResults(): Promise<void> {\n if (this._completedToolResults.length === 0) return;\n\n if (this._pendingExecutionId === null) {\n // Context lost - this shouldn't happen, but handle gracefully\n const errorObj = new OctavusError({\n errorType: 'internal_error',\n message: 'Cannot continue execution: execution ID was lost.',\n source: 'client',\n retryable: false,\n });\n this.setError(errorObj);\n this.setStatus('error');\n this.options.onError?.(errorObj);\n return;\n }\n\n // Combine server results (from mixed tools scenario) with client results\n const allResults = [...this._serverToolResults, ...this._completedToolResults];\n const executionId = this._pendingExecutionId;\n this._serverToolResults = [];\n this._completedToolResults = [];\n this._pendingExecutionId = null;\n\n this.setStatus('streaming');\n\n try {\n // Use the transport's continuation method (works for both HTTP and Socket)\n for await (const event of this.transport.continueWithToolResults(executionId, allResults)) {\n if (this.streamingState === null) break;\n this.handleStreamEvent(event, this.streamingState);\n }\n } catch (err) {\n const errorObj = OctavusError.isInstance(err)\n ? err\n : new OctavusError({\n errorType: 'internal_error',\n message: err instanceof Error ? err.message : 'Unknown error',\n source: 'client',\n retryable: false,\n cause: err,\n });\n\n this.setError(errorObj);\n this.setStatus('error');\n this.streamingState = null;\n this.options.onError?.(errorObj);\n }\n }\n\n /**\n * Handle client tool request event.\n *\n * IMPORTANT: Interactive tools must be registered synchronously (before any await)\n * to avoid a race condition where the finish event is processed before tools are added.\n */\n private async handleClientToolRequest(\n toolCalls: PendingToolCall[],\n state: StreamingState,\n ): Promise<void> {\n this._clientToolAbortController = new AbortController();\n\n // FIRST PASS: Register all interactive tools synchronously (no await)\n // This ensures pending tools are populated before finish event is processed\n for (const tc of toolCalls) {\n const handler = this.options.clientTools?.[tc.toolName];\n if (handler === 'interactive') {\n const toolState: PendingToolState = {\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n args: tc.args,\n source: tc.source,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n };\n // Add to both maps\n this._pendingToolsByCallId.set(tc.toolCallId, toolState);\n const existing = this._pendingToolsByName.get(tc.toolName) ?? [];\n this._pendingToolsByName.set(tc.toolName, [...existing, toolState]);\n }\n }\n if (this._pendingToolsByCallId.size > 0) {\n this.updatePendingClientToolsCache();\n }\n\n // SECOND PASS: Execute automatic handlers and handle missing handlers\n for (const tc of toolCalls) {\n const handler = this.options.clientTools?.[tc.toolName];\n\n if (handler === 'interactive') {\n // Already registered above, just update UI state\n const toolPartIndex = state.parts.findIndex(\n (p: UIMessagePart) => p.type === 'tool-call' && p.toolCallId === tc.toolCallId,\n );\n if (toolPartIndex >= 0) {\n const part = state.parts[toolPartIndex] as UIToolCallPart;\n // Keep running status - user will see the tool is \"executing\" while modal is open\n state.parts[toolPartIndex] = { ...part };\n }\n } else if (handler) {\n try {\n const collectedFiles: FileReference[] = [];\n const result = await handler(tc.args, {\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n signal: this._clientToolAbortController.signal,\n addFile: (file) => collectedFiles.push(file),\n });\n\n this._completedToolResults.push({\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n result,\n files: collectedFiles.length > 0 ? collectedFiles : undefined,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n });\n\n this.emitToolOutputAvailable(tc.toolCallId, result);\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : 'Tool execution failed';\n this._completedToolResults.push({\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n error: errorMessage,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n });\n\n this.emitToolOutputError(tc.toolCallId, errorMessage);\n }\n } else {\n // No handler registered - treat as error\n const errorMessage = `No client handler for tool: ${tc.toolName}`;\n this._completedToolResults.push({\n toolCallId: tc.toolCallId,\n toolName: tc.toolName,\n error: errorMessage,\n outputVariable: tc.outputVariable,\n blockIndex: tc.blockIndex,\n thread: tc.thread,\n workerId: tc.workerId,\n });\n\n this.emitToolOutputError(tc.toolCallId, errorMessage);\n }\n }\n\n // If no interactive tools, mark as ready to continue.\n // We wait for the finish event to arrive first to avoid a race condition where\n // the finish event gets delivered to the continuation's event resolver.\n if (this._pendingToolsByCallId.size === 0 && this._completedToolResults.length > 0) {\n this._readyToContinue = true;\n\n // If finish event already arrived while we were executing async tools,\n // trigger continuation now instead of waiting forever\n if (this._finishEventReceived) {\n this._readyToContinue = false;\n this._finishEventReceived = false;\n void this.continueWithClientToolResults();\n }\n }\n }\n}\n","import type { FileReference } from '@octavus/core';\n\n/**\n * Response from the upload URLs endpoint\n */\nexport interface UploadUrlsResponse {\n files: {\n id: string;\n uploadUrl: string;\n downloadUrl: string;\n }[];\n}\n\n/**\n * Options for uploading files\n */\nexport interface UploadFilesOptions {\n /**\n * Function to request upload URLs from the platform.\n * Consumer apps must implement this to authenticate with the platform.\n *\n * @param files - Array of file metadata to request URLs for\n * @returns Response with presigned upload and download URLs\n *\n * @example\n * ```typescript\n * requestUploadUrls: async (files) => {\n * const response = await fetch('/api/upload-urls', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, files }),\n * });\n * return response.json();\n * }\n * ```\n */\n requestUploadUrls: (\n files: { filename: string; mediaType: string; size: number }[],\n ) => Promise<UploadUrlsResponse>;\n\n /**\n * Callback for upload progress (0-100 per file).\n * Called multiple times during upload with real-time progress.\n *\n * @param fileIndex - Index of the file being uploaded\n * @param progress - Progress percentage (0-100)\n */\n onProgress?: (fileIndex: number, progress: number) => void;\n}\n\n/**\n * Upload a single file to S3 with progress tracking.\n * Uses XMLHttpRequest for upload progress events (fetch doesn't support this).\n */\nfunction uploadFileWithProgress(\n url: string,\n file: File,\n onProgress?: (progress: number) => void,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n const xhr = new XMLHttpRequest();\n\n xhr.upload.addEventListener('progress', (event) => {\n if (event.lengthComputable) {\n const progress = Math.round((event.loaded / event.total) * 100);\n onProgress?.(progress);\n }\n });\n\n xhr.addEventListener('load', () => {\n if (xhr.status >= 200 && xhr.status < 300) {\n resolve();\n } else {\n reject(new Error(`Upload failed with status ${xhr.status}`));\n }\n });\n\n xhr.addEventListener('error', () => {\n reject(new Error('Upload failed: network error'));\n });\n\n xhr.addEventListener('abort', () => {\n reject(new Error('Upload aborted'));\n });\n\n xhr.open('PUT', url);\n xhr.setRequestHeader('Content-Type', file.type || 'application/octet-stream');\n xhr.send(file);\n });\n}\n\n/**\n * Upload files to the Octavus platform.\n *\n * This function:\n * 1. Requests presigned upload URLs from the platform\n * 2. Uploads each file directly to S3 with progress tracking\n * 3. Returns file references that can be used in trigger input\n *\n * @param files - Files to upload (from file input or drag/drop)\n * @param options - Upload configuration\n * @returns Array of file references with download URLs\n *\n * @example\n * ```typescript\n * const fileRefs = await uploadFiles(fileInputRef.current.files, {\n * requestUploadUrls: async (files) => {\n * const response = await fetch('/api/upload-urls', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, files }),\n * });\n * return response.json();\n * },\n * onProgress: (fileIndex, progress) => {\n * console.log(`File ${fileIndex}: ${progress}%`);\n * },\n * });\n * ```\n */\nexport async function uploadFiles(\n files: FileList | File[],\n options: UploadFilesOptions,\n): Promise<FileReference[]> {\n const fileArray = Array.from(files);\n\n if (fileArray.length === 0) {\n return [];\n }\n\n const { files: uploadInfos } = await options.requestUploadUrls(\n fileArray.map((f) => ({\n filename: f.name,\n mediaType: f.type || 'application/octet-stream',\n size: f.size,\n })),\n );\n\n const references: FileReference[] = [];\n\n for (let i = 0; i < fileArray.length; i++) {\n const file = fileArray[i]!;\n const uploadInfo = uploadInfos[i]!;\n\n await uploadFileWithProgress(uploadInfo.uploadUrl, file, (progress) => {\n options.onProgress?.(i, progress);\n });\n\n references.push({\n id: uploadInfo.id,\n mediaType: file.type || 'application/octet-stream',\n url: uploadInfo.downloadUrl,\n filename: file.name,\n size: file.size,\n });\n }\n\n return references;\n}\n","import { safeParseStreamEvent, isAbortError, type StreamEvent } from '@octavus/core';\n\n/**\n * Parse SSE stream events.\n *\n * @param response - The HTTP response with SSE body\n * @param signal - Optional abort signal to cancel reading\n */\nexport async function* parseSSEStream(\n response: Response,\n signal?: AbortSignal,\n): AsyncGenerator<StreamEvent, void, unknown> {\n const reader = response.body?.getReader();\n if (!reader) {\n throw new Error('Response body is not readable');\n }\n\n const decoder = new TextDecoder();\n let buffer = '';\n\n try {\n let reading = true;\n while (reading) {\n // Check if aborted before reading\n if (signal?.aborted) {\n return;\n }\n\n let readResult: ReadableStreamReadResult<Uint8Array>;\n try {\n readResult = await reader.read();\n } catch (err) {\n // Handle abort errors gracefully - exit without throwing\n if (isAbortError(err)) {\n return;\n }\n throw err;\n }\n\n const { done, value } = readResult;\n\n if (done) {\n reading = false;\n continue;\n }\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() ?? '';\n\n for (const line of lines) {\n if (line.startsWith('data: ') && line !== 'data: [DONE]') {\n try {\n const parsed = safeParseStreamEvent(JSON.parse(line.slice(6)));\n if (parsed.success) {\n yield parsed.data;\n }\n // Skip malformed events silently\n } catch {\n // Skip malformed JSON - no logging in production\n }\n }\n }\n }\n } finally {\n reader.releaseLock();\n }\n}\n","import type { StreamEvent, ToolResult } from '@octavus/core';\n\n// =============================================================================\n// Base Transport Interface\n// =============================================================================\n\n/**\n * Transport interface for delivering events from server to client.\n *\n * Abstracts the connection mechanism (HTTP/SSE or WebSocket) behind a unified\n * async iterator interface. Use `createHttpTransport` or `createSocketTransport`\n * to create an implementation.\n */\nexport interface Transport {\n /**\n * Trigger the agent and stream events.\n * @param triggerName - The trigger name defined in the agent's protocol\n * @param input - Input parameters for variable substitution\n */\n trigger(triggerName: string, input?: Record<string, unknown>): AsyncIterable<StreamEvent>;\n\n /**\n * Continue execution with tool results after client-side tool handling.\n *\n * @param executionId - The execution ID from the client-tool-request event\n * @param results - All tool results (server + client) to send\n */\n continueWithToolResults(executionId: string, results: ToolResult[]): AsyncIterable<StreamEvent>;\n\n /** Stop the current stream. Safe to call when no stream is active. */\n stop(): void;\n}\n\n// =============================================================================\n// Socket Transport (extends Transport with connection management)\n// =============================================================================\n\n/**\n * Connection states for socket transport.\n *\n * - `disconnected`: Not connected (initial state or after disconnect)\n * - `connecting`: Connection attempt in progress\n * - `connected`: Successfully connected and ready\n * - `error`: Connection failed (check error in listener callback)\n */\nexport type ConnectionState = 'disconnected' | 'connecting' | 'connected' | 'error';\n\n/**\n * Callback for connection state changes.\n */\nexport type ConnectionStateListener = (state: ConnectionState, error?: Error) => void;\n\n/**\n * Socket transport with connection management capabilities.\n *\n * Extends the base Transport interface with methods for managing persistent\n * WebSocket/SockJS connections. Use this when you need:\n * - Eager connection (connect before first message)\n * - Connection status UI indicators\n * - Manual connection lifecycle control\n *\n * Created via `createSocketTransport()`.\n */\nexport interface SocketTransport extends Transport {\n /**\n * Current connection state.\n *\n * - `disconnected`: Not connected (initial state)\n * - `connecting`: Connection in progress\n * - `connected`: Ready to send/receive\n * - `error`: Connection failed\n */\n readonly connectionState: ConnectionState;\n\n /**\n * Subscribe to connection state changes.\n *\n * The listener is called immediately with the current state, then again\n * whenever the state changes.\n *\n * @param listener - Callback invoked on state changes\n * @returns Unsubscribe function\n *\n * @example\n * ```typescript\n * const unsubscribe = transport.onConnectionStateChange((state, error) => {\n * setConnectionState(state);\n * if (error) setConnectionError(error);\n * });\n * ```\n */\n onConnectionStateChange(listener: ConnectionStateListener): () => void;\n\n /**\n * Eagerly establish the connection.\n *\n * By default, socket transport connects lazily on first `trigger()`. Call\n * this method to establish the connection early (e.g., on component mount):\n * - Faster first message response\n * - Show accurate connection status in UI\n * - Handle connection errors before user interaction\n *\n * Safe to call multiple times - resolves immediately if already connected.\n *\n * @example\n * ```tsx\n * useEffect(() => {\n * transport.connect()\n * .then(() => console.log('Connected'))\n * .catch((err) => console.error('Failed:', err));\n *\n * return () => transport.disconnect();\n * }, [transport]);\n * ```\n */\n connect(): Promise<void>;\n\n /**\n * Close the connection and clean up resources.\n *\n * The transport will reconnect automatically on next `trigger()` call.\n */\n disconnect(): void;\n}\n\n// =============================================================================\n// Type Guards\n// =============================================================================\n\n/**\n * Check if a transport is a SocketTransport with connection management.\n *\n * @example\n * ```typescript\n * if (isSocketTransport(transport)) {\n * transport.connect(); // TypeScript knows this is available\n * }\n * ```\n */\nexport function isSocketTransport(transport: Transport): transport is SocketTransport {\n return (\n 'connect' in transport &&\n 'disconnect' in transport &&\n 'connectionState' in transport &&\n 'onConnectionStateChange' in transport\n );\n}\n","import { isAbortError, type ToolResult } from '@octavus/core';\nimport { parseSSEStream } from '@/stream/reader';\nimport type { Transport } from './types';\n\n// =============================================================================\n// Request Types\n// =============================================================================\n\n/** Start a new trigger execution */\nexport interface TriggerRequest {\n type: 'trigger';\n triggerName: string;\n input?: Record<string, unknown>;\n}\n\n/** Continue execution after client-side tool handling */\nexport interface ContinueRequest {\n type: 'continue';\n executionId: string;\n toolResults: ToolResult[];\n}\n\n/** All request types supported by the HTTP transport */\nexport type HttpRequest = TriggerRequest | ContinueRequest;\n\n// =============================================================================\n// Transport Options\n// =============================================================================\n\n/** Request options passed to the request callback */\nexport interface HttpRequestOptions {\n /** Abort signal to cancel the request */\n signal?: AbortSignal;\n}\n\n/**\n * Options for creating an HTTP transport.\n */\nexport interface HttpTransportOptions {\n /**\n * Function to make requests to your backend.\n * Receives a discriminated union with `type` to identify the request kind.\n *\n * @param request - The request payload (check `request.type` for the kind)\n * @param options - Request options including abort signal\n * @returns Response with SSE stream body\n *\n * @example\n * ```typescript\n * request: (payload, options) =>\n * fetch('/api/trigger', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, ...payload }),\n * signal: options?.signal,\n * })\n * ```\n */\n request: (request: HttpRequest, options?: HttpRequestOptions) => Promise<Response>;\n}\n\n// =============================================================================\n// Transport Implementation\n// =============================================================================\n\n/**\n * Create an HTTP transport using native fetch() and SSE parsing.\n * This is the default transport for Next.js and other HTTP-based applications.\n *\n * @example\n * ```typescript\n * const transport = createHttpTransport({\n * request: (payload, options) =>\n * fetch('/api/trigger', {\n * method: 'POST',\n * headers: { 'Content-Type': 'application/json' },\n * body: JSON.stringify({ sessionId, ...payload }),\n * signal: options?.signal,\n * }),\n * });\n * ```\n */\nexport function createHttpTransport(options: HttpTransportOptions): Transport {\n let abortController: AbortController | null = null;\n\n async function* streamResponse(responsePromise: Promise<Response>) {\n try {\n const response = await responsePromise;\n\n if (!response.ok) {\n const errorText = await response.text().catch(() => `Request failed: ${response.status}`);\n throw new Error(errorText);\n }\n\n if (!response.body) {\n throw new Error('Response body is empty');\n }\n\n for await (const event of parseSSEStream(response, abortController!.signal)) {\n if (abortController?.signal.aborted) {\n break;\n }\n yield event;\n }\n } catch (err) {\n if (isAbortError(err)) {\n return;\n }\n throw err;\n }\n }\n\n return {\n async *trigger(triggerName, input) {\n abortController = new AbortController();\n const response = options.request(\n { type: 'trigger', triggerName, input },\n { signal: abortController.signal },\n );\n yield* streamResponse(response);\n },\n\n async *continueWithToolResults(executionId, toolResults) {\n abortController = new AbortController();\n const response = options.request(\n { type: 'continue', executionId, toolResults },\n { signal: abortController.signal },\n );\n yield* streamResponse(response);\n },\n\n stop() {\n abortController?.abort();\n abortController = null;\n },\n };\n}\n","import { safeParseStreamEvent, type StreamEvent, type ToolResult } from '@octavus/core';\nimport type { SocketTransport, ConnectionState, ConnectionStateListener } from './types';\n\n// =============================================================================\n// Types\n// =============================================================================\n\n/**\n * Socket interface compatible with both WebSocket and SockJS.\n *\n * Uses MessageEvent for the message handler, which both WebSocket and SockJS support.\n * The `| undefined` union accommodates SockJS's optional property typing.\n */\nexport interface SocketLike {\n send(data: string): void;\n close(): void;\n readyState: number;\n onmessage: ((event: MessageEvent) => void) | null | undefined;\n onclose: ((event: CloseEvent) => void) | null | undefined;\n}\n\n/** WebSocket readyState constants */\nconst SOCKET_OPEN = 1;\n\n// =============================================================================\n// Transport Options\n// =============================================================================\n\n/**\n * Options for creating a socket transport.\n */\nexport interface SocketTransportOptions {\n /**\n * Function to create and connect the socket.\n * Works directly with WebSocket and SockJS - no wrappers needed.\n *\n * @example Native WebSocket\n * ```typescript\n * connect: () => new Promise((resolve, reject) => {\n * const ws = new WebSocket('wss://api.example.com/stream?sessionId=xxx');\n * ws.onopen = () => resolve(ws);\n * ws.onerror = () => reject(new Error('Connection failed'));\n * })\n * ```\n *\n * @example SockJS\n * ```typescript\n * connect: () => new Promise((resolve, reject) => {\n * const sock = new SockJS('/chat-service');\n * sock.onopen = () => resolve(sock);\n * sock.onerror = () => reject(new Error('Connection failed'));\n * })\n * ```\n */\n connect: () => Promise<SocketLike>;\n\n /**\n * Called for every message received (parsed as JSON).\n * Use this to handle custom (non-Octavus) events.\n * Octavus StreamEvents are handled automatically by the transport.\n *\n * @example\n * ```typescript\n * onMessage: (data) => {\n * const msg = data as { type: string };\n * if (msg.type === 'typing-indicator') {\n * setIsTyping(true);\n * }\n * if (msg.type === 'presence-update') {\n * updatePresence(msg.users);\n * }\n * }\n * ```\n */\n onMessage?: (data: unknown) => void;\n\n /**\n * Called when the socket connection closes.\n * Use this for cleanup or reconnection logic.\n */\n onClose?: () => void;\n}\n\n/**\n * Create a socket transport that works with any WebSocket-like connection.\n * Supports native WebSocket, SockJS, or any compatible socket implementation.\n *\n * The server should send StreamEvent format (same as SSE) over the socket.\n * Unknown events are safely ignored using Zod validation.\n *\n * ## Connection Lifecycle\n *\n * By default, the socket connects **lazily** on the first `send()` call.\n * Use `connect()` to establish the connection eagerly (e.g., on component mount):\n *\n * ```typescript\n * // Eager connection for UI status indicators\n * useEffect(() => {\n * transport.connect()\n * .then(() => console.log('Connected'))\n * .catch((err) => console.error('Failed:', err));\n *\n * return () => transport.disconnect();\n * }, [transport]);\n * ```\n *\n * @example Basic usage with WebSocket\n * ```typescript\n * const transport = createSocketTransport({\n * connect: () => new Promise((resolve, reject) => {\n * const ws = new WebSocket(`wss://api.octavus.ai/stream?sessionId=${sessionId}`);\n * ws.onopen = () => resolve(ws);\n * ws.onerror = () => reject(new Error('Connection failed'));\n * }),\n * });\n * ```\n *\n * @example With SockJS and connection state\n * ```typescript\n * const transport = createSocketTransport({\n * connect: () => new Promise((resolve, reject) => {\n * const sock = new SockJS('/octavus-stream');\n * sock.onopen = () => resolve(sock);\n * sock.onerror = () => reject(new Error('Connection failed'));\n * }),\n * });\n *\n * // Subscribe to connection state changes\n * transport.onConnectionStateChange((state, error) => {\n * setConnectionState(state);\n * if (error) setConnectionError(error);\n * });\n * ```\n */\nexport function createSocketTransport(options: SocketTransportOptions): SocketTransport {\n let socket: SocketLike | null = null;\n let eventQueue: StreamEvent[] = [];\n let eventResolver: ((event: StreamEvent | null) => void) | null = null;\n let isStreaming = false;\n\n let connectionState: ConnectionState = 'disconnected';\n let connectionError: Error | undefined;\n let connectionPromise: Promise<void> | null = null;\n const connectionListeners = new Set<ConnectionStateListener>();\n\n function setConnectionState(state: ConnectionState, error?: Error) {\n connectionState = state;\n connectionError = error;\n connectionListeners.forEach((listener) => listener(state, error));\n }\n\n function enqueueEvent(event: StreamEvent) {\n if (eventResolver) {\n eventResolver(event);\n eventResolver = null;\n } else {\n eventQueue.push(event);\n }\n }\n\n function nextEvent(): Promise<StreamEvent | null> {\n if (eventQueue.length > 0) {\n return Promise.resolve(eventQueue.shift()!);\n }\n if (!isStreaming) {\n return Promise.resolve(null);\n }\n return new Promise((resolve) => {\n eventResolver = resolve;\n });\n }\n\n function setupSocketHandlers(sock: SocketLike): void {\n sock.onmessage = (e: MessageEvent) => {\n try {\n const data: unknown = typeof e.data === 'string' ? JSON.parse(e.data) : e.data;\n\n options.onMessage?.(data);\n\n const result = safeParseStreamEvent(data);\n if (result.success) {\n const event = result.data;\n enqueueEvent(event);\n\n if (event.type === 'finish' || event.type === 'error') {\n isStreaming = false;\n }\n }\n } catch {\n // Malformed JSON, skip\n }\n };\n\n sock.onclose = () => {\n socket = null;\n connectionPromise = null;\n setConnectionState('disconnected');\n options.onClose?.();\n\n isStreaming = false;\n if (eventResolver) {\n eventResolver(null);\n eventResolver = null;\n }\n };\n }\n\n async function ensureConnected(): Promise<void> {\n // Already connected\n if (socket?.readyState === SOCKET_OPEN) {\n return;\n }\n\n // Connection in progress - wait for it\n if (connectionPromise) {\n await connectionPromise;\n return;\n }\n\n // Start new connection\n setConnectionState('connecting');\n\n connectionPromise = (async () => {\n try {\n const sock = await options.connect();\n socket = sock;\n setupSocketHandlers(sock);\n setConnectionState('connected');\n } catch (err) {\n socket = null;\n connectionPromise = null;\n const error = err instanceof Error ? err : new Error('Connection failed');\n setConnectionState('error', error);\n throw error;\n }\n })();\n\n await connectionPromise;\n }\n\n return {\n // =========================================================================\n // Connection Management\n // =========================================================================\n\n get connectionState(): ConnectionState {\n return connectionState;\n },\n\n onConnectionStateChange(listener: ConnectionStateListener): () => void {\n connectionListeners.add(listener);\n // Immediately notify with current state\n listener(connectionState, connectionError);\n return () => connectionListeners.delete(listener);\n },\n\n async connect(): Promise<void> {\n await ensureConnected();\n },\n\n disconnect(): void {\n if (socket) {\n socket.close();\n socket = null;\n }\n connectionPromise = null;\n isStreaming = false;\n if (eventResolver) {\n eventResolver(null);\n eventResolver = null;\n }\n setConnectionState('disconnected');\n },\n\n // =========================================================================\n // Streaming\n // =========================================================================\n\n async *trigger(triggerName, input) {\n await ensureConnected();\n\n eventQueue = [];\n eventResolver = null; // Clear any pending resolver\n isStreaming = true;\n\n // Note: clientToolResults not sent here - socket uses sendClientToolResults() for continuation\n socket!.send(\n JSON.stringify({\n type: 'trigger',\n triggerName,\n input,\n }),\n );\n\n while (true) {\n const event = await nextEvent();\n if (event === null) break;\n yield event;\n if (event.type === 'finish' || event.type === 'error') break;\n }\n },\n\n stop() {\n if (socket?.readyState === SOCKET_OPEN) {\n socket.send(JSON.stringify({ type: 'stop' }));\n }\n isStreaming = false;\n if (eventResolver) {\n eventResolver(null);\n eventResolver = null;\n }\n },\n\n /**\n * Continue execution with tool results after client-side tool handling.\n * @param executionId - The execution ID from the client-tool-request event\n * @param toolResults - All tool results (server + client) to send\n */\n async *continueWithToolResults(executionId: string, toolResults: ToolResult[]) {\n await ensureConnected();\n\n eventQueue = [];\n eventResolver = null; // Clear any pending resolver from previous operation\n isStreaming = true;\n\n socket!.send(\n JSON.stringify({\n type: 'continue',\n executionId,\n toolResults,\n }),\n );\n\n while (true) {\n const event = await nextEvent();\n if (event === null) break;\n yield event;\n if (event.type === 'finish' || event.type === 'error') break;\n }\n },\n };\n}\n","export {\n OctavusChat,\n type OctavusChatOptions,\n type ChatStatus,\n type UserMessageInput,\n type ClientToolContext,\n type ClientToolHandler,\n type InteractiveTool,\n} from './chat';\n\nexport { uploadFiles, type UploadFilesOptions, type UploadUrlsResponse } from './files';\n\nexport { parseSSEStream } from './stream/reader';\n\n// Transport exports\nexport {\n createHttpTransport,\n createSocketTransport,\n isSocketTransport,\n type Transport,\n type SocketTransport,\n type ConnectionState,\n type ConnectionStateListener,\n type HttpTransportOptions,\n type HttpRequestOptions,\n type HttpRequest,\n type TriggerRequest,\n type ContinueRequest,\n type SocketLike,\n type SocketTransportOptions,\n} from './transports';\n\nexport type * from '@octavus/core';\nexport {\n // Error classes\n AppError,\n NotFoundError,\n ValidationError,\n ConflictError,\n ForbiddenError,\n OctavusError,\n // Error type guards\n isRateLimitError,\n isAuthenticationError,\n isProviderError,\n isToolError,\n isRetryableError,\n isValidationError,\n // Error event helpers\n createErrorEvent,\n errorToStreamEvent,\n createInternalErrorEvent,\n createApiErrorEvent,\n // Utilities\n generateId,\n isAbortError,\n // Thread helpers\n MAIN_THREAD,\n resolveThread,\n isMainThread,\n threadForPart,\n isOtherThread,\n // Type guards\n isFileReference,\n isFileReferenceArray,\n // Safe parse helpers\n safeParseStreamEvent,\n safeParseUIMessage,\n safeParseUIMessages,\n // Skills\n OCTAVUS_SKILL_TOOLS,\n isOctavusSkillTool,\n getSkillSlugFromToolCall,\n} from '@octavus/core';\n"],"mappings":"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","names":["isAbortError","isAbortError","safeParseStreamEvent","OctavusError","generateId","isAbortError","threadForPart","isFileReferenceArray","safeParseStreamEvent"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@octavus/client-sdk",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.8.0",
|
|
4
4
|
"description": "Framework-agnostic client SDK for Octavus agents",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"author": "Octavus AI <dev@octavus.ai>",
|
|
@@ -39,7 +39,7 @@
|
|
|
39
39
|
"access": "public"
|
|
40
40
|
},
|
|
41
41
|
"dependencies": {
|
|
42
|
-
"@octavus/core": "^2.
|
|
42
|
+
"@octavus/core": "^2.8.0"
|
|
43
43
|
},
|
|
44
44
|
"devDependencies": {
|
|
45
45
|
"tsup": "^8.3.5",
|