patchwork-os 0.2.0-alpha.0 → 0.2.0-alpha.10

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 (132) hide show
  1. package/README.md +41 -46
  2. package/dist/bridge.js +23 -10
  3. package/dist/bridge.js.map +1 -1
  4. package/dist/claudeDriver.d.ts +3 -1
  5. package/dist/claudeDriver.js +48 -0
  6. package/dist/claudeDriver.js.map +1 -1
  7. package/dist/commands/dashboard.d.ts +47 -0
  8. package/dist/commands/dashboard.js +319 -0
  9. package/dist/commands/dashboard.js.map +1 -0
  10. package/dist/config.d.ts +2 -2
  11. package/dist/config.js +5 -2
  12. package/dist/config.js.map +1 -1
  13. package/dist/connectors/github.d.ts +94 -0
  14. package/dist/connectors/github.js +350 -0
  15. package/dist/connectors/github.js.map +1 -0
  16. package/dist/connectors/gmail.d.ts +40 -0
  17. package/dist/connectors/gmail.js +304 -0
  18. package/dist/connectors/gmail.js.map +1 -0
  19. package/dist/connectors/googleCalendar.d.ts +57 -0
  20. package/dist/connectors/googleCalendar.js +308 -0
  21. package/dist/connectors/googleCalendar.js.map +1 -0
  22. package/dist/connectors/linear.d.ts +117 -0
  23. package/dist/connectors/linear.js +248 -0
  24. package/dist/connectors/linear.js.map +1 -0
  25. package/dist/connectors/mcpClient.d.ts +56 -0
  26. package/dist/connectors/mcpClient.js +189 -0
  27. package/dist/connectors/mcpClient.js.map +1 -0
  28. package/dist/connectors/mcpOAuth.d.ts +83 -0
  29. package/dist/connectors/mcpOAuth.js +363 -0
  30. package/dist/connectors/mcpOAuth.js.map +1 -0
  31. package/dist/connectors/sentry.d.ts +43 -0
  32. package/dist/connectors/sentry.js +197 -0
  33. package/dist/connectors/sentry.js.map +1 -0
  34. package/dist/connectors/slack.d.ts +50 -0
  35. package/dist/connectors/slack.js +254 -0
  36. package/dist/connectors/slack.js.map +1 -0
  37. package/dist/drivers/claude/api.d.ts +11 -0
  38. package/dist/drivers/claude/api.js +54 -0
  39. package/dist/drivers/claude/api.js.map +1 -0
  40. package/dist/drivers/claude/envSanitizer.d.ts +7 -0
  41. package/dist/drivers/claude/envSanitizer.js +18 -0
  42. package/dist/drivers/claude/envSanitizer.js.map +1 -0
  43. package/dist/drivers/claude/streamParser.d.ts +38 -0
  44. package/dist/drivers/claude/streamParser.js +34 -0
  45. package/dist/drivers/claude/streamParser.js.map +1 -0
  46. package/dist/drivers/claude/subprocess.d.ts +19 -0
  47. package/dist/drivers/claude/subprocess.js +216 -0
  48. package/dist/drivers/claude/subprocess.js.map +1 -0
  49. package/dist/drivers/claude/subprocessSettings.d.ts +9 -0
  50. package/dist/drivers/claude/subprocessSettings.js +55 -0
  51. package/dist/drivers/claude/subprocessSettings.js.map +1 -0
  52. package/dist/drivers/gemini/index.d.ts +14 -0
  53. package/dist/drivers/gemini/index.js +176 -0
  54. package/dist/drivers/gemini/index.js.map +1 -0
  55. package/dist/drivers/grok/index.d.ts +11 -0
  56. package/dist/drivers/grok/index.js +22 -0
  57. package/dist/drivers/grok/index.js.map +1 -0
  58. package/dist/drivers/index.d.ts +18 -0
  59. package/dist/drivers/index.js +31 -0
  60. package/dist/drivers/index.js.map +1 -0
  61. package/dist/drivers/openai/index.d.ts +24 -0
  62. package/dist/drivers/openai/index.js +110 -0
  63. package/dist/drivers/openai/index.js.map +1 -0
  64. package/dist/drivers/types.d.ts +72 -0
  65. package/dist/drivers/types.js +30 -0
  66. package/dist/drivers/types.js.map +1 -0
  67. package/dist/index.js +116 -22
  68. package/dist/index.js.map +1 -1
  69. package/dist/recipes/yamlRunner.d.ts +95 -0
  70. package/dist/recipes/yamlRunner.js +588 -0
  71. package/dist/recipes/yamlRunner.js.map +1 -0
  72. package/dist/recipesHttp.d.ts +13 -1
  73. package/dist/recipesHttp.js +9 -1
  74. package/dist/recipesHttp.js.map +1 -1
  75. package/dist/server.d.ts +3 -1
  76. package/dist/server.js +490 -2
  77. package/dist/server.js.map +1 -1
  78. package/dist/tools/addLinearComment.d.ts +55 -0
  79. package/dist/tools/addLinearComment.js +70 -0
  80. package/dist/tools/addLinearComment.js.map +1 -0
  81. package/dist/tools/createLinearIssue.d.ts +84 -0
  82. package/dist/tools/createLinearIssue.js +146 -0
  83. package/dist/tools/createLinearIssue.js.map +1 -0
  84. package/dist/tools/ctxGetTaskContext.d.ts +4 -1
  85. package/dist/tools/ctxGetTaskContext.js +45 -2
  86. package/dist/tools/ctxGetTaskContext.js.map +1 -1
  87. package/dist/tools/fetchCalendarEvents.d.ts +94 -0
  88. package/dist/tools/fetchCalendarEvents.js +97 -0
  89. package/dist/tools/fetchCalendarEvents.js.map +1 -0
  90. package/dist/tools/fetchGithubIssue.d.ts +80 -0
  91. package/dist/tools/fetchGithubIssue.js +84 -0
  92. package/dist/tools/fetchGithubIssue.js.map +1 -0
  93. package/dist/tools/fetchGithubPR.d.ts +89 -0
  94. package/dist/tools/fetchGithubPR.js +96 -0
  95. package/dist/tools/fetchGithubPR.js.map +1 -0
  96. package/dist/tools/fetchLinearIssue.d.ts +112 -0
  97. package/dist/tools/fetchLinearIssue.js +129 -0
  98. package/dist/tools/fetchLinearIssue.js.map +1 -0
  99. package/dist/tools/fetchSentryIssue.d.ts +143 -0
  100. package/dist/tools/fetchSentryIssue.js +150 -0
  101. package/dist/tools/fetchSentryIssue.js.map +1 -0
  102. package/dist/tools/fetchSlackProfile.d.ts +43 -0
  103. package/dist/tools/fetchSlackProfile.js +43 -0
  104. package/dist/tools/fetchSlackProfile.js.map +1 -0
  105. package/dist/tools/getConnectorStatus.d.ts +58 -0
  106. package/dist/tools/getConnectorStatus.js +56 -0
  107. package/dist/tools/getConnectorStatus.js.map +1 -0
  108. package/dist/tools/github/index.d.ts +1 -1
  109. package/dist/tools/github/index.js +1 -1
  110. package/dist/tools/github/index.js.map +1 -1
  111. package/dist/tools/github/pr.d.ts +122 -0
  112. package/dist/tools/github/pr.js +152 -0
  113. package/dist/tools/github/pr.js.map +1 -1
  114. package/dist/tools/index.js +27 -1
  115. package/dist/tools/index.js.map +1 -1
  116. package/dist/tools/slackListChannels.d.ts +65 -0
  117. package/dist/tools/slackListChannels.js +70 -0
  118. package/dist/tools/slackListChannels.js.map +1 -0
  119. package/dist/tools/slackPostMessage.d.ts +57 -0
  120. package/dist/tools/slackPostMessage.js +72 -0
  121. package/dist/tools/slackPostMessage.js.map +1 -0
  122. package/dist/tools/updateLinearIssue.d.ts +89 -0
  123. package/dist/tools/updateLinearIssue.js +103 -0
  124. package/dist/tools/updateLinearIssue.js.map +1 -0
  125. package/package.json +1 -1
  126. package/scripts/start-all.sh +56 -19
  127. package/templates/recipes/ctx-loop-test.yaml +75 -0
  128. package/templates/recipes/gmail-health-check.yaml +19 -0
  129. package/templates/recipes/inbox-triage.yaml +15 -0
  130. package/templates/recipes/morning-brief.yaml +72 -0
  131. package/templates/recipes/sentry-to-linear.yaml +77 -0
  132. package/templates/scheduled-tasks/morning-brief/SKILL.md +37 -0
