@xpr-agents/sdk 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/utils.js ADDED
@@ -0,0 +1,235 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.calculateTrustScore = calculateTrustScore;
4
+ exports.getTrustRating = getTrustRating;
5
+ exports.validationResultFromNumber = validationResultFromNumber;
6
+ exports.validationResultToNumber = validationResultToNumber;
7
+ exports.disputeStatusFromNumber = disputeStatusFromNumber;
8
+ exports.safeJsonParse = safeJsonParse;
9
+ exports.parseCapabilities = parseCapabilities;
10
+ exports.parseSpecializations = parseSpecializations;
11
+ exports.parseTags = parseTags;
12
+ exports.formatXpr = formatXpr;
13
+ exports.parseXpr = parseXpr;
14
+ exports.safeParseInt = safeParseInt;
15
+ exports.formatTimestamp = formatTimestamp;
16
+ exports.calculateWeightedAverage = calculateWeightedAverage;
17
+ exports.isValidAccountName = isValidAccountName;
18
+ exports.isValidUrl = isValidUrl;
19
+ exports.getKycWeight = getKycWeight;
20
+ /**
21
+ * Calculate trust score for an agent
22
+ * Combines KYC level, stake, reputation, and longevity
23
+ * @param agent - Agent data
24
+ * @param agentScore - Aggregated score from feedback
25
+ * @param kycLevel - KYC verification level (0-4)
26
+ * @param stakeAmount - Staked XPR in smallest units (optional, fetched from system staking)
27
+ */
28
+ function calculateTrustScore(agent, agentScore, kycLevel, stakeAmount = 0) {
29
+ const breakdown = {
30
+ kyc: 0,
31
+ stake: 0,
32
+ reputation: 0,
33
+ longevity: 0,
34
+ };
35
+ // KYC score (0-30 points)
36
+ // Level 0 = 0, Level 1 = 10, Level 2 = 20, Level 3 = 30
37
+ breakdown.kyc = Math.min(kycLevel * 10, 30);
38
+ // Stake score (0-20 points, caps at 10000 XPR)
39
+ // Every 500 XPR = 1 point, max 20 points
40
+ const stakeXpr = stakeAmount / 10000; // Convert from smallest unit
41
+ breakdown.stake = Math.min(Math.floor(stakeXpr / 500), 20);
42
+ // Reputation score (0-40 points)
43
+ if (agentScore && agentScore.total_weight > 0) {
44
+ // avg_score is 0-10000 (representing 0-100.00%)
45
+ // We need to convert to 0-40 points
46
+ // If avg_score is 10000 (100%), that means perfect 5/5 rating
47
+ breakdown.reputation = Math.floor((agentScore.avg_score / 10000) * 40);
48
+ }
49
+ // Longevity score (0-10 points, 1 point per month)
50
+ const now = Math.floor(Date.now() / 1000);
51
+ const monthsActive = Math.floor((now - agent.registered_at) / (30 * 24 * 60 * 60));
52
+ breakdown.longevity = Math.min(monthsActive, 10);
53
+ // Total score (0-100)
54
+ const total = breakdown.kyc + breakdown.stake + breakdown.reputation + breakdown.longevity;
55
+ return {
56
+ agent: agent.account,
57
+ total,
58
+ breakdown,
59
+ rating: getTrustRating(total),
60
+ };
61
+ }
62
+ /**
63
+ * Get trust rating label from numeric score
64
+ */
65
+ function getTrustRating(score) {
66
+ if (score >= 80)
67
+ return 'verified';
68
+ if (score >= 60)
69
+ return 'high';
70
+ if (score >= 40)
71
+ return 'medium';
72
+ if (score >= 20)
73
+ return 'low';
74
+ return 'untrusted';
75
+ }
76
+ /**
77
+ * Convert validation result number to string
78
+ */
79
+ function validationResultFromNumber(result) {
80
+ switch (result) {
81
+ case 0:
82
+ return 'fail';
83
+ case 1:
84
+ return 'pass';
85
+ case 2:
86
+ return 'partial';
87
+ default:
88
+ return 'fail';
89
+ }
90
+ }
91
+ /**
92
+ * Convert validation result string to number
93
+ */
94
+ function validationResultToNumber(result) {
95
+ switch (result) {
96
+ case 'fail':
97
+ return 0;
98
+ case 'pass':
99
+ return 1;
100
+ case 'partial':
101
+ return 2;
102
+ default:
103
+ return 0;
104
+ }
105
+ }
106
+ /**
107
+ * Convert dispute status number to string
108
+ */
109
+ function disputeStatusFromNumber(status) {
110
+ switch (status) {
111
+ case 0:
112
+ return 'pending';
113
+ case 1:
114
+ return 'upheld';
115
+ case 2:
116
+ return 'rejected';
117
+ case 3:
118
+ return 'cancelled';
119
+ default:
120
+ return 'pending';
121
+ }
122
+ }
123
+ /**
124
+ * Parse JSON safely with fallback
125
+ */
126
+ function safeJsonParse(json, fallback) {
127
+ try {
128
+ return JSON.parse(json);
129
+ }
130
+ catch {
131
+ return fallback;
132
+ }
133
+ }
134
+ /**
135
+ * Parse capabilities string to array
136
+ */
137
+ function parseCapabilities(capabilities) {
138
+ if (!capabilities)
139
+ return [];
140
+ return safeJsonParse(capabilities, []);
141
+ }
142
+ /**
143
+ * Parse specializations string to array
144
+ */
145
+ function parseSpecializations(specializations) {
146
+ if (!specializations)
147
+ return [];
148
+ return safeJsonParse(specializations, []);
149
+ }
150
+ /**
151
+ * Parse tags string to array (comma-separated)
152
+ */
153
+ function parseTags(tags) {
154
+ if (!tags)
155
+ return [];
156
+ return tags.split(',').map((t) => t.trim()).filter((t) => t.length > 0);
157
+ }
158
+ /**
159
+ * Format XPR amount from smallest unit
160
+ */
161
+ function formatXpr(amount) {
162
+ return (amount / 10000).toFixed(4) + ' XPR';
163
+ }
164
+ /**
165
+ * Parse XPR amount string to smallest unit (integer math to avoid float precision issues)
166
+ * "100.5000 XPR" → 1005000, "0.7000 XPR" → 7000
167
+ */
168
+ function parseXpr(amount) {
169
+ const match = amount.match(/^(\d+)(?:\.(\d{1,4}))?/);
170
+ if (!match)
171
+ return 0;
172
+ const whole = parseInt(match[1], 10) || 0;
173
+ const fracStr = (match[2] || '').padEnd(4, '0').slice(0, 4);
174
+ const frac = parseInt(fracStr, 10) || 0;
175
+ return whole * 10000 + frac;
176
+ }
177
+ /**
178
+ * Safe parseInt with fallback - returns fallback on NaN
179
+ */
180
+ function safeParseInt(value, fallback = 0) {
181
+ if (value === undefined || value === null || value === '')
182
+ return fallback;
183
+ const parsed = parseInt(value, 10);
184
+ return isNaN(parsed) ? fallback : parsed;
185
+ }
186
+ /**
187
+ * Format timestamp to ISO string
188
+ */
189
+ function formatTimestamp(timestamp) {
190
+ return new Date(timestamp * 1000).toISOString();
191
+ }
192
+ /**
193
+ * Calculate weighted average score
194
+ */
195
+ function calculateWeightedAverage(scores) {
196
+ if (scores.length === 0)
197
+ return 0;
198
+ let totalScore = 0;
199
+ let totalWeight = 0;
200
+ for (const { score, weight } of scores) {
201
+ totalScore += score * weight;
202
+ totalWeight += weight;
203
+ }
204
+ if (totalWeight === 0)
205
+ return 0;
206
+ return totalScore / totalWeight;
207
+ }
208
+ /**
209
+ * Validate XPR account name
210
+ * Must be 1-12 characters, a-z, 1-5, and .
211
+ */
212
+ function isValidAccountName(name) {
213
+ if (!name || name.length > 12)
214
+ return false;
215
+ return /^[a-z1-5.]+$/.test(name);
216
+ }
217
+ /**
218
+ * Validate URL format
219
+ */
220
+ function isValidUrl(url) {
221
+ try {
222
+ new URL(url);
223
+ return true;
224
+ }
225
+ catch {
226
+ return false;
227
+ }
228
+ }
229
+ /**
230
+ * Get KYC level weight for feedback scoring
231
+ */
232
+ function getKycWeight(kycLevel) {
233
+ return 1 + kycLevel;
234
+ }
235
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../src/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Agent,\n  AgentScore,\n  TrustScore,\n  TrustScoreBreakdown,\n  TrustRating,\n  ValidationResult,\n  DisputeStatus,\n} from './types';\n\n/**\n * Calculate trust score for an agent\n * Combines KYC level, stake, reputation, and longevity\n * @param agent - Agent data\n * @param agentScore - Aggregated score from feedback\n * @param kycLevel - KYC verification level (0-4)\n * @param stakeAmount - Staked XPR in smallest units (optional, fetched from system staking)\n */\nexport function calculateTrustScore(\n  agent: Agent,\n  agentScore: AgentScore | null,\n  kycLevel: number,\n  stakeAmount: number = 0\n): TrustScore {\n  const breakdown: TrustScoreBreakdown = {\n    kyc: 0,\n    stake: 0,\n    reputation: 0,\n    longevity: 0,\n  };\n\n  // KYC score (0-30 points)\n  // Level 0 = 0, Level 1 = 10, Level 2 = 20, Level 3 = 30\n  breakdown.kyc = Math.min(kycLevel * 10, 30);\n\n  // Stake score (0-20 points, caps at 10000 XPR)\n  // Every 500 XPR = 1 point, max 20 points\n  const stakeXpr = stakeAmount / 10000; // Convert from smallest unit\n  breakdown.stake = Math.min(Math.floor(stakeXpr / 500), 20);\n\n  // Reputation score (0-40 points)\n  if (agentScore && agentScore.total_weight > 0) {\n    // avg_score is 0-10000 (representing 0-100.00%)\n    // We need to convert to 0-40 points\n    // If avg_score is 10000 (100%), that means perfect 5/5 rating\n    breakdown.reputation = Math.floor((agentScore.avg_score / 10000) * 40);\n  }\n\n  // Longevity score (0-10 points, 1 point per month)\n  const now = Math.floor(Date.now() / 1000);\n  const monthsActive = Math.floor((now - agent.registered_at) / (30 * 24 * 60 * 60));\n  breakdown.longevity = Math.min(monthsActive, 10);\n\n  // Total score (0-100)\n  const total = breakdown.kyc + breakdown.stake + breakdown.reputation + breakdown.longevity;\n\n  return {\n    agent: agent.account,\n    total,\n    breakdown,\n    rating: getTrustRating(total),\n  };\n}\n\n/**\n * Get trust rating label from numeric score\n */\nexport function getTrustRating(score: number): TrustRating {\n  if (score >= 80) return 'verified';\n  if (score >= 60) return 'high';\n  if (score >= 40) return 'medium';\n  if (score >= 20) return 'low';\n  return 'untrusted';\n}\n\n/**\n * Convert validation result number to string\n */\nexport function validationResultFromNumber(result: number): ValidationResult {\n  switch (result) {\n    case 0:\n      return 'fail';\n    case 1:\n      return 'pass';\n    case 2:\n      return 'partial';\n    default:\n      return 'fail';\n  }\n}\n\n/**\n * Convert validation result string to number\n */\nexport function validationResultToNumber(result: ValidationResult): number {\n  switch (result) {\n    case 'fail':\n      return 0;\n    case 'pass':\n      return 1;\n    case 'partial':\n      return 2;\n    default:\n      return 0;\n  }\n}\n\n/**\n * Convert dispute status number to string\n */\nexport function disputeStatusFromNumber(status: number): DisputeStatus {\n  switch (status) {\n    case 0:\n      return 'pending';\n    case 1:\n      return 'upheld';\n    case 2:\n      return 'rejected';\n    case 3:\n      return 'cancelled';\n    default:\n      return 'pending';\n  }\n}\n\n/**\n * Parse JSON safely with fallback\n */\nexport function safeJsonParse<T>(json: string, fallback: T): T {\n  try {\n    return JSON.parse(json) as T;\n  } catch {\n    return fallback;\n  }\n}\n\n/**\n * Parse capabilities string to array\n */\nexport function parseCapabilities(capabilities: string): string[] {\n  if (!capabilities) return [];\n  return safeJsonParse<string[]>(capabilities, []);\n}\n\n/**\n * Parse specializations string to array\n */\nexport function parseSpecializations(specializations: string): string[] {\n  if (!specializations) return [];\n  return safeJsonParse<string[]>(specializations, []);\n}\n\n/**\n * Parse tags string to array (comma-separated)\n */\nexport function parseTags(tags: string): string[] {\n  if (!tags) return [];\n  return tags.split(',').map((t) => t.trim()).filter((t) => t.length > 0);\n}\n\n/**\n * Format XPR amount from smallest unit\n */\nexport function formatXpr(amount: number): string {\n  return (amount / 10000).toFixed(4) + ' XPR';\n}\n\n/**\n * Parse XPR amount string to smallest unit (integer math to avoid float precision issues)\n * \"100.5000 XPR\" → 1005000, \"0.7000 XPR\" → 7000\n */\nexport function parseXpr(amount: string): number {\n  const match = amount.match(/^(\\d+)(?:\\.(\\d{1,4}))?/);\n  if (!match) return 0;\n  const whole = parseInt(match[1], 10) || 0;\n  const fracStr = (match[2] || '').padEnd(4, '0').slice(0, 4);\n  const frac = parseInt(fracStr, 10) || 0;\n  return whole * 10000 + frac;\n}\n\n/**\n * Safe parseInt with fallback - returns fallback on NaN\n */\nexport function safeParseInt(value: string | undefined | null, fallback: number = 0): number {\n  if (value === undefined || value === null || value === '') return fallback;\n  const parsed = parseInt(value, 10);\n  return isNaN(parsed) ? fallback : parsed;\n}\n\n/**\n * Format timestamp to ISO string\n */\nexport function formatTimestamp(timestamp: number): string {\n  return new Date(timestamp * 1000).toISOString();\n}\n\n/**\n * Calculate weighted average score\n */\nexport function calculateWeightedAverage(\n  scores: Array<{ score: number; weight: number }>\n): number {\n  if (scores.length === 0) return 0;\n\n  let totalScore = 0;\n  let totalWeight = 0;\n\n  for (const { score, weight } of scores) {\n    totalScore += score * weight;\n    totalWeight += weight;\n  }\n\n  if (totalWeight === 0) return 0;\n  return totalScore / totalWeight;\n}\n\n/**\n * Validate XPR account name\n * Must be 1-12 characters, a-z, 1-5, and .\n */\nexport function isValidAccountName(name: string): boolean {\n  if (!name || name.length > 12) return false;\n  return /^[a-z1-5.]+$/.test(name);\n}\n\n/**\n * Validate URL format\n */\nexport function isValidUrl(url: string): boolean {\n  try {\n    new URL(url);\n    return true;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Get KYC level weight for feedback scoring\n */\nexport function getKycWeight(kycLevel: number): number {\n  return 1 + kycLevel;\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,65 @@
1
+ {
2
+ "name": "@xpr-agents/sdk",
3
+ "version": "0.1.0",
4
+ "description": "TypeScript SDK for XPR Network Trustless Agent Registry - register agents, submit feedback, validate outputs, and manage escrow jobs",
5
+ "author": "XPR Network",
6
+ "license": "MIT",
7
+ "homepage": "https://github.com/XPRNetwork/xpr-agents#readme",
8
+ "repository": {
9
+ "type": "git",
10
+ "url": "git+https://github.com/XPRNetwork/xpr-agents.git",
11
+ "directory": "sdk"
12
+ },
13
+ "bugs": {
14
+ "url": "https://github.com/XPRNetwork/xpr-agents/issues"
15
+ },
16
+ "keywords": [
17
+ "xpr",
18
+ "xpr-network",
19
+ "proton",
20
+ "blockchain",
21
+ "agents",
22
+ "ai-agents",
23
+ "registry",
24
+ "reputation",
25
+ "validation",
26
+ "escrow",
27
+ "trustless"
28
+ ],
29
+ "main": "dist/index.js",
30
+ "types": "dist/index.d.ts",
31
+ "files": [
32
+ "dist",
33
+ "README.md"
34
+ ],
35
+ "sideEffects": false,
36
+ "engines": {
37
+ "node": ">=18.0.0"
38
+ },
39
+ "scripts": {
40
+ "build": "tsc",
41
+ "clean": "rm -rf dist",
42
+ "prepublishOnly": "npm run clean && npm run build",
43
+ "test": "jest",
44
+ "lint": "eslint src/**/*.ts"
45
+ },
46
+ "dependencies": {
47
+ "@proton/js": "^28.1.2"
48
+ },
49
+ "peerDependencies": {
50
+ "@proton/web-sdk": "^4.4.0"
51
+ },
52
+ "peerDependenciesMeta": {
53
+ "@proton/web-sdk": {
54
+ "optional": true
55
+ }
56
+ },
57
+ "devDependencies": {
58
+ "@proton/web-sdk": "^4.4.1",
59
+ "@types/jest": "^29.5.0",
60
+ "@types/node": "^20.0.0",
61
+ "jest": "^29.5.0",
62
+ "ts-jest": "^29.1.0",
63
+ "typescript": "^5.0.0"
64
+ }
65
+ }