@copilotkitnext/angular 0.0.0-max-changeset-20260109174803

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.
Files changed (118) hide show
  1. package/LICENSE +10 -0
  2. package/README.md +530 -0
  3. package/dist/LICENSE +10 -0
  4. package/dist/README.md +530 -0
  5. package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
  6. package/dist/esm2022/index.mjs +2 -0
  7. package/dist/esm2022/lib/agent-context.mjs +25 -0
  8. package/dist/esm2022/lib/agent.mjs +73 -0
  9. package/dist/esm2022/lib/chat-config.mjs +35 -0
  10. package/dist/esm2022/lib/chat-state.mjs +18 -0
  11. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-buttons.mjs +344 -0
  12. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-renderer.mjs +260 -0
  13. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-toolbar.mjs +22 -0
  14. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.mjs +415 -0
  15. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
  16. package/dist/esm2022/lib/components/chat/copilot-chat-audio-recorder.mjs +196 -0
  17. package/dist/esm2022/lib/components/chat/copilot-chat-buttons.mjs +299 -0
  18. package/dist/esm2022/lib/components/chat/copilot-chat-input-defaults.mjs +39 -0
  19. package/dist/esm2022/lib/components/chat/copilot-chat-input.mjs +634 -0
  20. package/dist/esm2022/lib/components/chat/copilot-chat-input.types.mjs +10 -0
  21. package/dist/esm2022/lib/components/chat/copilot-chat-message-view-cursor.mjs +27 -0
  22. package/dist/esm2022/lib/components/chat/copilot-chat-message-view.mjs +268 -0
  23. package/dist/esm2022/lib/components/chat/copilot-chat-message-view.types.mjs +2 -0
  24. package/dist/esm2022/lib/components/chat/copilot-chat-textarea.mjs +139 -0
  25. package/dist/esm2022/lib/components/chat/copilot-chat-tool-calls-view.mjs +36 -0
  26. package/dist/esm2022/lib/components/chat/copilot-chat-toolbar.mjs +20 -0
  27. package/dist/esm2022/lib/components/chat/copilot-chat-tools-menu.mjs +203 -0
  28. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-branch-navigation.mjs +98 -0
  29. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-buttons.mjs +182 -0
  30. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-renderer.mjs +28 -0
  31. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-toolbar.mjs +25 -0
  32. package/dist/esm2022/lib/components/chat/copilot-chat-user-message.mjs +308 -0
  33. package/dist/esm2022/lib/components/chat/copilot-chat-user-message.types.mjs +2 -0
  34. package/dist/esm2022/lib/components/chat/copilot-chat-view-disclaimer.mjs +48 -0
  35. package/dist/esm2022/lib/components/chat/copilot-chat-view-feather.mjs +41 -0
  36. package/dist/esm2022/lib/components/chat/copilot-chat-view-handlers.mjs +19 -0
  37. package/dist/esm2022/lib/components/chat/copilot-chat-view-input-container.mjs +96 -0
  38. package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.mjs +89 -0
  39. package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-view.mjs +456 -0
  40. package/dist/esm2022/lib/components/chat/copilot-chat-view.mjs +404 -0
  41. package/dist/esm2022/lib/components/chat/copilot-chat-view.types.mjs +2 -0
  42. package/dist/esm2022/lib/components/chat/copilot-chat.mjs +167 -0
  43. package/dist/esm2022/lib/config.mjs +9 -0
  44. package/dist/esm2022/lib/copilotkit.mjs +115 -0
  45. package/dist/esm2022/lib/directives/copilotkit-agent-context.mjs +130 -0
  46. package/dist/esm2022/lib/directives/stick-to-bottom.mjs +170 -0
  47. package/dist/esm2022/lib/directives/tooltip.mjs +217 -0
  48. package/dist/esm2022/lib/human-in-the-loop.mjs +19 -0
  49. package/dist/esm2022/lib/render-tool-calls.mjs +138 -0
  50. package/dist/esm2022/lib/resize-observer.mjs +152 -0
  51. package/dist/esm2022/lib/scroll-position.mjs +124 -0
  52. package/dist/esm2022/lib/slots/copilot-slot.mjs +156 -0
  53. package/dist/esm2022/lib/slots/index.mjs +4 -0
  54. package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
  55. package/dist/esm2022/lib/slots/slot.utils.mjs +235 -0
  56. package/dist/esm2022/lib/tools.mjs +31 -0
  57. package/dist/esm2022/lib/utils.mjs +10 -0
  58. package/dist/esm2022/public-api.mjs +48 -0
  59. package/dist/fesm2022/copilotkitnext-angular.mjs +6310 -0
  60. package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -0
  61. package/dist/index.d.ts +1 -0
  62. package/dist/lib/agent-context.d.ts +12 -0
  63. package/dist/lib/agent.d.ts +68 -0
  64. package/dist/lib/chat-config.d.ts +23 -0
  65. package/dist/lib/chat-state.d.ts +10 -0
  66. package/dist/lib/components/chat/copilot-chat-assistant-message-buttons.d.ts +68 -0
  67. package/dist/lib/components/chat/copilot-chat-assistant-message-renderer.d.ts +26 -0
  68. package/dist/lib/components/chat/copilot-chat-assistant-message-toolbar.d.ts +7 -0
  69. package/dist/lib/components/chat/copilot-chat-assistant-message.d.ts +208 -0
  70. package/dist/lib/components/chat/copilot-chat-assistant-message.types.d.ts +31 -0
  71. package/dist/lib/components/chat/copilot-chat-audio-recorder.d.ts +40 -0
  72. package/dist/lib/components/chat/copilot-chat-buttons.d.ts +65 -0
  73. package/dist/lib/components/chat/copilot-chat-input-defaults.d.ts +38 -0
  74. package/dist/lib/components/chat/copilot-chat-input.d.ts +133 -0
  75. package/dist/lib/components/chat/copilot-chat-input.types.d.ts +129 -0
  76. package/dist/lib/components/chat/copilot-chat-message-view-cursor.d.ts +11 -0
  77. package/dist/lib/components/chat/copilot-chat-message-view.d.ts +430 -0
  78. package/dist/lib/components/chat/copilot-chat-message-view.types.d.ts +24 -0
  79. package/dist/lib/components/chat/copilot-chat-textarea.d.ts +41 -0
  80. package/dist/lib/components/chat/copilot-chat-tool-calls-view.d.ts +70 -0
  81. package/dist/lib/components/chat/copilot-chat-toolbar.d.ts +7 -0
  82. package/dist/lib/components/chat/copilot-chat-tools-menu.d.ts +20 -0
  83. package/dist/lib/components/chat/copilot-chat-user-message-branch-navigation.d.ts +35 -0
  84. package/dist/lib/components/chat/copilot-chat-user-message-buttons.d.ts +35 -0
  85. package/dist/lib/components/chat/copilot-chat-user-message-renderer.d.ts +8 -0
  86. package/dist/lib/components/chat/copilot-chat-user-message-toolbar.d.ts +7 -0
  87. package/dist/lib/components/chat/copilot-chat-user-message.d.ts +71 -0
  88. package/dist/lib/components/chat/copilot-chat-user-message.types.d.ts +27 -0
  89. package/dist/lib/components/chat/copilot-chat-view-disclaimer.d.ts +15 -0
  90. package/dist/lib/components/chat/copilot-chat-view-feather.d.ts +15 -0
  91. package/dist/lib/components/chat/copilot-chat-view-handlers.d.ts +11 -0
  92. package/dist/lib/components/chat/copilot-chat-view-input-container.d.ts +23 -0
  93. package/dist/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.d.ts +16 -0
  94. package/dist/lib/components/chat/copilot-chat-view-scroll-view.d.ts +129 -0
  95. package/dist/lib/components/chat/copilot-chat-view.d.ts +284 -0
  96. package/dist/lib/components/chat/copilot-chat-view.types.d.ts +42 -0
  97. package/dist/lib/components/chat/copilot-chat.d.ts +82 -0
  98. package/dist/lib/config.d.ts +16 -0
  99. package/dist/lib/copilotkit.d.ts +29 -0
  100. package/dist/lib/directives/copilotkit-agent-context.d.ts +68 -0
  101. package/dist/lib/directives/stick-to-bottom.d.ts +62 -0
  102. package/dist/lib/directives/tooltip.d.ts +33 -0
  103. package/dist/lib/human-in-the-loop.d.ts +13 -0
  104. package/dist/lib/render-tool-calls.d.ts +90 -0
  105. package/dist/lib/resize-observer.d.ts +44 -0
  106. package/dist/lib/scroll-position.d.ts +50 -0
  107. package/dist/lib/slots/copilot-slot.d.ts +34 -0
  108. package/dist/lib/slots/index.d.ts +3 -0
  109. package/dist/lib/slots/slot.types.d.ts +55 -0
  110. package/dist/lib/slots/slot.utils.d.ts +110 -0
  111. package/dist/lib/tools.d.ts +65 -0
  112. package/dist/lib/utils.d.ts +6 -0
  113. package/dist/public-api.d.ts +47 -0
  114. package/dist/styles.css +1882 -0
  115. package/eslint.config.mjs +20 -0
  116. package/package.json +101 -0
  117. package/tsconfig.json +33 -0
  118. package/vitest.config.mts +45 -0
