@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/README.md +165 -0
- package/dist/AgentRegistry.d.ts +182 -0
- package/dist/AgentRegistry.js +892 -0
- package/dist/EscrowRegistry.d.ts +253 -0
- package/dist/EscrowRegistry.js +731 -0
- package/dist/FeedbackRegistry.d.ts +83 -0
- package/dist/FeedbackRegistry.js +424 -0
- package/dist/ValidationRegistry.d.ts +115 -0
- package/dist/ValidationRegistry.js +638 -0
- package/dist/index.d.ts +25 -0
- package/dist/index.js +59 -0
- package/dist/types.d.ts +345 -0
- package/dist/types.js +4 -0
- package/dist/utils.d.ts +79 -0
- package/dist/utils.js +235 -0
- package/package.json +65 -0
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
|
+
}
|