@@ -0,0 +1,350 @@
1
+ /**
2
+ * GitHub connector — routes through the official GitHub MCP server.
3
+ *
4
+ * Endpoint: https://api.githubcopilot.com/mcp/
5
+ * Auth: OAuth 2.0 via github.com/login/oauth (requires pre-registered
6
+ * Patchwork OS OAuth App; client_id via PATCHWORK_GITHUB_CLIENT_ID env).
7
+ *
8
+ * HTTP routes (wired in src/server.ts):
9
+ * GET /connections/github/authorize — returns { url } for popup
10
+ * GET /connections/github/callback — token exchange
11
+ * POST /connections/github/test — ping MCP server
12
+ * DELETE /connections/github — delete token
13
+ *
14
+ * Exports preserved for yamlRunner; listIssues/listPRs are now async.
15
+ */
16
+ import { McpClient } from "./mcpClient.js";
17
+ import { completeAuthorize, getAccessToken, isConnected, loadTokenFile, revoke, startAuthorize, vendorConfig, } from "./mcpOAuth.js";
18
+ const GITHUB_MCP_ENDPOINT = "https://api.githubcopilot.com/mcp/";
19
+ // ── MCP client (memoized) ────────────────────────────────────────────────────
20
+ let _client = null;
21
+ function client() {
22
+ if (!_client) {
23
+ _client = new McpClient(GITHUB_MCP_ENDPOINT, () => getAccessToken("github"));
24
+ }
25
+ return _client;
26
+ }
27
+ // ── Status ───────────────────────────────────────────────────────────────────
28
+ export function getStatus() {
29
+ if (!isConnected("github"))
30
+ return { connected: false };
31
+ const file = loadTokenFile("github");
32
+ return { connected: true, user: file?.profile?.login };
33
+ }
34
+ function parseRepo(opts) {
35
+ if (!opts.repo)
36
+ return {};
37
+ const [owner, repo] = opts.repo.split("/");
38
+ return { owner, repo };
39
+ }
40
+ function coerceIssue(raw, fallbackRepo) {
41
+ return {
42
+ number: raw.number,
43
+ title: raw.title,
44
+ repo: raw.repository?.full_name ??
45
+ raw.repository?.nameWithOwner ??
46
+ fallbackRepo,
47
+ url: raw.html_url ?? raw.url ?? "",
48
+ labels: (raw.labels ?? [])
49
+ .map((l) => (typeof l === "string" ? l : (l.name ?? "")))
50
+ .filter(Boolean),
51
+ updatedAt: raw.updated_at ?? raw.updatedAt ?? "",
52
+ };
53
+ }
54
+ // ── Listing ──────────────────────────────────────────────────────────────────
55
+ export async function listIssues(opts = {}) {
56
+ if (!isConnected("github"))
57
+ return [];
58
+ const { owner, repo } = parseRepo(opts);
59
+ const args = {
60
+ state: "open",
61
+ perPage: Math.min(opts.limit ?? 20, 50),
62
+ };
63
+ if (owner)
64
+ args.owner = owner;
65
+ if (repo)
66
+ args.repo = repo;
67
+ if (opts.assignee)
68
+ args.assignee = opts.assignee === "@me" ? "@me" : opts.assignee;
69
+ if (opts.mention)
70
+ args.mentioned = opts.mention;
71
+ try {
72
+ const res = await client().callTool("list_issues", args, {
73
+ cacheKey: `gh:issues:${JSON.stringify(args)}`,
74
+ cacheTtlMs: 60_000,
75
+ });
76
+ const parsed = McpClient.extractJson(res);
77
+ const arr = Array.isArray(parsed) ? parsed : (parsed.items ?? []);
78
+ const fallbackRepo = opts.repo ?? "";
79
+ return arr.map((i) => coerceIssue(i, fallbackRepo));
80
+ }
81
+ catch {
82
+ return [];
83
+ }
84
+ }
85
+ export async function listPRs(opts = {}) {
86
+ if (!isConnected("github"))
87
+ return [];
88
+ const { owner, repo } = parseRepo(opts);
89
+ const args = {
90
+ state: "open",
91
+ perPage: Math.min(opts.limit ?? 20, 50),
92
+ };
93
+ if (owner)
94
+ args.owner = owner;
95
+ if (repo)
96
+ args.repo = repo;
97
+ if (opts.author)
98
+ args.author = opts.author === "@me" ? "@me" : opts.author;
99
+ if (opts.reviewRequested)
100
+ args.reviewRequested = opts.reviewRequested;
101
+ try {
102
+ const res = await client().callTool("list_pull_requests", args, {
103
+ cacheKey: `gh:prs:${JSON.stringify(args)}`,
104
+ cacheTtlMs: 60_000,
105
+ });
106
+ const parsed = McpClient.extractJson(res);
107
+ const arr = Array.isArray(parsed) ? parsed : (parsed.items ?? []);
108
+ const fallbackRepo = opts.repo ?? "";
109
+ return arr.map((p) => ({
110
+ ...coerceIssue(p, fallbackRepo),
111
+ isDraft: Boolean(p.draft ?? p.isDraft ?? false),
112
+ reviewDecision: p.review_decision ?? p.reviewDecision ?? "",
113
+ }));
114
+ }
115
+ catch {
116
+ return [];
117
+ }
118
+ }
119
+ function parseIssueRef(ref) {
120
+ // https://github.com/owner/repo/issues/42
121
+ const urlMatch = ref.match(/github\.com\/([^/]+)\/([^/]+)\/issues\/(\d+)/);
122
+ if (urlMatch) {
123
+ return {
124
+ owner: urlMatch[1],
125
+ repo: urlMatch[2],
126
+ number: Number(urlMatch[3]),
127
+ };
128
+ }
129
+ // owner/repo#42 or owner/repo/42
130
+ const shortMatch = ref.match(/^([^/]+)\/([^/#]+)[/#](\d+)$/);
131
+ if (shortMatch) {
132
+ return {
133
+ owner: shortMatch[1],
134
+ repo: shortMatch[2],
135
+ number: Number(shortMatch[3]),
136
+ };
137
+ }
138
+ throw new Error(`Cannot parse GitHub issue ref: ${ref}`);
139
+ }
140
+ function parsePRRef(ref) {
141
+ const urlMatch = ref.match(/github\.com\/([^/]+)\/([^/]+)\/pull\/(\d+)/);
142
+ if (urlMatch) {
143
+ return {
144
+ owner: urlMatch[1],
145
+ repo: urlMatch[2],
146
+ number: Number(urlMatch[3]),
147
+ };
148
+ }
149
+ const shortMatch = ref.match(/^([^/]+)\/([^/#]+)[/#](\d+)$/);
150
+ if (shortMatch) {
151
+ return {
152
+ owner: shortMatch[1],
153
+ repo: shortMatch[2],
154
+ number: Number(shortMatch[3]),
155
+ };
156
+ }
157
+ throw new Error(`Cannot parse GitHub PR ref: ${ref}`);
158
+ }
159
+ export async function fetchGitHubIssue(ref, signal) {
160
+ if (!isConnected("github"))
161
+ throw new Error("GitHub not connected. GET /connections/github/auth first.");
162
+ const { owner, repo, number } = parseIssueRef(ref);
163
+ const res = await client().callTool("issue_read", { owner, repo, issue_number: number, method: "get" }, { signal });
164
+ const raw = McpClient.extractJson(res);
165
+ return {
166
+ number: raw.number ?? number,
167
+ title: raw.title ?? "",
168
+ body: raw.body ?? "",
169
+ state: raw.state ?? "",
170
+ url: raw.html_url ?? raw.url ?? "",
171
+ repo: `${owner}/${repo}`,
172
+ author: raw.user?.login ??
173
+ raw.author ??
174
+ "",
175
+ labels: (raw.labels ?? []).map((l) => l.name ?? String(l)),
176
+ assignees: (raw.assignees ?? []).map((a) => a.login ?? String(a)),
177
+ createdAt: raw.created_at ?? "",
178
+ updatedAt: raw.updated_at ?? "",
179
+ comments: raw.comments ?? 0,
180
+ };
181
+ }
182
+ export async function fetchGitHubPR(ref, signal) {
183
+ if (!isConnected("github"))
184
+ throw new Error("GitHub not connected. GET /connections/github/auth first.");
185
+ const { owner, repo, number } = parsePRRef(ref);
186
+ const res = await client().callTool("pull_request_read", { owner, repo, pullNumber: number, method: "get" }, { signal });
187
+ const raw = McpClient.extractJson(res);
188
+ return {
189
+ number: raw.number ?? number,
190
+ title: raw.title ?? "",
191
+ body: raw.body ?? "",
192
+ state: raw.state ?? "",
193
+ url: raw.html_url ?? raw.url ?? "",
194
+ repo: `${owner}/${repo}`,
195
+ author: raw.user?.login ??
196
+ raw.author ??
197
+ "",
198
+ isDraft: Boolean(raw.draft ?? raw.isDraft ?? false),
199
+ reviewDecision: raw.review_decision ?? raw.reviewDecision ?? "",
200
+ labels: (raw.labels ?? []).map((l) => l.name ?? String(l)),
201
+ headBranch: raw.head?.ref ??
202
+ raw.headRefName ??
203
+ "",
204
+ baseBranch: raw.base?.ref ??
205
+ raw.baseRefName ??
206
+ "",
207
+ createdAt: raw.created_at ?? "",
208
+ updatedAt: raw.updated_at ?? "",
209
+ additions: raw.additions ?? 0,
210
+ deletions: raw.deletions ?? 0,
211
+ };
212
+ }
213
+ // ── HTTP handlers ────────────────────────────────────────────────────────────
214
+ export async function handleGithubAuthorize() {
215
+ const config = vendorConfig("github");
216
+ if (!config.preregisteredClientId) {
217
+ return {
218
+ status: 400,
219
+ contentType: "application/json",
220
+ body: JSON.stringify({
221
+ ok: false,
222
+ error: "PATCHWORK_GITHUB_CLIENT_ID not set — register a GitHub OAuth App first",
223
+ }),
224
+ };
225
+ }
226
+ try {
227
+ const { url } = await startAuthorize(config);
228
+ return { status: 302, body: "", redirect: url };
229
+ }
230
+ catch (err) {
231
+ return {
232
+ status: 400,
233
+ contentType: "application/json",
234
+ body: JSON.stringify({
235
+ ok: false,
236
+ error: err instanceof Error ? err.message : String(err),
237
+ }),
238
+ };
239
+ }
240
+ }
241
+ export async function handleGithubCallback(code, state, error) {
242
+ if (error) {
243
+ return {
244
+ status: 400,
245
+ contentType: "text/html",
246
+ body: `<html><body><h2>GitHub connect failed</h2><pre>${error}</pre></body></html>`,
247
+ };
248
+ }
249
+ if (!code || !state) {
250
+ return {
251
+ status: 400,
252
+ contentType: "text/html",
253
+ body: `<html><body><h2>GitHub connect failed</h2><pre>missing code or state</pre></body></html>`,
254
+ };
255
+ }
256
+ const config = vendorConfig("github");
257
+ try {
258
+ await completeAuthorize(config, code, state);
259
+ // Capture user login for status display
260
+ let login = "";
261
+ try {
262
+ const token = await getAccessToken("github");
263
+ const res = await fetch("https://api.github.com/user", {
264
+ headers: {
265
+ Authorization: `Bearer ${token}`,
266
+ Accept: "application/json",
267
+ },
268
+ });
269
+ if (res.ok) {
270
+ const u = (await res.json());
271
+ login = u.login ?? "";
272
+ if (login) {
273
+ const file = loadTokenFile("github");
274
+ if (file) {
275
+ const { writeFileSync, mkdirSync } = await import("node:fs");
276
+ const { homedir } = await import("node:os");
277
+ const path = await import("node:path");
278
+ const p = path.join(homedir(), ".patchwork", "tokens", "github-mcp.json");
279
+ mkdirSync(path.dirname(p), { recursive: true, mode: 0o700 });
280
+ file.profile = { ...(file.profile ?? {}), login };
281
+ writeFileSync(p, JSON.stringify(file, null, 2), { mode: 0o600 });
282
+ }
283
+ }
284
+ }
285
+ }
286
+ catch {
287
+ // Profile fetch is best-effort
288
+ }
289
+ return {
290
+ status: 200,
291
+ contentType: "text/html",
292
+ body: `<html><body><h2>GitHub connected${login ? ` as ${login}` : ""}</h2><script>window.close();</script></body></html>`,
293
+ };
294
+ }
295
+ catch (err) {
296
+ return {
297
+ status: 400,
298
+ contentType: "text/html",
299
+ body: `<html><body><h2>GitHub connect failed</h2><pre>${err instanceof Error ? err.message : String(err)}</pre></body></html>`,
300
+ };
301
+ }
302
+ }
303
+ export async function handleGithubTest() {
304
+ const s = getStatus();
305
+ if (!s.connected) {
306
+ return {
307
+ status: 200,
308
+ contentType: "application/json",
309
+ body: JSON.stringify({ ok: false, message: "Not connected" }),
310
+ };
311
+ }
312
+ try {
313
+ const ok = await client().ping({ timeoutMs: 10_000 });
314
+ if (ok) {
315
+ return {
316
+ status: 200,
317
+ contentType: "application/json",
318
+ body: JSON.stringify({
319
+ ok: true,
320
+ message: `Connected as ${s.user ?? "unknown"}`,
321
+ }),
322
+ };
323
+ }
324
+ return {
325
+ status: 200,
326
+ contentType: "application/json",
327
+ body: JSON.stringify({ ok: false, message: "MCP ping failed" }),
328
+ };
329
+ }
330
+ catch (err) {
331
+ return {
332
+ status: 200,
333
+ contentType: "application/json",
334
+ body: JSON.stringify({
335
+ ok: false,
336
+ message: err instanceof Error ? err.message : String(err),
337
+ }),
338
+ };
339
+ }
340
+ }
341
+ export async function handleGithubDisconnect() {
342
+ await revoke("github");
343
+ _client = null;
344
+ return {
345
+ status: 200,
346
+ contentType: "application/json",
347
+ body: JSON.stringify({ ok: true }),
348
+ };
349
+ }
350
+ //# sourceMappingURL=github.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"github.js","sourceRoot":"","sources":["../../src/connectors/github.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,40 @@
1
+ /**
2
+ * Gmail OAuth 2.0 connector.
3
+ *
4
+ * Handles:
5
+ * GET /connections/gmail/auth — redirect to Google consent screen
6
+ * GET /connections/gmail/callback — exchange code for tokens, store locally
7
+ * POST /connections/gmail/test — verify stored token works
8
+ * DELETE /connections/gmail — revoke + delete stored token
9
+ * GET /connections — list connector statuses
10
+ *
11
+ * Tokens stored at ~/.patchwork/tokens/gmail.json (mode 0600, never leaves machine).
12
+ * Client credentials read from env: GMAIL_CLIENT_ID, GMAIL_CLIENT_SECRET.
13
+ */
14
+ export interface GmailTokens {
15
+ access_token: string;
16
+ refresh_token?: string;
17
+ expiry_date?: number;
18
+ token_type?: string;
19
+ scope?: string;
20
+ }
21
+ export interface ConnectorStatus {
22
+ id: string;
23
+ status: "connected" | "disconnected";
24
+ lastSync?: string;
25
+ email?: string;
26
+ }
27
+ export declare function loadTokens(): GmailTokens | null;
28
+ /** Returns a valid access token, refreshing if needed. */
29
+ export declare function getValidAccessToken(): Promise<string>;
30
+ export interface ConnectorHandlerResult {
31
+ status: number;
32
+ body: string;
33
+ contentType?: string;
34
+ redirect?: string;
35
+ }
36
+ export declare function handleConnectionsList(): Promise<ConnectorHandlerResult>;
37
+ export declare function handleGmailAuthRedirect(): ConnectorHandlerResult;
38
+ export declare function handleGmailCallback(code: string | null, state: string | null, error: string | null): Promise<ConnectorHandlerResult>;
39
+ export declare function handleGmailTest(): Promise<ConnectorHandlerResult>;
40
+ export declare function handleGmailDisconnect(): Promise<ConnectorHandlerResult>;