kompass-sdk 0.7.0 → 0.9.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.
@@ -1,44 +1,48 @@
1
1
  /**
2
2
  * Agent Discovery Protocol (ADP) Source Adapter
3
- * agentdiscovery.io open protocol for autonomous agent commerce
3
+ * Uses the real ADP v2 API at agentdiscovery.io
4
4
  */
5
- const ADP_BASE = "https://agentdiscovery.io";
5
+ const ADP_BASE = "https://agentdiscovery.io/api/adp/v2";
6
6
  export const adpAdapter = {
7
7
  name: "adp",
8
- displayName: "Agent Discovery Protocol",
8
+ displayName: "Agent Discovery Protocol (ADP v2)",
9
9
  async search(query, options) {
10
10
  try {
11
- // Try the ADP API
12
- const res = await fetch(`${ADP_BASE}/api/agents?q=${encodeURIComponent(query)}&limit=${options?.limit ?? 20}`, {
11
+ // First try discovery endpoint with intent
12
+ const discoverRes = await fetch(`${ADP_BASE}/discover`, {
13
+ method: "POST",
14
+ headers: { "Content-Type": "application/json" },
15
+ body: JSON.stringify({ intent: query, category: query }),
13
16
  signal: AbortSignal.timeout(options?.timeout ?? 5000),
14
17
  });
15
- if (!res.ok)
18
+ if (discoverRes.ok) {
19
+ const data = await discoverRes.json();
20
+ const agents = data.matches ?? data.agents ?? data.results ?? [];
21
+ if (Array.isArray(agents) && agents.length > 0) {
22
+ return agents.slice(0, options?.limit ?? 20).map(mapAdpAgent);
23
+ }
24
+ }
25
+ // Fallback: list all agents
26
+ const listRes = await fetch(`${ADP_BASE}/agents`, {
27
+ signal: AbortSignal.timeout(options?.timeout ?? 5000),
28
+ });
29
+ if (!listRes.ok)
16
30
  return [];
17
- const data = await res.json();
18
- const agents = Array.isArray(data) ? data : data.agents ?? data.data ?? [];
19
- return agents.map((agent) => ({
20
- id: `adp:${agent.id ?? agent.address}`,
21
- nativeId: String(agent.id ?? agent.address),
22
- name: agent.name ?? "ADP Agent",
23
- description: agent.description ?? "",
24
- categories: agent.categories ?? agent.tags ?? ["general"],
25
- capabilities: agent.capabilities ?? [agent.description ?? ""],
26
- source: "adp",
27
- protocol: (agent.protocol ?? "http"),
28
- endpoints: {
29
- http: agent.endpoint ?? agent.url,
30
- a2a: agent.a2aEndpoint,
31
- x402: agent.x402Endpoint,
32
- },
33
- reputation: agent.reputation ? {
34
- score: agent.reputation.score ?? 0,
35
- count: agent.reputation.count ?? 0,
36
- source: "adp",
37
- } : undefined,
38
- pricing: agent.pricing,
39
- verified: agent.verified ?? false,
40
- raw: agent,
41
- }));
31
+ const data = await listRes.json();
32
+ const agents = data.agents ?? [];
33
+ if (!Array.isArray(agents))
34
+ return [];
35
+ // Client-side filter by query
36
+ const lower = query.toLowerCase();
37
+ return agents
38
+ .filter((a) => {
39
+ if (!query)
40
+ return true;
41
+ const text = `${a.name ?? ""} ${a.role ?? ""} ${(a.categories ?? []).join(" ")} ${(a.capabilities ?? []).map((c) => c.description ?? c.key ?? "").join(" ")}`.toLowerCase();
42
+ return lower.split(/\s+/).some((t) => text.includes(t));
43
+ })
44
+ .slice(0, options?.limit ?? 20)
45
+ .map(mapAdpAgent);
42
46
  }
43
47
  catch {
44
48
  return [];
@@ -46,9 +50,7 @@ export const adpAdapter = {
46
50
  },
47
51
  async ping() {
48
52
  try {
49
- const res = await fetch(`${ADP_BASE}/api/health`, {
50
- signal: AbortSignal.timeout(3000),
51
- });
53
+ const res = await fetch(`${ADP_BASE}/agents`, { signal: AbortSignal.timeout(3000) });
52
54
  return res.ok;
53
55
  }
54
56
  catch {
@@ -56,4 +58,23 @@ export const adpAdapter = {
56
58
  }
57
59
  },
58
60
  };
61
+ function mapAdpAgent(agent) {
62
+ return {
63
+ id: `adp:${agent.did ?? agent.id ?? agent.name}`,
64
+ nativeId: agent.did ?? agent.id ?? "",
65
+ name: agent.name ?? "ADP Agent",
66
+ description: (agent.capabilities ?? []).map((c) => c.description ?? c.key ?? "").join(". ") || (agent.role ?? ""),
67
+ categories: agent.categories ?? [agent.role ?? "general"],
68
+ capabilities: (agent.capabilities ?? []).map((c) => `${c.key}: ${c.description ?? ""}`),
69
+ source: "adp",
70
+ protocol: "a2a",
71
+ endpoints: {
72
+ http: agent.endpoint ?? agent.url,
73
+ a2a: agent.a2a_endpoint,
74
+ },
75
+ pricing: agent.pricing,
76
+ verified: !!agent.did,
77
+ raw: agent,
78
+ };
79
+ }
59
80
  //# sourceMappingURL=adp.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"adp.js","sourceRoot":"","sources":["../../src/sources/adp.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAIH,MAAM,QAAQ,GAAG,2BAA2B,CAAC;AAE7C,MAAM,CAAC,MAAM,UAAU,GAAkB;IACvC,IAAI,EAAE,KAAK;IACX,WAAW,EAAE,0BAA0B;IAEvC,KAAK,CAAC,MAAM,CAAC,KAAa,EAAE,OAA6B;QACvD,IAAI,CAAC;YACH,kBAAkB;YAClB,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,GAAG,QAAQ,iBAAiB,kBAAkB,CAAC,KAAK,CAAC,UAAU,OAAO,EAAE,KAAK,IAAI,EAAE,EAAE,EAAE;gBAC7G,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,OAAO,EAAE,OAAO,IAAI,IAAI,CAAC;aACtD,CAAC,CAAC;YAEH,IAAI,CAAC,GAAG,CAAC,EAAE;gBAAE,OAAO,EAAE,CAAC;YAEvB,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC;YAC9B,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC;YAE3E,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC,KAAU,EAAE,EAAE,CAAC,CAAC;gBACjC,EAAE,EAAE,OAAO,KAAK,CAAC,EAAE,IAAI,KAAK,CAAC,OAAO,EAAE;gBACtC,QAAQ,EAAE,MAAM,CAAC,KAAK,CAAC,EAAE,IAAI,KAAK,CAAC,OAAO,CAAC;gBAC3C,IAAI,EAAE,KAAK,CAAC,IAAI,IAAI,WAAW;gBAC/B,WAAW,EAAE,KAAK,CAAC,WAAW,IAAI,EAAE;gBACpC,UAAU,EAAE,KAAK,CAAC,UAAU,IAAI,KAAK,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC;gBACzD,YAAY,EAAE,KAAK,CAAC,YAAY,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,EAAE,CAAC;gBAC7D,MAAM,EAAE,KAAc;gBACtB,QAAQ,EAAE,CAAC,KAAK,CAAC,QAAQ,IAAI,MAAM,CAAQ;gBAC3C,SAAS,EAAE;oBACT,IAAI,EAAE,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC,GAAG;oBACjC,GAAG,EAAE,KAAK,CAAC,WAAW;oBACtB,IAAI,EAAE,KAAK,CAAC,YAAY;iBACzB;gBACD,UAAU,EAAE,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC;oBAC7B,KAAK,EAAE,KAAK,CAAC,UAAU,CAAC,KAAK,IAAI,CAAC;oBAClC,KAAK,EAAE,KAAK,CAAC,UAAU,CAAC,KAAK,IAAI,CAAC;oBAClC,MAAM,EAAE,KAAK;iBACd,CAAC,CAAC,CAAC,SAAS;gBACb,OAAO,EAAE,KAAK,CAAC,OAAO;gBACtB,QAAQ,EAAE,KAAK,CAAC,QAAQ,IAAI,KAAK;gBACjC,GAAG,EAAE,KAAK;aACX,CAAC,CAAC,CAAC;QACN,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,EAAE,CAAC;QACZ,CAAC;IACH,CAAC;IAED,KAAK,CAAC,IAAI;QACR,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,GAAG,QAAQ,aAAa,EAAE;gBAChD,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC;aAClC,CAAC,CAAC;YACH,OAAO,GAAG,CAAC,EAAE,CAAC;QAChB,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;CACF,CAAC"}
1
+ {"version":3,"file":"adp.js","sourceRoot":"","sources":["../../src/sources/adp.ts"],"names":[],"mappings":"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"}
package/dist/unified.d.ts CHANGED
@@ -12,6 +12,7 @@
12
12
  import { type UnifiedAgent } from "./sources/index.js";
13
13
  import { type AggregatorResult } from "./aggregator.js";
14
14
  import { type MatchResult } from "./matching.js";
15
+ import { type RankedAgent } from "./ranking.js";
15
16
  import { type BridgeHireOptions, type BridgeHireResult } from "./bridge.js";
16
17
  import { type DoOptions, type DoResult } from "./router.js";
17
18
  import type { ReputationWriterConfig } from "./reputation-writer.js";
@@ -40,8 +41,10 @@ export interface FindOptions {
40
41
  minScore?: number;
41
42
  }
42
43
  export interface FindResult extends AggregatorResult {
43
- /** Agents re-ranked by multi-layer matching */
44
+ /** Agents re-ranked by multi-layer matching (v1) */
44
45
  ranked: MatchResult[];
46
+ /** Agents re-ranked by v2 engine (Bayesian + geometric + value) */
47
+ rankedV2?: RankedAgent[];
45
48
  }
46
49
  export declare class KompassUnified {
47
50
  private aggregator;
@@ -1 +1 @@
1
- {"version":3,"file":"unified.d.ts","sourceRoot":"","sources":["../src/unified.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAwC,KAAK,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAC7F,OAAO,EAAsC,KAAK,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAC5F,OAAO,EAAc,KAAK,WAAW,EAAE,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAkB,KAAK,iBAAiB,EAAE,KAAK,gBAAgB,EAAE,MAAM,aAAa,CAAC;AAC5F,OAAO,EAAoB,KAAK,SAAS,EAAE,KAAK,QAAQ,EAAE,MAAM,aAAa,CAAC;AAC9E,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAErE,MAAM,WAAW,aAAa;IAC5B,mCAAmC;IACnC,OAAO,CAAC,EAAE,MAAM,GAAG,cAAc,CAAC;IAClC,6CAA6C;IAC7C,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;IAC1B,gDAAgD;IAChD,UAAU,CAAC,EAAE,GAAG,CAAC;IACjB,uDAAuD;IACvD,aAAa,CAAC,EAAE,GAAG,CAAC;IACpB,2EAA2E;IAC3E,gBAAgB,CAAC,EAAE,sBAAsB,CAAC;IAC1C,kCAAkC;IAClC,MAAM,CAAC,EAAE,GAAG,CAAC;CACd;AAED,MAAM,WAAW,WAAW;IAC1B,kBAAkB;IAClB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,uBAAuB;IACvB,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC;IACnB,+BAA+B;IAC/B,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,kCAAkC;IAClC,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,UAAW,SAAQ,gBAAgB;IAClD,+CAA+C;IAC/C,MAAM,EAAE,WAAW,EAAE,CAAC;CACvB;AAED,qBAAa,cAAc;IACzB,OAAO,CAAC,UAAU,CAAa;IAC/B,OAAO,CAAC,MAAM,CAAiB;IAC/B,OAAO,CAAC,MAAM,CAAmB;IACjC,OAAO,CAAC,QAAQ,CAAkB;IAElC,OAAO,CAAC,MAAM,CAAC,CAAgB;IAE/B,OAAO;WAQM,MAAM,CAAC,MAAM,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,cAAc,CAAC;IAcpE;;;OAGG;IACG,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,CAAC;IAsBrE;;;OAGG;IACG,IAAI,CAAC,KAAK,EAAE,YAAY,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,CAAC,iBAAiB,CAAC,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAI9G;;;OAGG;IACG,WAAW,CACf,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,OAAO,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,iBAAiB,CAAC,GACjD,OAAO,CAAC;QAAE,SAAS,EAAE,UAAU,CAAC;QAAC,IAAI,EAAE,gBAAgB,CAAA;KAAE,CAAC;IAa7D;;OAEG;IACG,OAAO,IAAI,OAAO,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,OAAO,CAAA;KAAE,EAAE,CAAC;IAInF;;;;;;;;;OASG;IACG,EAAE,CAAC,OAAO,EAAE,SAAS,GAAG,OAAO,CAAC,QAAQ,CAAC;IAI/C;;OAEG;IACH,IAAI,WAAW,IAAI,MAAM,CAExB;CACF"}
1
+ {"version":3,"file":"unified.d.ts","sourceRoot":"","sources":["../src/unified.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAwC,KAAK,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAC7F,OAAO,EAAsC,KAAK,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAC5F,OAAO,EAAc,KAAK,WAAW,EAAE,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAgB,KAAK,WAAW,EAAE,MAAM,cAAc,CAAC;AAC9D,OAAO,EAAkB,KAAK,iBAAiB,EAAE,KAAK,gBAAgB,EAAE,MAAM,aAAa,CAAC;AAC5F,OAAO,EAAoB,KAAK,SAAS,EAAE,KAAK,QAAQ,EAAE,MAAM,aAAa,CAAC;AAC9E,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAErE,MAAM,WAAW,aAAa;IAC5B,mCAAmC;IACnC,OAAO,CAAC,EAAE,MAAM,GAAG,cAAc,CAAC;IAClC,6CAA6C;IAC7C,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;IAC1B,gDAAgD;IAChD,UAAU,CAAC,EAAE,GAAG,CAAC;IACjB,uDAAuD;IACvD,aAAa,CAAC,EAAE,GAAG,CAAC;IACpB,2EAA2E;IAC3E,gBAAgB,CAAC,EAAE,sBAAsB,CAAC;IAC1C,kCAAkC;IAClC,MAAM,CAAC,EAAE,GAAG,CAAC;CACd;AAED,MAAM,WAAW,WAAW;IAC1B,kBAAkB;IAClB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,uBAAuB;IACvB,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC;IACnB,+BAA+B;IAC/B,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,kCAAkC;IAClC,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,UAAW,SAAQ,gBAAgB;IAClD,oDAAoD;IACpD,MAAM,EAAE,WAAW,EAAE,CAAC;IACtB,mEAAmE;IACnE,QAAQ,CAAC,EAAE,WAAW,EAAE,CAAC;CAC1B;AAED,qBAAa,cAAc;IACzB,OAAO,CAAC,UAAU,CAAa;IAC/B,OAAO,CAAC,MAAM,CAAiB;IAC/B,OAAO,CAAC,MAAM,CAAmB;IACjC,OAAO,CAAC,QAAQ,CAAkB;IAElC,OAAO,CAAC,MAAM,CAAC,CAAgB;IAE/B,OAAO;WAQM,MAAM,CAAC,MAAM,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,cAAc,CAAC;IAcpE;;;OAGG;IACG,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,CAAC;IA0BrE;;;OAGG;IACG,IAAI,CAAC,KAAK,EAAE,YAAY,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,CAAC,iBAAiB,CAAC,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAI9G;;;OAGG;IACG,WAAW,CACf,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,OAAO,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,iBAAiB,CAAC,GACjD,OAAO,CAAC;QAAE,SAAS,EAAE,UAAU,CAAC;QAAC,IAAI,EAAE,gBAAgB,CAAA;KAAE,CAAC;IAa7D;;OAEG;IACG,OAAO,IAAI,OAAO,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,OAAO,CAAA;KAAE,EAAE,CAAC;IAInF;;;;;;;;;OASG;IACG,EAAE,CAAC,OAAO,EAAE,SAAS,GAAG,OAAO,CAAC,QAAQ,CAAC;IAI/C;;OAEG;IACH,IAAI,WAAW,IAAI,MAAM,CAExB;CACF"}
package/dist/unified.js CHANGED
@@ -12,6 +12,7 @@
12
12
  import { createAllSources } from "./sources/index.js";
13
13
  import { Aggregator } from "./aggregator.js";
14
14
  import { rankAgents } from "./matching.js";
15
+ import { rankAgentsV2 } from "./ranking.js";
15
16
  import { ProtocolBridge } from "./bridge.js";
16
17
  import { CapabilityRouter } from "./router.js";
17
18
  export class KompassUnified {
@@ -48,16 +49,19 @@ export class KompassUnified {
48
49
  sources: options?.sources,
49
50
  timeout: options?.timeout,
50
51
  });
51
- // Re-rank with multi-layer matching
52
+ // Re-rank with v2 ranking engine (Bayesian + geometric + value-for-money)
53
+ const rankedV2 = rankAgentsV2(aggregatorResult.agents, query);
54
+ // Also keep v1 for backward compat
52
55
  const ranked = rankAgents(aggregatorResult.agents, query);
53
- // Apply limit and min score
56
+ // Apply limit and min score using v2 composite
54
57
  const filtered = options?.minScore
55
- ? ranked.filter((r) => r.scores.total >= options.minScore)
56
- : ranked;
58
+ ? rankedV2.filter((r) => r.scores.composite >= options.minScore)
59
+ : rankedV2;
57
60
  return {
58
61
  ...aggregatorResult,
59
62
  agents: filtered.slice(0, options?.limit ?? 20).map((r) => r.agent),
60
- ranked: filtered.slice(0, options?.limit ?? 20),
63
+ ranked: ranked.slice(0, options?.limit ?? 20), // v1 for backward compat
64
+ rankedV2: filtered.slice(0, options?.limit ?? 20), // v2 with proper scoring
61
65
  };
62
66
  }
63
67
  /**
@@ -1 +1 @@
1
- {"version":3,"file":"unified.js","sourceRoot":"","sources":["../src/unified.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"unified.js","sourceRoot":"","sources":["../src/unified.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "kompass-sdk",
3
- "version": "0.7.0",
3
+ "version": "0.9.0",
4
4
  "description": "Get any job done. Universal agent capability discovery + coordination across ACP, MCP, x402, A2A, skills, L402, and ERC-8004.",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
package/src/cli.ts CHANGED
@@ -249,10 +249,11 @@ program
249
249
  .option("--log <path>", "write structured agent_log.json to path")
250
250
  .option("--manifest <path>", "write agent.json manifest to path")
251
251
  .option("-w, --wallet <path>", "path to Kompass wallet keystore")
252
+ .option("--context <text>", "context from a previous step to pass to the agent")
252
253
  .action(
253
254
  async (
254
255
  task: string,
255
- opts: { budget: string; cost: string; protocol?: string; dryRun?: boolean; json?: boolean; log?: string; manifest?: string; wallet?: string }
256
+ opts: { budget: string; cost: string; protocol?: string; dryRun?: boolean; json?: boolean; log?: string; manifest?: string; wallet?: string; context?: string }
256
257
  ) => {
257
258
  const { KompassUnified } = await import("./unified.js");
258
259
  const { writeFileSync } = await import("fs");
@@ -277,6 +278,7 @@ program
277
278
  costPreference: opts.cost as any,
278
279
  preferProtocol: opts.protocol,
279
280
  dryRun: opts.dryRun,
281
+ context: opts.context,
280
282
  });
281
283
 
282
284
  if (opts.json) {
package/src/ranking.ts ADDED
@@ -0,0 +1,336 @@
1
+ /**
2
+ * Kompass Ranking Engine
3
+ *
4
+ * Fair, cross-protocol ranking of agents/services using:
5
+ * 1. Source-specific reputation normalization
6
+ * 2. Bayesian average for cold-start protection
7
+ * 3. Value-for-money scoring (cost:quality ratio)
8
+ * 4. Geometric mean composite for final ranking
9
+ * 5. Wilson score for confidence-adjusted reputation
10
+ *
11
+ * Based on: npm scoring, Reddit Wilson, Twitter log-engagement,
12
+ * Amazon sales velocity, Airbnb market-relative pricing,
13
+ * IMDB Bayesian average, AgentRank (0xIntuition)
14
+ */
15
+
16
+ import type { UnifiedAgent } from "./sources/types.js";
17
+
18
+ // ── Types ────────────────────────────────────────────────
19
+
20
+ export interface NormalizedReputation {
21
+ /** Quality signal 0-100 (success rate, reliability) */
22
+ quality: number;
23
+ /** Volume signal 0-100 (log-normalized job count, installs) */
24
+ volume: number;
25
+ /** Recency signal 0-100 (time since last active) */
26
+ recency: number;
27
+ /** Confidence 0-1 (how much evidence backs this score) */
28
+ confidence: number;
29
+ }
30
+
31
+ export interface RankingScores {
32
+ /** How well agent matches the query (0-100) */
33
+ relevance: number;
34
+ /** Bayesian-adjusted quality (0-100) */
35
+ quality: number;
36
+ /** Log-normalized usage volume (0-100) */
37
+ volume: number;
38
+ /** Time-decay recency (0-100) */
39
+ recency: number;
40
+ /** Value for money — quality relative to price (0-100) */
41
+ value: number;
42
+ /** Final composite score */
43
+ composite: number;
44
+ }
45
+
46
+ export interface RankedAgent {
47
+ agent: UnifiedAgent;
48
+ scores: RankingScores;
49
+ explanation: string;
50
+ }
51
+
52
+ // ── Configuration ────────────────────────────────────────
53
+
54
+ const WEIGHTS = {
55
+ relevance: 0.30,
56
+ quality: 0.25,
57
+ volume: 0.15,
58
+ recency: 0.10,
59
+ value: 0.20,
60
+ };
61
+
62
+ const BAYESIAN_C = 10; // Confidence parameter for Bayesian average
63
+ const GLOBAL_AVG_QUALITY = 50; // Prior assumption: average quality is 50/100
64
+
65
+ // ── Step 1: Source-Specific Normalization ─────────────────
66
+
67
+ export function normalizeReputation(agent: UnifiedAgent): NormalizedReputation {
68
+ const rep = agent.reputation;
69
+ if (!rep) return { quality: GLOBAL_AVG_QUALITY, volume: 0, recency: 50, confidence: 0 };
70
+
71
+ switch (rep.source) {
72
+ case "acp":
73
+ // successRate is 0-100%, count is completed jobs
74
+ return {
75
+ quality: rep.score,
76
+ volume: logNormalize(rep.count, 1000),
77
+ recency: recencyScore(agent.lastSeen),
78
+ confidence: Math.min(1, rep.count / 20), // Full confidence at 20+ jobs
79
+ };
80
+
81
+ case "402index":
82
+ // reliability_score is 0-100, count is uptime percentage (0-100)
83
+ return {
84
+ quality: rep.score,
85
+ volume: rep.count, // Uptime as volume proxy
86
+ recency: recencyScore(agent.lastSeen),
87
+ confidence: 0.7, // Automated monitoring = moderate confidence
88
+ };
89
+
90
+ case "8004scan":
91
+ // total_score is 0-100, count is star_count
92
+ return {
93
+ quality: rep.score,
94
+ volume: logNormalize(rep.count, 100),
95
+ recency: recencyScore(agent.lastSeen),
96
+ confidence: Math.min(1, rep.count / 10),
97
+ };
98
+
99
+ case "skills-sh-installs":
100
+ // Installs tell popularity, NOT quality
101
+ return {
102
+ quality: GLOBAL_AVG_QUALITY, // Unknown quality — use prior
103
+ volume: logNormalize(rep.count, 100000),
104
+ recency: 50,
105
+ confidence: Math.min(0.5, rep.count / 1000), // Installs give weak quality signal, cap at 0.5
106
+ };
107
+
108
+ case "clawhub-relevance":
109
+ // Search relevance, NOT reputation
110
+ return {
111
+ quality: GLOBAL_AVG_QUALITY,
112
+ volume: 0,
113
+ recency: 50,
114
+ confidence: 0, // Relevance says nothing about quality
115
+ };
116
+
117
+ default:
118
+ return {
119
+ quality: rep.score > 0 ? Math.min(rep.score, 100) : GLOBAL_AVG_QUALITY,
120
+ volume: logNormalize(rep.count, 100),
121
+ recency: recencyScore(agent.lastSeen),
122
+ confidence: 0.3,
123
+ };
124
+ }
125
+ }
126
+
127
+ // ── Step 2: Bayesian Average ─────────────────────────────
128
+
129
+ /**
130
+ * Bayesian average prevents new agents with 1 job at 100%
131
+ * from outranking established agents with 95% over 200 jobs.
132
+ *
133
+ * Formula: (count * score + C * prior) / (count + C)
134
+ *
135
+ * Examples:
136
+ * - 2 jobs at 100% → 66.7 (pulls toward prior)
137
+ * - 200 jobs at 95% → 93.3 (mostly own data)
138
+ * - 0 jobs → 50.0 (pure prior)
139
+ */
140
+ function bayesianQuality(normalized: NormalizedReputation): number {
141
+ const effectiveCount = normalized.confidence * 20; // Scale confidence to pseudo-count
142
+ return (effectiveCount * normalized.quality + BAYESIAN_C * GLOBAL_AVG_QUALITY) /
143
+ (effectiveCount + BAYESIAN_C);
144
+ }
145
+
146
+ // ── Step 3: Value Score ──────────────────────────────────
147
+
148
+ /**
149
+ * Value = quality relative to price.
150
+ * A $0.01 agent with 82% success is better value than $0.25 at 76%.
151
+ *
152
+ * Free services get a moderate value score (good value but unknown quality).
153
+ */
154
+ function valueScore(agent: UnifiedAgent, bayesianQ: number, allAgents: UnifiedAgent[]): number {
155
+ const price = parseAgentPrice(agent);
156
+
157
+ if (price === null) return 50; // Unknown price → neutral
158
+ if (price === 0) return Math.min(bayesianQ, 70); // Free → capped (free doesn't mean best)
159
+
160
+ // Find min price among agents with similar capabilities
161
+ const comparablePrices = allAgents
162
+ .map(parseAgentPrice)
163
+ .filter((p): p is number => p !== null && p > 0);
164
+
165
+ if (comparablePrices.length === 0) return 50;
166
+
167
+ const minPrice = Math.min(...comparablePrices);
168
+ const priceRatio = minPrice / price; // 1.0 = cheapest, <1 = more expensive
169
+
170
+ // Value = quality-adjusted price efficiency
171
+ // High quality + low price = high value
172
+ // Low quality + high price = low value
173
+ return Math.min(100, (bayesianQ * 0.6 + priceRatio * 100 * 0.4));
174
+ }
175
+
176
+ // ── Step 4: Geometric Mean Composite ─────────────────────
177
+
178
+ /**
179
+ * Geometric mean ensures ALL factors must be non-zero.
180
+ * An agent that's free but unreliable scores zero.
181
+ * An agent that's perfect quality but irrelevant also scores low.
182
+ */
183
+ function compositeScore(scores: Omit<RankingScores, "composite">): number {
184
+ const eps = 1; // Prevent zero from killing the product
185
+ const r = scores.relevance + eps;
186
+ const q = scores.quality + eps;
187
+ const v = scores.volume + eps;
188
+ const t = scores.recency + eps;
189
+ const p = scores.value + eps;
190
+
191
+ const totalWeight = WEIGHTS.relevance + WEIGHTS.quality + WEIGHTS.volume + WEIGHTS.recency + WEIGHTS.value;
192
+
193
+ return Math.pow(
194
+ Math.pow(r, WEIGHTS.relevance) *
195
+ Math.pow(q, WEIGHTS.quality) *
196
+ Math.pow(v, WEIGHTS.volume) *
197
+ Math.pow(t, WEIGHTS.recency) *
198
+ Math.pow(p, WEIGHTS.value),
199
+ 1 / totalWeight
200
+ );
201
+ }
202
+
203
+ // ── Step 5: Wilson Score (Confidence-Adjusted) ───────────
204
+
205
+ /**
206
+ * Lower bound of Wilson score confidence interval.
207
+ * Penalizes agents with few interactions even if success rate is high.
208
+ *
209
+ * Agent with 5/5 success → Wilson ~0.57
210
+ * Agent with 95/100 success → Wilson ~0.90
211
+ */
212
+ export function wilsonLowerBound(positive: number, total: number, z = 1.96): number {
213
+ if (total === 0) return 0;
214
+ const phat = positive / total;
215
+ const denominator = 1 + z * z / total;
216
+ const center = phat + z * z / (2 * total);
217
+ const spread = z * Math.sqrt((phat * (1 - phat) + z * z / (4 * total)) / total);
218
+ return Math.max(0, (center - spread) / denominator);
219
+ }
220
+
221
+ // ── Main Ranking Function ────────────────────────────────
222
+
223
+ /**
224
+ * Rank agents using the full pipeline:
225
+ * Normalize → Bayesian adjust → Value score → Geometric composite
226
+ */
227
+ export function rankAgentsV2(
228
+ agents: UnifiedAgent[],
229
+ query: string
230
+ ): RankedAgent[] {
231
+ // Calculate relevance for each agent (reuse existing BM25 + structured matching)
232
+ const queryTerms = query.toLowerCase().split(/\s+/).filter((t) => t.length > 2);
233
+
234
+ return agents
235
+ .map((agent) => {
236
+ // Relevance (BM25-lite + structured)
237
+ const relevance = calculateRelevance(agent, queryTerms);
238
+
239
+ // Normalize reputation per source
240
+ const normalized = normalizeReputation(agent);
241
+
242
+ // Bayesian-adjusted quality
243
+ const quality = bayesianQuality(normalized);
244
+
245
+ // Volume
246
+ const volume = normalized.volume;
247
+
248
+ // Recency
249
+ const recency = normalized.recency;
250
+
251
+ // Value for money
252
+ const value = valueScore(agent, quality, agents);
253
+
254
+ const scores: RankingScores = {
255
+ relevance,
256
+ quality,
257
+ volume,
258
+ recency,
259
+ value,
260
+ composite: 0,
261
+ };
262
+
263
+ scores.composite = compositeScore(scores);
264
+
265
+ // Build explanation
266
+ const explanation = buildExplanation(agent, scores, normalized);
267
+
268
+ return { agent, scores, explanation };
269
+ })
270
+ .sort((a, b) => b.scores.composite - a.scores.composite);
271
+ }
272
+
273
+ // ── Helpers ──────────────────────────────────────────────
274
+
275
+ function logNormalize(value: number, maxExpected: number): number {
276
+ if (value <= 0) return 0;
277
+ return Math.min(100, (Math.log(1 + value) / Math.log(1 + maxExpected)) * 100);
278
+ }
279
+
280
+ function recencyScore(lastSeen?: number): number {
281
+ if (!lastSeen) return 50; // Unknown → neutral
282
+ const hoursSince = (Date.now() - lastSeen) / (1000 * 60 * 60);
283
+ if (hoursSince < 1) return 100;
284
+ if (hoursSince < 24) return 80;
285
+ if (hoursSince < 168) return 60; // 1 week
286
+ if (hoursSince < 720) return 40; // 1 month
287
+ return 20;
288
+ }
289
+
290
+ function parseAgentPrice(agent: UnifiedAgent): number | null {
291
+ if (!agent.pricing?.amount) return null;
292
+ const amount = agent.pricing.amount;
293
+ if (typeof amount === "number") return amount;
294
+ const parsed = parseFloat(amount);
295
+ return isNaN(parsed) ? null : parsed;
296
+ }
297
+
298
+ function calculateRelevance(agent: UnifiedAgent, queryTerms: string[]): number {
299
+ const text = `${agent.name} ${agent.description} ${agent.categories.join(" ")} ${agent.capabilities.join(" ")}`.toLowerCase();
300
+ let score = 0;
301
+
302
+ for (const term of queryTerms) {
303
+ if (agent.name.toLowerCase().includes(term)) score += 15;
304
+ if (agent.description.toLowerCase().includes(term)) score += 8;
305
+ if (agent.categories.some((c) => c.includes(term))) score += 12;
306
+ if (agent.capabilities.some((c) => c.toLowerCase().includes(term))) score += 5;
307
+ }
308
+
309
+ return Math.min(100, score);
310
+ }
311
+
312
+ function buildExplanation(agent: UnifiedAgent, scores: RankingScores, normalized: NormalizedReputation): string {
313
+ const parts: string[] = [];
314
+
315
+ if (scores.relevance > 30) parts.push("Strong match");
316
+ else if (scores.relevance > 15) parts.push("Moderate match");
317
+ else parts.push("Weak match");
318
+
319
+ if (normalized.confidence > 0.5) {
320
+ if (scores.quality > 70) parts.push("proven quality");
321
+ else if (scores.quality > 50) parts.push("decent quality");
322
+ else parts.push("unproven quality");
323
+ } else {
324
+ parts.push("limited track record");
325
+ }
326
+
327
+ const price = parseAgentPrice(agent);
328
+ if (price === 0) parts.push("free");
329
+ else if (price !== null && price < 0.05) parts.push("very affordable");
330
+ else if (price !== null && price > 1) parts.push("premium priced");
331
+
332
+ if (scores.value > 70) parts.push("great value");
333
+ else if (scores.value < 30) parts.push("questionable value");
334
+
335
+ return parts.join(". ") + ".";
336
+ }
package/src/router.ts CHANGED
@@ -36,6 +36,8 @@ export interface DoOptions {
36
36
  timeout?: number;
37
37
  /** Return options without executing (dry run) */
38
38
  dryRun?: boolean;
39
+ /** Context from a previous step — passed to the agent as additional info */
40
+ context?: string;
39
41
  /** Additional payload to send to the capability */
40
42
  payload?: Record<string, unknown>;
41
43
  }
@@ -85,7 +87,11 @@ export class CapabilityRouter {
85
87
  }
86
88
 
87
89
  async do(options: DoOptions): Promise<DoResult> {
88
- const { task, dryRun = false } = options;
90
+ const { dryRun = false } = options;
91
+ // Prepend context from previous step if provided
92
+ const task = options.context
93
+ ? `${options.task}\n\nContext from previous step:\n${options.context.slice(0, 2000)}`
94
+ : options.task;
89
95
 
90
96
  // 1. Search all sources
91
97
  const searchResult = await this.aggregator.search(task, {