@@ -0,0 +1,167 @@
1
+ import { Component, input, ChangeDetectionStrategy, ViewEncapsulation, signal, effect, ChangeDetectorRef, Injector, computed, inject, } from "@angular/core";
2
+ import { CommonModule } from "@angular/common";
3
+ import { CopilotChatView } from "./copilot-chat-view";
4
+ import { DEFAULT_AGENT_ID, randomUUID } from "@copilotkitnext/shared";
5
+ import { injectAgentStore } from "../../agent";
6
+ import { ChatState } from "../../chat-state";
7
+ import * as i0 from "@angular/core";
8
+ /**
9
+ * CopilotChat component - Angular equivalent of React's <CopilotChat>
10
+ * Provides a complete chat interface that wires an agent to the chat view
11
+ *
12
+ * @example
13
+ * ```html
14
+ * <copilot-chat [agentId]="'default'" [threadId]="'abc123'"></copilot-chat>
15
+ * ```
16
+ */
17
+ export class CopilotChat {
18
+ inputValue = signal("");
19
+ agentId = input();
20
+ threadId = input();
21
+ inputComponent = input();
22
+ resolvedAgentId = computed(() => this.agentId() ?? DEFAULT_AGENT_ID);
23
+ agentStore = injectAgentStore(this.resolvedAgentId);
24
+ // readonly chatConfig = injectChatConfig();
25
+ cdr = inject(ChangeDetectorRef);
26
+ injector = inject(Injector);
27
+ messages = computed(() => this.agentStore()?.messages());
28
+ isRunning = computed(() => this.agentStore()?.isRunning());
29
+ showCursor = signal(false);
30
+ generatedThreadId = randomUUID();
31
+ hasConnectedOnce = false;
32
+ constructor() {
33
+ // Connect once when agent becomes available
34
+ // Connect once when agent becomes available
35
+ effect(() => {
36
+ const a = this.agentStore()?.agent;
37
+ if (!a)
38
+ return;
39
+ // Apply thread id when agent is available
40
+ a.threadId = this.threadId() || this.generatedThreadId;
41
+ if (!this.hasConnectedOnce) {
42
+ this.hasConnectedOnce = true;
43
+ if ("isCopilotKitAgent" in a) {
44
+ this.connectToAgent(a);
45
+ }
46
+ else {
47
+ // Non-CopilotKit agent: nothing to connect; keep default cursor state
48
+ }
49
+ }
50
+ }, { allowSignalWrites: true });
51
+ // Keep agent threadId in sync with input
52
+ effect(() => {
53
+ const a = this.agentStore()?.agent;
54
+ if (a) {
55
+ a.threadId = this.threadId() || this.generatedThreadId;
56
+ }
57
+ });
58
+ }
59
+ async connectToAgent(agent) {
60
+ if (!agent)
61
+ return;
62
+ this.showCursor.set(true);
63
+ this.cdr.markForCheck();
64
+ try {
65
+ await agent.runAgent({ forwardedProps: { __copilotkitConnect: true } }, {
66
+ onTextMessageStartEvent: () => {
67
+ this.showCursor.set(false);
68
+ this.cdr.detectChanges();
69
+ },
70
+ onToolCallStartEvent: () => {
71
+ this.showCursor.set(false);
72
+ this.cdr.detectChanges();
73
+ },
74
+ });
75
+ this.showCursor.set(false);
76
+ this.cdr.markForCheck();
77
+ }
78
+ catch (error) {
79
+ console.error("Failed to connect to agent:", error);
80
+ this.showCursor.set(false);
81
+ this.cdr.markForCheck();
82
+ }
83
+ }
84
+ async submitInput(value) {
85
+ const agent = this.agentStore()?.agent;
86
+ if (!agent || !value.trim())
87
+ return;
88
+ // Add user message
89
+ const userMessage = {
90
+ id: randomUUID(),
91
+ role: "user",
92
+ content: value,
93
+ };
94
+ agent.addMessage(userMessage);
95
+ // Clear the input
96
+ this.inputValue.set("");
97
+ // Show cursor while processing
98
+ this.showCursor.set(true);
99
+ this.cdr.markForCheck();
100
+ // Run the agent with named subscriber callbacks
101
+ try {
102
+ await agent.runAgent({}, {
103
+ onTextMessageStartEvent: () => {
104
+ this.showCursor.set(false);
105
+ this.cdr.detectChanges();
106
+ },
107
+ onToolCallStartEvent: () => {
108
+ this.showCursor.set(false);
109
+ this.cdr.detectChanges();
110
+ },
111
+ });
112
+ }
113
+ catch (error) {
114
+ console.error("Agent run error:", error);
115
+ }
116
+ finally {
117
+ this.showCursor.set(false);
118
+ this.cdr.markForCheck();
119
+ }
120
+ }
121
+ changeInput(value) {
122
+ this.inputValue.set(value);
123
+ }
124
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChat, deps: [], target: i0.ɵɵFactoryTarget.Component });
125
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: CopilotChat, isStandalone: true, selector: "copilot-chat", inputs: { agentId: { classPropertyName: "agentId", publicName: "agentId", isSignal: true, isRequired: false, transformFunction: null }, threadId: { classPropertyName: "threadId", publicName: "threadId", isSignal: true, isRequired: false, transformFunction: null }, inputComponent: { classPropertyName: "inputComponent", publicName: "inputComponent", isSignal: true, isRequired: false, transformFunction: null } }, providers: [
126
+ {
127
+ provide: ChatState,
128
+ useExisting: CopilotChat,
129
+ },
130
+ ], ngImport: i0, template: `
131
+ <copilot-chat-view
132
+ [messages]="messages() ?? []"
133
+ [autoScroll]="true"
134
+ [messageViewClass]="'w-full'"
135
+ [showCursor]="showCursor()"
136
+ [inputComponent]="inputComponent()"
137
+ >
138
+ </copilot-chat-view>
139
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: CopilotChatView, selector: "copilot-chat-view", inputs: ["messages", "autoScroll", "showCursor", "messageViewComponent", "messageViewTemplate", "messageViewClass", "scrollViewComponent", "scrollViewTemplate", "scrollViewClass", "scrollToBottomButtonComponent", "scrollToBottomButtonTemplate", "scrollToBottomButtonClass", "inputComponent", "inputTemplate", "inputContainerComponent", "inputContainerTemplate", "inputContainerClass", "featherComponent", "featherTemplate", "featherClass", "disclaimerComponent", "disclaimerTemplate", "disclaimerClass", "disclaimerText"], outputs: ["assistantMessageThumbsUp", "assistantMessageThumbsDown", "assistantMessageReadAloud", "assistantMessageRegenerate", "userMessageCopy", "userMessageEdit"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
140
+ }
141
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChat, decorators: [{
142
+ type: Component,
143
+ args: [{
144
+ selector: "copilot-chat",
145
+ standalone: true,
146
+ imports: [CommonModule, CopilotChatView],
147
+ changeDetection: ChangeDetectionStrategy.OnPush,
148
+ encapsulation: ViewEncapsulation.None,
149
+ template: `
150
+ <copilot-chat-view
151
+ [messages]="messages() ?? []"
152
+ [autoScroll]="true"
153
+ [messageViewClass]="'w-full'"
154
+ [showCursor]="showCursor()"
155
+ [inputComponent]="inputComponent()"
156
+ >
157
+ </copilot-chat-view>
158
+ `,
159
+ providers: [
160
+ {
161
+ provide: ChatState,
162
+ useExisting: CopilotChat,
163
+ },
164
+ ],
165
+ }]
166
+ }], ctorParameters: () => [] });
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat.js","sourceRoot":"","sources":["../../../../../src/lib/components/chat/copilot-chat.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  input,\n  ChangeDetectionStrategy,\n  ViewEncapsulation,\n  signal,\n  effect,\n  ChangeDetectorRef,\n  Injector,\n  Type,\n  computed,\n  inject,\n} from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { CopilotChatView } from \"./copilot-chat-view\";\n\nimport { DEFAULT_AGENT_ID, randomUUID } from \"@copilotkitnext/shared\";\nimport { Message, AbstractAgent } from \"@ag-ui/client\";\nimport { injectAgentStore } from \"../../agent\";\nimport { ChatState } from \"../../chat-state\";\n\n/**\n * CopilotChat component - Angular equivalent of React's <CopilotChat>\n * Provides a complete chat interface that wires an agent to the chat view\n *\n * @example\n * ```html\n * <copilot-chat [agentId]=\"'default'\" [threadId]=\"'abc123'\"></copilot-chat>\n * ```\n */\n@Component({\n  selector: \"copilot-chat\",\n  standalone: true,\n  imports: [CommonModule, CopilotChatView],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <copilot-chat-view\n      [messages]=\"messages() ?? []\"\n      [autoScroll]=\"true\"\n      [messageViewClass]=\"'w-full'\"\n      [showCursor]=\"showCursor()\"\n      [inputComponent]=\"inputComponent()\"\n    >\n    </copilot-chat-view>\n  `,\n  providers: [\n    {\n      provide: ChatState,\n      useExisting: CopilotChat,\n    },\n  ],\n})\nexport class CopilotChat implements ChatState {\n  readonly inputValue = signal<string>(\"\");\n  readonly agentId = input<string | undefined>();\n  readonly threadId = input<string | undefined>();\n  readonly inputComponent = input<Type<any> | undefined>();\n  private readonly resolvedAgentId = computed(\n    () => this.agentId() ?? DEFAULT_AGENT_ID\n  );\n  readonly agentStore = injectAgentStore(this.resolvedAgentId);\n  // readonly chatConfig = injectChatConfig();\n  readonly cdr = inject(ChangeDetectorRef);\n  readonly injector = inject(Injector);\n\n  protected messages = computed(() => this.agentStore()?.messages());\n  protected isRunning = computed(() => this.agentStore()?.isRunning());\n  protected showCursor = signal<boolean>(false);\n\n  private generatedThreadId: string = randomUUID();\n  private hasConnectedOnce = false;\n\n  constructor() {\n    // Connect once when agent becomes available\n    // Connect once when agent becomes available\n    effect(\n      () => {\n        const a = this.agentStore()?.agent;\n        if (!a) return;\n        // Apply thread id when agent is available\n        a.threadId = this.threadId() || this.generatedThreadId;\n        if (!this.hasConnectedOnce) {\n          this.hasConnectedOnce = true;\n          if (\"isCopilotKitAgent\" in (a as any)) {\n            this.connectToAgent(a);\n          } else {\n            // Non-CopilotKit agent: nothing to connect; keep default cursor state\n          }\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    // Keep agent threadId in sync with input\n    effect(() => {\n      const a = this.agentStore()?.agent;\n      if (a) {\n        a.threadId = this.threadId() || this.generatedThreadId;\n      }\n    });\n  }\n\n  private async connectToAgent(agent: AbstractAgent): Promise<void> {\n    if (!agent) return;\n\n    this.showCursor.set(true);\n    this.cdr.markForCheck();\n\n    try {\n      await agent.runAgent(\n        { forwardedProps: { __copilotkitConnect: true } },\n        {\n          onTextMessageStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n          onToolCallStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n        }\n      );\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    } catch (error) {\n      console.error(\"Failed to connect to agent:\", error);\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    }\n  }\n\n  async submitInput(value: string): Promise<void> {\n    const agent = this.agentStore()?.agent;\n    if (!agent || !value.trim()) return;\n\n    // Add user message\n    const userMessage: Message = {\n      id: randomUUID(),\n      role: \"user\",\n      content: value,\n    };\n    agent.addMessage(userMessage);\n\n    // Clear the input\n    this.inputValue.set(\"\");\n\n    // Show cursor while processing\n    this.showCursor.set(true);\n    this.cdr.markForCheck();\n\n    // Run the agent with named subscriber callbacks\n    try {\n      await agent.runAgent(\n        {},\n        {\n          onTextMessageStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n          onToolCallStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n        }\n      );\n    } catch (error) {\n      console.error(\"Agent run error:\", error);\n    } finally {\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    }\n  }\n\n  changeInput(value: string): void {\n    this.inputValue.set(value);\n  }\n}\n"]}
@@ -0,0 +1,9 @@
1
+ import { inject, InjectionToken } from "@angular/core";
2
+ export const COPILOT_KIT_CONFIG = new InjectionToken("COPILOT_KIT_CONFIG");
3
+ export function injectCopilotKitConfig() {
4
+ return inject(COPILOT_KIT_CONFIG);
5
+ }
6
+ export function provideCopilotKit(config) {
7
+ return { provide: COPILOT_KIT_CONFIG, useValue: config };
8
+ }
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2xpYi9jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxjQUFjLEVBQVksTUFBTSxlQUFlLENBQUM7QUFvQmpFLE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLElBQUksY0FBYyxDQUNsRCxvQkFBb0IsQ0FDckIsQ0FBQztBQUVGLE1BQU0sVUFBVSxzQkFBc0I7SUFDcEMsT0FBTyxNQUFNLENBQUMsa0JBQWtCLENBQUMsQ0FBQztBQUNwQyxDQUFDO0FBRUQsTUFBTSxVQUFVLGlCQUFpQixDQUFDLE1BQXdCO0lBQ3hELE9BQU8sRUFBRSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsUUFBUSxFQUFFLE1BQU0sRUFBRSxDQUFDO0FBQzNELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBpbmplY3QsIEluamVjdGlvblRva2VuLCBQcm92aWRlciB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBBYnN0cmFjdEFnZW50IH0gZnJvbSBcIkBhZy11aS9jbGllbnRcIjtcbmltcG9ydCB7XG4gIENsaWVudFRvb2wsXG4gIEZyb250ZW5kVG9vbENvbmZpZyxcbiAgSHVtYW5JblRoZUxvb3BDb25maWcsXG4gIFJlbmRlclRvb2xDYWxsQ29uZmlnLFxufSBmcm9tIFwiLi90b29sc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIENvcGlsb3RLaXRDb25maWcge1xuICBydW50aW1lVXJsPzogc3RyaW5nO1xuICBoZWFkZXJzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbiAgcHJvcGVydGllcz86IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBhZ2VudHM/OiBSZWNvcmQ8c3RyaW5nLCBBYnN0cmFjdEFnZW50PjtcbiAgdG9vbHM/OiBDbGllbnRUb29sW107XG4gIHJlbmRlclRvb2xDYWxscz86IFJlbmRlclRvb2xDYWxsQ29uZmlnW107XG4gIGZyb250ZW5kVG9vbHM/OiBGcm9udGVuZFRvb2xDb25maWdbXTtcbiAgaHVtYW5JblRoZUxvb3A/OiBIdW1hbkluVGhlTG9vcENvbmZpZ1tdO1xufVxuXG5leHBvcnQgY29uc3QgQ09QSUxPVF9LSVRfQ09ORklHID0gbmV3IEluamVjdGlvblRva2VuPENvcGlsb3RLaXRDb25maWc+KFxuICBcIkNPUElMT1RfS0lUX0NPTkZJR1wiXG4pO1xuXG5leHBvcnQgZnVuY3Rpb24gaW5qZWN0Q29waWxvdEtpdENvbmZpZygpOiBDb3BpbG90S2l0Q29uZmlnIHtcbiAgcmV0dXJuIGluamVjdChDT1BJTE9UX0tJVF9DT05GSUcpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZUNvcGlsb3RLaXQoY29uZmlnOiBDb3BpbG90S2l0Q29uZmlnKTogUHJvdmlkZXIge1xuICByZXR1cm4geyBwcm92aWRlOiBDT1BJTE9UX0tJVF9DT05GSUcsIHVzZVZhbHVlOiBjb25maWcgfTtcbn1cbiJdfQ==
@@ -0,0 +1,115 @@
1
+ import { CopilotKitCore } from "@copilotkitnext/core";
2
+ import { Injectable, Injector, runInInjectionContext, signal, inject } from "@angular/core";
3
+ import { injectCopilotKitConfig } from "./config";
4
+ import { HumanInTheLoop } from "./human-in-the-loop";
5
+ import * as i0 from "@angular/core";
6
+ export class CopilotKit {
7
+ #config = injectCopilotKitConfig();
8
+ #hitl = inject(HumanInTheLoop);
9
+ #rootInjector = inject(Injector);
10
+ #agents = signal(this.#config.agents ?? {});
11
+ agents = this.#agents.asReadonly();
12
+ core = new CopilotKitCore({
13
+ runtimeUrl: this.#config.runtimeUrl,
14
+ headers: this.#config.headers,
15
+ properties: this.#config.properties,
16
+ agents__unsafe_dev_only: this.#config.agents,
17
+ tools: this.#config.tools,
18
+ });
19
+ #toolCallRenderConfigs = signal([]);
20
+ #clientToolCallRenderConfigs = signal([]);
21
+ #humanInTheLoopToolRenderConfigs = signal([]);
22
+ toolCallRenderConfigs = this.#toolCallRenderConfigs.asReadonly();
23
+ clientToolCallRenderConfigs = this.#clientToolCallRenderConfigs.asReadonly();
24
+ humanInTheLoopToolRenderConfigs = this.#humanInTheLoopToolRenderConfigs.asReadonly();
25
+ constructor() {
26
+ this.#config.renderToolCalls?.forEach((renderConfig) => {
27
+ this.addRenderToolCall(renderConfig);
28
+ });
29
+ this.#config.tools?.forEach((tool) => {
30
+ if (tool.renderer && tool.parameters) {
31
+ this.addRenderToolCall({
32
+ name: tool.name,
33
+ args: tool.parameters,
34
+ component: tool.renderer,
35
+ agentId: tool.agentId,
36
+ });
37
+ }
38
+ });
39
+ this.#config.frontendTools?.forEach((clientTool) => {
40
+ this.addFrontendTool({ ...clientTool, injector: this.#rootInjector });
41
+ });
42
+ this.#config.humanInTheLoop?.forEach((humanInTheLoopTool) => {
43
+ this.addHumanInTheLoop(humanInTheLoopTool);
44
+ });
45
+ this.core.subscribe({
46
+ onAgentsChanged: () => {
47
+ this.#agents.set(this.core.agents);
48
+ },
49
+ });
50
+ }
51
+ #bindClientTool(clientToolWithInjector) {
52
+ const { injector, handler, ...frontendCandidate } = clientToolWithInjector;
53
+ return {
54
+ ...frontendCandidate,
55
+ handler: (args) => runInInjectionContext(injector, () => handler(args)),
56
+ };
57
+ }
58
+ addFrontendTool(clientToolWithInjector) {
59
+ const tool = this.#bindClientTool(clientToolWithInjector);
60
+ this.core.addTool(tool);
61
+ this.#clientToolCallRenderConfigs.update((current) => [...current, clientToolWithInjector]);
62
+ }
63
+ addRenderToolCall(renderConfig) {
64
+ this.#toolCallRenderConfigs.update((current) => [...current, renderConfig]);
65
+ }
66
+ #bindHumanInTheLoopTool(humanInTheLoopTool) {
67
+ return {
68
+ ...humanInTheLoopTool,
69
+ handler: (args, toolCall) => {
70
+ return this.#hitl.onResult(toolCall.id, humanInTheLoopTool.name);
71
+ },
72
+ };
73
+ }
74
+ addHumanInTheLoop(humanInTheLoopTool) {
75
+ this.#humanInTheLoopToolRenderConfigs.update((current) => [...current, humanInTheLoopTool]);
76
+ const tool = this.#bindHumanInTheLoopTool(humanInTheLoopTool);
77
+ this.core.addTool(tool);
78
+ }
79
+ #isSameAgentId(target, agentId) {
80
+ if (agentId) {
81
+ return target.agentId === agentId;
82
+ }
83
+ return true;
84
+ }
85
+ removeTool(toolName, agentId) {
86
+ this.core.removeTool(toolName);
87
+ this.#clientToolCallRenderConfigs.update((current) => current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)));
88
+ this.#humanInTheLoopToolRenderConfigs.update((current) => current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)));
89
+ this.#toolCallRenderConfigs.update((current) => current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)));
90
+ }
91
+ getAgent(agentId) {
92
+ return this.core.getAgent(agentId);
93
+ }
94
+ updateRuntime(options) {
95
+ if (options.runtimeUrl !== undefined) {
96
+ this.core.setRuntimeUrl(options.runtimeUrl);
97
+ }
98
+ if (options.headers !== undefined) {
99
+ this.core.setHeaders(options.headers);
100
+ }
101
+ if (options.properties !== undefined) {
102
+ this.core.setProperties(options.properties);
103
+ }
104
+ if (options.agents !== undefined) {
105
+ this.core.setAgents__unsafe_dev_only(options.agents);
106
+ }
107
+ }
108
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKit, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
109
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKit, providedIn: "root" });
110
+ }
111
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKit, decorators: [{
112
+ type: Injectable,
113
+ args: [{ providedIn: "root" }]
114
+ }], ctorParameters: () => [] });
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilotkit.js","sourceRoot":"","sources":["../../../src/lib/copilotkit.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractAgent } from \"@ag-ui/client\";\nimport { FrontendTool, CopilotKitCore } from \"@copilotkitnext/core\";\nimport { Injectable, Injector, Signal, WritableSignal, runInInjectionContext, signal, inject } from \"@angular/core\";\nimport { FrontendToolConfig, HumanInTheLoopConfig, RenderToolCallConfig } from \"./tools\";\nimport { injectCopilotKitConfig } from \"./config\";\nimport { HumanInTheLoop } from \"./human-in-the-loop\";\n\n@Injectable({ providedIn: \"root\" })\nexport class CopilotKit {\n  readonly #config = injectCopilotKitConfig();\n  readonly #hitl = inject(HumanInTheLoop);\n  readonly #rootInjector = inject(Injector);\n  readonly #agents = signal<Record<string, AbstractAgent>>(this.#config.agents ?? {});\n  readonly agents = this.#agents.asReadonly();\n\n  readonly core = new CopilotKitCore({\n    runtimeUrl: this.#config.runtimeUrl,\n    headers: this.#config.headers,\n    properties: this.#config.properties,\n    agents__unsafe_dev_only: this.#config.agents,\n    tools: this.#config.tools,\n  });\n\n  readonly #toolCallRenderConfigs: WritableSignal<RenderToolCallConfig[]> = signal([]);\n  readonly #clientToolCallRenderConfigs: WritableSignal<FrontendToolConfig[]> = signal([]);\n  readonly #humanInTheLoopToolRenderConfigs: WritableSignal<HumanInTheLoopConfig[]> = signal([]);\n\n  readonly toolCallRenderConfigs: Signal<RenderToolCallConfig[]> = this.#toolCallRenderConfigs.asReadonly();\n  readonly clientToolCallRenderConfigs: Signal<FrontendToolConfig[]> = this.#clientToolCallRenderConfigs.asReadonly();\n  readonly humanInTheLoopToolRenderConfigs: Signal<HumanInTheLoopConfig[]> =\n    this.#humanInTheLoopToolRenderConfigs.asReadonly();\n\n  constructor() {\n    this.#config.renderToolCalls?.forEach((renderConfig) => {\n      this.addRenderToolCall(renderConfig);\n    });\n\n    this.#config.tools?.forEach((tool) => {\n      if (tool.renderer && tool.parameters) {\n        this.addRenderToolCall({\n          name: tool.name,\n          args: tool.parameters,\n          component: tool.renderer,\n          agentId: tool.agentId,\n        });\n      }\n    });\n\n    this.#config.frontendTools?.forEach((clientTool) => {\n      this.addFrontendTool({ ...clientTool, injector: this.#rootInjector });\n    });\n\n    this.#config.humanInTheLoop?.forEach((humanInTheLoopTool) => {\n      this.addHumanInTheLoop(humanInTheLoopTool);\n    });\n\n    this.core.subscribe({\n      onAgentsChanged: () => {\n        this.#agents.set(this.core.agents);\n      },\n    });\n  }\n\n  #bindClientTool(\n    clientToolWithInjector: FrontendToolConfig & {\n      injector: Injector;\n    },\n  ): FrontendTool {\n    const { injector, handler, ...frontendCandidate } = clientToolWithInjector;\n\n    return {\n      ...frontendCandidate,\n      handler: (args) => runInInjectionContext(injector, () => handler(args)),\n    };\n  }\n\n  addFrontendTool(\n    clientToolWithInjector: FrontendToolConfig & {\n      injector: Injector;\n    },\n  ): void {\n    const tool = this.#bindClientTool(clientToolWithInjector);\n\n    this.core.addTool(tool);\n\n    this.#clientToolCallRenderConfigs.update((current) => [...current, clientToolWithInjector]);\n  }\n\n  addRenderToolCall(renderConfig: RenderToolCallConfig): void {\n    this.#toolCallRenderConfigs.update((current) => [...current, renderConfig]);\n  }\n\n  #bindHumanInTheLoopTool(humanInTheLoopTool: HumanInTheLoopConfig): FrontendTool {\n    return {\n      ...humanInTheLoopTool,\n      handler: (args, toolCall) => {\n        return this.#hitl.onResult(toolCall.id, humanInTheLoopTool.name);\n      },\n    };\n  }\n\n  addHumanInTheLoop(humanInTheLoopTool: HumanInTheLoopConfig): void {\n    this.#humanInTheLoopToolRenderConfigs.update((current) => [...current, humanInTheLoopTool]);\n\n    const tool = this.#bindHumanInTheLoopTool(humanInTheLoopTool);\n\n    this.core.addTool(tool);\n  }\n\n  #isSameAgentId<T extends { agentId?: string }>(target: T, agentId?: string): boolean {\n    if (agentId) {\n      return target.agentId === agentId;\n    }\n\n    return true;\n  }\n\n  removeTool(toolName: string, agentId?: string): void {\n    this.core.removeTool(toolName);\n    this.#clientToolCallRenderConfigs.update((current) =>\n      current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)),\n    );\n    this.#humanInTheLoopToolRenderConfigs.update((current) =>\n      current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)),\n    );\n    this.#toolCallRenderConfigs.update((current) =>\n      current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)),\n    );\n  }\n\n  getAgent(agentId: string): AbstractAgent | undefined {\n    return this.core.getAgent(agentId);\n  }\n\n  updateRuntime(options: {\n    runtimeUrl?: string;\n    headers?: Record<string, string>;\n    properties?: Record<string, unknown>;\n    agents?: Record<string, AbstractAgent>;\n  }): void {\n    if (options.runtimeUrl !== undefined) {\n      this.core.setRuntimeUrl(options.runtimeUrl);\n    }\n    if (options.headers !== undefined) {\n      this.core.setHeaders(options.headers);\n    }\n    if (options.properties !== undefined) {\n      this.core.setProperties(options.properties);\n    }\n    if (options.agents !== undefined) {\n      this.core.setAgents__unsafe_dev_only(options.agents);\n    }\n  }\n}\n"]}
@@ -0,0 +1,130 @@
1
+ import { Directive, Input, Inject, } from "@angular/core";
2
+ import { CopilotKit } from "../copilotkit";
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "../copilotkit";
5
+ /**
6
+ * Directive to manage agent context in CopilotKit.
7
+ * Automatically adds context on init, updates on changes, and removes on destroy.
8
+ *
9
+ * @example
10
+ * ```html
11
+ * <!-- With separate inputs -->
12
+ * <div copilotkitAgentContext
13
+ * [description]="'User preferences'"
14
+ * [value]="userSettings">
15
+ * </div>
16
+ *
17
+ * <!-- With context object -->
18
+ * <div [copilotkitAgentContext]="contextObject">
19
+ * </div>
20
+ *
21
+ * <!-- With dynamic values -->
22
+ * <div copilotkitAgentContext
23
+ * description="Form state"
24
+ * [value]="formData$ | async">
25
+ * </div>
26
+ * ```
27
+ */
28
+ export class CopilotKitAgentContext {
29
+ copilotkit;
30
+ contextId;
31
+ constructor(copilotkit) {
32
+ this.copilotkit = copilotkit;
33
+ }
34
+ /**
35
+ * Context object containing both description and value.
36
+ * If provided, this takes precedence over individual inputs.
37
+ */
38
+ context;
39
+ /**
40
+ * Description of the context.
41
+ * Used when context object is not provided.
42
+ */
43
+ description;
44
+ /**
45
+ * Value of the context.
46
+ * Used when context object is not provided.
47
+ */
48
+ value;
49
+ ngOnInit() {
50
+ this.addContext();
51
+ }
52
+ ngOnChanges(changes) {
53
+ // Check if any relevant input has changed
54
+ const hasContextChange = "context" in changes;
55
+ const hasDescriptionChange = "description" in changes;
56
+ const hasValueChange = "value" in changes;
57
+ if (hasContextChange || hasDescriptionChange || hasValueChange) {
58
+ // Skip the first change as ngOnInit handles initial setup
59
+ if (this.contextId) {
60
+ this.updateContext();
61
+ }
62
+ }
63
+ }
64
+ ngOnDestroy() {
65
+ this.removeContext();
66
+ }
67
+ /**
68
+ * Adds the context to CopilotKit
69
+ */
70
+ addContext() {
71
+ const contextToAdd = this.getContext();
72
+ if (contextToAdd) {
73
+ this.contextId = this.copilotkit.core.addContext(contextToAdd);
74
+ }
75
+ }
76
+ /**
77
+ * Updates the context by removing the old one and adding a new one
78
+ */
79
+ updateContext() {
80
+ this.removeContext();
81
+ this.addContext();
82
+ }
83
+ /**
84
+ * Removes the current context from CopilotKit
85
+ */
86
+ removeContext() {
87
+ if (this.contextId) {
88
+ this.copilotkit.core.removeContext(this.contextId);
89
+ this.contextId = undefined;
90
+ }
91
+ }
92
+ /**
93
+ * Gets the context object from inputs
94
+ */
95
+ getContext() {
96
+ // If context object is provided, use it
97
+ if (this.context) {
98
+ return this.context;
99
+ }
100
+ // Otherwise, build from individual inputs
101
+ // Note: null is a valid value, but undefined means not set
102
+ if (this.description !== undefined && this.value !== undefined) {
103
+ return {
104
+ description: this.description,
105
+ value: this.value,
106
+ };
107
+ }
108
+ return null;
109
+ }
110
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKitAgentContext, deps: [{ token: CopilotKit }], target: i0.ɵɵFactoryTarget.Directive });
111
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.14", type: CopilotKitAgentContext, isStandalone: true, selector: "[copilotkitAgentContext]", inputs: { context: ["copilotkitAgentContext", "context"], description: "description", value: "value" }, usesOnChanges: true, ngImport: i0 });
112
+ }
113
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKitAgentContext, decorators: [{
114
+ type: Directive,
115
+ args: [{
116
+ selector: "[copilotkitAgentContext]",
117
+ standalone: true,
118
+ }]
119
+ }], ctorParameters: () => [{ type: i1.CopilotKit, decorators: [{
120
+ type: Inject,
121
+ args: [CopilotKit]
122
+ }] }], propDecorators: { context: [{
123
+ type: Input,
124
+ args: ["copilotkitAgentContext"]
125
+ }], description: [{
126
+ type: Input
127
+ }], value: [{
128
+ type: Input
129
+ }] } });
130
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdGtpdC1hZ2VudC1jb250ZXh0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2xpYi9kaXJlY3RpdmVzL2NvcGlsb3RraXQtYWdlbnQtY29udGV4dC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsU0FBUyxFQUNULEtBQUssRUFLTCxNQUFNLEdBQ1AsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBRzNDOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JHO0FBS0gsTUFBTSxPQUFPLHNCQUFzQjtJQUdnQjtJQUZ6QyxTQUFTLENBQVU7SUFFM0IsWUFBaUQsVUFBc0I7UUFBdEIsZUFBVSxHQUFWLFVBQVUsQ0FBWTtJQUFHLENBQUM7SUFFM0U7OztPQUdHO0lBQzhCLE9BQU8sQ0FBVztJQUVuRDs7O09BR0c7SUFDTSxXQUFXLENBQVU7SUFFOUI7OztPQUdHO0lBQ00sS0FBSyxDQUFPO0lBRXJCLFFBQVE7UUFDTixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDcEIsQ0FBQztJQUVELFdBQVcsQ0FBQyxPQUFzQjtRQUNoQywwQ0FBMEM7UUFDMUMsTUFBTSxnQkFBZ0IsR0FBRyxTQUFTLElBQUksT0FBTyxDQUFDO1FBQzlDLE1BQU0sb0JBQW9CLEdBQUcsYUFBYSxJQUFJLE9BQU8sQ0FBQztRQUN0RCxNQUFNLGNBQWMsR0FBRyxPQUFPLElBQUksT0FBTyxDQUFDO1FBRTFDLElBQUksZ0JBQWdCLElBQUksb0JBQW9CLElBQUksY0FBYyxFQUFFLENBQUM7WUFDL0QsMERBQTBEO1lBQzFELElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDdkIsQ0FBQztRQUNILENBQUM7SUFDSCxDQUFDO0lBRUQsV0FBVztRQUNULElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztJQUN2QixDQUFDO0lBRUQ7O09BRUc7SUFDSyxVQUFVO1FBQ2hCLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztRQUV2QyxJQUFJLFlBQVksRUFBRSxDQUFDO1lBQ2pCLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQ2pFLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxhQUFhO1FBQ25CLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztRQUNyQixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDcEIsQ0FBQztJQUVEOztPQUVHO0lBQ0ssYUFBYTtRQUNuQixJQUFJLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztZQUNuQixJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ25ELElBQUksQ0FBQyxTQUFTLEdBQUcsU0FBUyxDQUFDO1FBQzdCLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxVQUFVO1FBQ2hCLHdDQUF3QztRQUN4QyxJQUFJLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNqQixPQUFPLElBQUksQ0FBQyxPQUFPLENBQUM7UUFDdEIsQ0FBQztRQUVELDBDQUEwQztRQUMxQywyREFBMkQ7UUFDM0QsSUFBSSxJQUFJLENBQUMsV0FBVyxLQUFLLFNBQVMsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLFNBQVMsRUFBRSxDQUFDO1lBQy9ELE9BQU87Z0JBQ0wsV0FBVyxFQUFFLElBQUksQ0FBQyxXQUFXO2dCQUM3QixLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUs7YUFDbEIsQ0FBQztRQUNKLENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7d0dBN0ZVLHNCQUFzQixrQkFHYixVQUFVOzRGQUhuQixzQkFBc0I7OzRGQUF0QixzQkFBc0I7a0JBSmxDLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLDBCQUEwQjtvQkFDcEMsVUFBVSxFQUFFLElBQUk7aUJBQ2pCOzswQkFJYyxNQUFNOzJCQUFDLFVBQVU7eUNBTUcsT0FBTztzQkFBdkMsS0FBSzt1QkFBQyx3QkFBd0I7Z0JBTXRCLFdBQVc7c0JBQW5CLEtBQUs7Z0JBTUcsS0FBSztzQkFBYixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgRGlyZWN0aXZlLFxuICBJbnB1dCxcbiAgT25Jbml0LFxuICBPbkNoYW5nZXMsXG4gIE9uRGVzdHJveSxcbiAgU2ltcGxlQ2hhbmdlcyxcbiAgSW5qZWN0LFxufSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xuaW1wb3J0IHsgQ29waWxvdEtpdCB9IGZyb20gXCIuLi9jb3BpbG90a2l0XCI7XG5pbXBvcnQgdHlwZSB7IENvbnRleHQgfSBmcm9tIFwiQGFnLXVpL2NsaWVudFwiO1xuXG4vKipcbiAqIERpcmVjdGl2ZSB0byBtYW5hZ2UgYWdlbnQgY29udGV4dCBpbiBDb3BpbG90S2l0LlxuICogQXV0b21hdGljYWxseSBhZGRzIGNvbnRleHQgb24gaW5pdCwgdXBkYXRlcyBvbiBjaGFuZ2VzLCBhbmQgcmVtb3ZlcyBvbiBkZXN0cm95LlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGBodG1sXG4gKiA8IS0tIFdpdGggc2VwYXJhdGUgaW5wdXRzIC0tPlxuICogPGRpdiBjb3BpbG90a2l0QWdlbnRDb250ZXh0XG4gKiAgICAgIFtkZXNjcmlwdGlvbl09XCInVXNlciBwcmVmZXJlbmNlcydcIlxuICogICAgICBbdmFsdWVdPVwidXNlclNldHRpbmdzXCI+XG4gKiA8L2Rpdj5cbiAqXG4gKiA8IS0tIFdpdGggY29udGV4dCBvYmplY3QgLS0+XG4gKiA8ZGl2IFtjb3BpbG90a2l0QWdlbnRDb250ZXh0XT1cImNvbnRleHRPYmplY3RcIj5cbiAqIDwvZGl2PlxuICpcbiAqIDwhLS0gV2l0aCBkeW5hbWljIHZhbHVlcyAtLT5cbiAqIDxkaXYgY29waWxvdGtpdEFnZW50Q29udGV4dFxuICogICAgICBkZXNjcmlwdGlvbj1cIkZvcm0gc3RhdGVcIlxuICogICAgICBbdmFsdWVdPVwiZm9ybURhdGEkIHwgYXN5bmNcIj5cbiAqIDwvZGl2PlxuICogYGBgXG4gKi9cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogXCJbY29waWxvdGtpdEFnZW50Q29udGV4dF1cIixcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbn0pXG5leHBvcnQgY2xhc3MgQ29waWxvdEtpdEFnZW50Q29udGV4dCBpbXBsZW1lbnRzIE9uSW5pdCwgT25DaGFuZ2VzLCBPbkRlc3Ryb3kge1xuICBwcml2YXRlIGNvbnRleHRJZD86IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcihASW5qZWN0KENvcGlsb3RLaXQpIHByaXZhdGUgcmVhZG9ubHkgY29waWxvdGtpdDogQ29waWxvdEtpdCkge31cblxuICAvKipcbiAgICogQ29udGV4dCBvYmplY3QgY29udGFpbmluZyBib3RoIGRlc2NyaXB0aW9uIGFuZCB2YWx1ZS5cbiAgICogSWYgcHJvdmlkZWQsIHRoaXMgdGFrZXMgcHJlY2VkZW5jZSBvdmVyIGluZGl2aWR1YWwgaW5wdXRzLlxuICAgKi9cbiAgQElucHV0KFwiY29waWxvdGtpdEFnZW50Q29udGV4dFwiKSBjb250ZXh0PzogQ29udGV4dDtcblxuICAvKipcbiAgICogRGVzY3JpcHRpb24gb2YgdGhlIGNvbnRleHQuXG4gICAqIFVzZWQgd2hlbiBjb250ZXh0IG9iamVjdCBpcyBub3QgcHJvdmlkZWQuXG4gICAqL1xuICBASW5wdXQoKSBkZXNjcmlwdGlvbj86IHN0cmluZztcblxuICAvKipcbiAgICogVmFsdWUgb2YgdGhlIGNvbnRleHQuXG4gICAqIFVzZWQgd2hlbiBjb250ZXh0IG9iamVjdCBpcyBub3QgcHJvdmlkZWQuXG4gICAqL1xuICBASW5wdXQoKSB2YWx1ZT86IGFueTtcblxuICBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLmFkZENvbnRleHQoKTtcbiAgfVxuXG4gIG5nT25DaGFuZ2VzKGNoYW5nZXM6IFNpbXBsZUNoYW5nZXMpOiB2b2lkIHtcbiAgICAvLyBDaGVjayBpZiBhbnkgcmVsZXZhbnQgaW5wdXQgaGFzIGNoYW5nZWRcbiAgICBjb25zdCBoYXNDb250ZXh0Q2hhbmdlID0gXCJjb250ZXh0XCIgaW4gY2hhbmdlcztcbiAgICBjb25zdCBoYXNEZXNjcmlwdGlvbkNoYW5nZSA9IFwiZGVzY3JpcHRpb25cIiBpbiBjaGFuZ2VzO1xuICAgIGNvbnN0IGhhc1ZhbHVlQ2hhbmdlID0gXCJ2YWx1ZVwiIGluIGNoYW5nZXM7XG5cbiAgICBpZiAoaGFzQ29udGV4dENoYW5nZSB8fCBoYXNEZXNjcmlwdGlvbkNoYW5nZSB8fCBoYXNWYWx1ZUNoYW5nZSkge1xuICAgICAgLy8gU2tpcCB0aGUgZmlyc3QgY2hhbmdlIGFzIG5nT25Jbml0IGhhbmRsZXMgaW5pdGlhbCBzZXR1cFxuICAgICAgaWYgKHRoaXMuY29udGV4dElkKSB7XG4gICAgICAgIHRoaXMudXBkYXRlQ29udGV4dCgpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMucmVtb3ZlQ29udGV4dCgpO1xuICB9XG5cbiAgLyoqXG4gICAqIEFkZHMgdGhlIGNvbnRleHQgdG8gQ29waWxvdEtpdFxuICAgKi9cbiAgcHJpdmF0ZSBhZGRDb250ZXh0KCk6IHZvaWQge1xuICAgIGNvbnN0IGNvbnRleHRUb0FkZCA9IHRoaXMuZ2V0Q29udGV4dCgpO1xuXG4gICAgaWYgKGNvbnRleHRUb0FkZCkge1xuICAgICAgdGhpcy5jb250ZXh0SWQgPSB0aGlzLmNvcGlsb3RraXQuY29yZS5hZGRDb250ZXh0KGNvbnRleHRUb0FkZCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFVwZGF0ZXMgdGhlIGNvbnRleHQgYnkgcmVtb3ZpbmcgdGhlIG9sZCBvbmUgYW5kIGFkZGluZyBhIG5ldyBvbmVcbiAgICovXG4gIHByaXZhdGUgdXBkYXRlQ29udGV4dCgpOiB2b2lkIHtcbiAgICB0aGlzLnJlbW92ZUNvbnRleHQoKTtcbiAgICB0aGlzLmFkZENvbnRleHQoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZW1vdmVzIHRoZSBjdXJyZW50IGNvbnRleHQgZnJvbSBDb3BpbG90S2l0XG4gICAqL1xuICBwcml2YXRlIHJlbW92ZUNvbnRleHQoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuY29udGV4dElkKSB7XG4gICAgICB0aGlzLmNvcGlsb3RraXQuY29yZS5yZW1vdmVDb250ZXh0KHRoaXMuY29udGV4dElkKTtcbiAgICAgIHRoaXMuY29udGV4dElkID0gdW5kZWZpbmVkO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBHZXRzIHRoZSBjb250ZXh0IG9iamVjdCBmcm9tIGlucHV0c1xuICAgKi9cbiAgcHJpdmF0ZSBnZXRDb250ZXh0KCk6IENvbnRleHQgfCBudWxsIHtcbiAgICAvLyBJZiBjb250ZXh0IG9iamVjdCBpcyBwcm92aWRlZCwgdXNlIGl0XG4gICAgaWYgKHRoaXMuY29udGV4dCkge1xuICAgICAgcmV0dXJuIHRoaXMuY29udGV4dDtcbiAgICB9XG5cbiAgICAvLyBPdGhlcndpc2UsIGJ1aWxkIGZyb20gaW5kaXZpZHVhbCBpbnB1dHNcbiAgICAvLyBOb3RlOiBudWxsIGlzIGEgdmFsaWQgdmFsdWUsIGJ1dCB1bmRlZmluZWQgbWVhbnMgbm90IHNldFxuICAgIGlmICh0aGlzLmRlc2NyaXB0aW9uICE9PSB1bmRlZmluZWQgJiYgdGhpcy52YWx1ZSAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBkZXNjcmlwdGlvbjogdGhpcy5kZXNjcmlwdGlvbixcbiAgICAgICAgdmFsdWU6IHRoaXMudmFsdWUsXG4gICAgICB9O1xuICAgIH1cblxuICAgIHJldHVybiBudWxsO1xuICB9XG59XG4iXX0=