@xyo-network/crypto-nft-collection-diviner-score-plugin 2.99.4 → 2.99.5

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 (89) hide show
  1. package/dist/browser/Plugin.d.cts +1 -1
  2. package/dist/browser/Plugin.d.mts +1 -1
  3. package/dist/browser/Plugin.d.ts +1 -1
  4. package/dist/browser/index.cjs +105 -56
  5. package/dist/browser/index.cjs.map +1 -1
  6. package/dist/browser/index.d.cts +3 -3
  7. package/dist/browser/index.d.mts +3 -3
  8. package/dist/browser/index.d.ts +3 -3
  9. package/dist/browser/index.mjs +190 -0
  10. package/dist/browser/{index.js.map → index.mjs.map} +1 -1
  11. package/dist/browser/lib/index.d.cts +1 -1
  12. package/dist/browser/lib/index.d.mts +1 -1
  13. package/dist/browser/lib/index.d.ts +1 -1
  14. package/dist/browser/lib/rating/analyzeNftCollection.d.cts +1 -1
  15. package/dist/browser/lib/rating/analyzeNftCollection.d.mts +1 -1
  16. package/dist/browser/lib/rating/analyzeNftCollection.d.ts +1 -1
  17. package/dist/browser/lib/rating/criteria/scoring/index.d.cts +2 -2
  18. package/dist/browser/lib/rating/criteria/scoring/index.d.mts +2 -2
  19. package/dist/browser/lib/rating/criteria/scoring/index.d.ts +2 -2
  20. package/dist/browser/lib/rating/criteria/scoring/metadata/index.d.cts +1 -1
  21. package/dist/browser/lib/rating/criteria/scoring/metadata/index.d.mts +1 -1
  22. package/dist/browser/lib/rating/criteria/scoring/metadata/index.d.ts +1 -1
  23. package/dist/browser/lib/rating/index.d.cts +2 -2
  24. package/dist/browser/lib/rating/index.d.mts +2 -2
  25. package/dist/browser/lib/rating/index.d.ts +2 -2
  26. package/dist/neutral/Plugin.d.cts +1 -1
  27. package/dist/neutral/Plugin.d.mts +1 -1
  28. package/dist/neutral/Plugin.d.ts +1 -1
  29. package/dist/neutral/index.cjs +105 -56
  30. package/dist/neutral/index.cjs.map +1 -1
  31. package/dist/neutral/index.d.cts +3 -3
  32. package/dist/neutral/index.d.mts +3 -3
  33. package/dist/neutral/index.d.ts +3 -3
  34. package/dist/neutral/index.mjs +190 -0
  35. package/dist/neutral/{index.js.map → index.mjs.map} +1 -1
  36. package/dist/neutral/lib/index.d.cts +1 -1
  37. package/dist/neutral/lib/index.d.mts +1 -1
  38. package/dist/neutral/lib/index.d.ts +1 -1
  39. package/dist/neutral/lib/rating/analyzeNftCollection.d.cts +1 -1
  40. package/dist/neutral/lib/rating/analyzeNftCollection.d.mts +1 -1
  41. package/dist/neutral/lib/rating/analyzeNftCollection.d.ts +1 -1
  42. package/dist/neutral/lib/rating/criteria/scoring/index.d.cts +2 -2
  43. package/dist/neutral/lib/rating/criteria/scoring/index.d.mts +2 -2
  44. package/dist/neutral/lib/rating/criteria/scoring/index.d.ts +2 -2
  45. package/dist/neutral/lib/rating/criteria/scoring/metadata/index.d.cts +1 -1
  46. package/dist/neutral/lib/rating/criteria/scoring/metadata/index.d.mts +1 -1
  47. package/dist/neutral/lib/rating/criteria/scoring/metadata/index.d.ts +1 -1
  48. package/dist/neutral/lib/rating/index.d.cts +2 -2
  49. package/dist/neutral/lib/rating/index.d.mts +2 -2
  50. package/dist/neutral/lib/rating/index.d.ts +2 -2
  51. package/dist/node/Plugin.d.cts +1 -1
  52. package/dist/node/Plugin.d.mts +1 -1
  53. package/dist/node/Plugin.d.ts +1 -1
  54. package/dist/node/index.cjs +111 -58
  55. package/dist/node/index.cjs.map +1 -1
  56. package/dist/node/index.d.cts +3 -3
  57. package/dist/node/index.d.mts +3 -3
  58. package/dist/node/index.d.ts +3 -3
  59. package/dist/node/index.mjs +194 -0
  60. package/dist/node/{index.js.map → index.mjs.map} +1 -1
  61. package/dist/node/lib/index.d.cts +1 -1
  62. package/dist/node/lib/index.d.mts +1 -1
  63. package/dist/node/lib/index.d.ts +1 -1
  64. package/dist/node/lib/rating/analyzeNftCollection.d.cts +1 -1
  65. package/dist/node/lib/rating/analyzeNftCollection.d.mts +1 -1
  66. package/dist/node/lib/rating/analyzeNftCollection.d.ts +1 -1
  67. package/dist/node/lib/rating/criteria/scoring/index.d.cts +2 -2
  68. package/dist/node/lib/rating/criteria/scoring/index.d.mts +2 -2
  69. package/dist/node/lib/rating/criteria/scoring/index.d.ts +2 -2
  70. package/dist/node/lib/rating/criteria/scoring/metadata/index.d.cts +1 -1
  71. package/dist/node/lib/rating/criteria/scoring/metadata/index.d.mts +1 -1
  72. package/dist/node/lib/rating/criteria/scoring/metadata/index.d.ts +1 -1
  73. package/dist/node/lib/rating/index.d.cts +2 -2
  74. package/dist/node/lib/rating/index.d.mts +2 -2
  75. package/dist/node/lib/rating/index.d.ts +2 -2
  76. package/package.json +20 -20
  77. package/src/Diviner.ts +1 -1
  78. package/src/Plugin.ts +1 -1
  79. package/src/index.ts +3 -3
  80. package/src/lib/index.ts +1 -1
  81. package/src/lib/rating/analyzeNftCollection.ts +2 -2
  82. package/src/lib/rating/criteria/index.ts +1 -1
  83. package/src/lib/rating/criteria/scoring/index.ts +2 -2
  84. package/src/lib/rating/criteria/scoring/metadata/index.ts +1 -1
  85. package/src/lib/rating/criteria/scoring/metadata/metadata.ts +2 -2
  86. package/src/lib/rating/index.ts +2 -2
  87. package/dist/browser/index.js +0 -143
  88. package/dist/neutral/index.js +0 -143
  89. package/dist/node/index.js +0 -143
@@ -1,4 +1,4 @@
1
- import { NftCollectionScoreDiviner } from './Diviner.js';
1
+ import { NftCollectionScoreDiviner } from './Diviner.ts';
2
2
  export declare const NftCollectionScoreDivinerPlugin: () => import("@xyo-network/payloadset-plugin").PayloadSetDivinerPlugin<NftCollectionScoreDiviner<import("@xylabs/object").BaseParamsFields & {
3
3
  account?: import("@xyo-network/account-model").AccountInstance | "random";
4
4
  addToResolvers?: boolean;
@@ -1,4 +1,4 @@
1
- import { NftCollectionScoreDiviner } from './Diviner.js';
1
+ import { NftCollectionScoreDiviner } from './Diviner.ts';
2
2
  export declare const NftCollectionScoreDivinerPlugin: () => import("@xyo-network/payloadset-plugin").PayloadSetDivinerPlugin<NftCollectionScoreDiviner<import("@xylabs/object").BaseParamsFields & {
3
3
  account?: import("@xyo-network/account-model").AccountInstance | "random";
4
4
  addToResolvers?: boolean;
@@ -1,4 +1,4 @@
1
- import { NftCollectionScoreDiviner } from './Diviner.js';
1
+ import { NftCollectionScoreDiviner } from './Diviner.ts';
2
2
  export declare const NftCollectionScoreDivinerPlugin: () => import("@xyo-network/payloadset-plugin").PayloadSetDivinerPlugin<NftCollectionScoreDiviner<import("@xylabs/object").BaseParamsFields & {
3
3
  account?: import("@xyo-network/account-model").AccountInstance | "random";
4
4
  addToResolvers?: boolean;
@@ -3,6 +3,7 @@ var __defProp = Object.defineProperty;
3
3
  var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
5
5
  var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
6
7
  var __export = (target, all) => {
7
8
  for (var name in all)
8
9
  __defProp(target, name, { get: all[name], enumerable: true });
@@ -36,46 +37,67 @@ var import_payload_builder = require("@xyo-network/payload-builder");
36
37
  // src/lib/rating/criteria/scoring/metadata/scoreIndividualAttributes.ts
37
38
  var import_crypto_nft_score_model = require("@xyo-network/crypto-nft-score-model");
38
39
  var maxScore = 10;
39
- var scoreIndividualAttributes = (info) => {
40
+ var scoreIndividualAttributes = /* @__PURE__ */ __name((info) => {
40
41
  const { attributes } = info.metrics.metadata;
41
42
  const entries = Object.entries(attributes);
42
- if (entries.length === 0) return [0, maxScore];
43
+ if (entries.length === 0) return [
44
+ 0,
45
+ maxScore
46
+ ];
43
47
  const scores = Object.entries(attributes).flatMap(([_trait, { values }]) => {
44
48
  return Object.entries(values).map(([_traitValue, metrics]) => {
45
49
  const rarity = Math.min(Math.round((1 - metrics.binomial.p) * maxScore), maxScore);
46
- return [rarity, maxScore];
50
+ return [
51
+ rarity,
52
+ maxScore
53
+ ];
47
54
  });
48
55
  });
49
- const total = scores.reduce(([a, b], [c, d]) => [a + c, b + d], [0, 0]);
56
+ const total = scores.reduce(([a, b], [c, d]) => [
57
+ a + c,
58
+ b + d
59
+ ], [
60
+ 0,
61
+ 0
62
+ ]);
50
63
  return (0, import_crypto_nft_score_model.normalize)(total, maxScore);
51
- };
64
+ }, "scoreIndividualAttributes");
52
65
 
53
66
  // src/lib/rating/criteria/scoring/metadata/scoreTotalAttributes.ts
54
67
  var import_crypto_nft_score_model2 = require("@xyo-network/crypto-nft-score-model");
55
68
  var maxScore2 = 10;
56
69
  var defaultMu = 0.15;
57
70
  var defaultSigma = 0.1;
58
- var gaussianProbabilityDensity = (x, mu = defaultMu, sigma = defaultSigma) => {
71
+ var gaussianProbabilityDensity = /* @__PURE__ */ __name((x, mu = defaultMu, sigma = defaultSigma) => {
59
72
  const sqrtTwoPi = Math.sqrt(2 * Math.PI);
60
73
  const denominator = sigma * sqrtTwoPi;
61
74
  const power = -0.5 * Math.pow((x - mu) / sigma, 2);
62
75
  return 1 / denominator * Math.exp(power);
63
- };
76
+ }, "gaussianProbabilityDensity");
64
77
  var maxProbabilityDensity = gaussianProbabilityDensity(defaultMu);
65
- var scoreTotalAttributes = (info) => {
78
+ var scoreTotalAttributes = /* @__PURE__ */ __name((info) => {
66
79
  const { attributes } = info.metrics.metadata;
67
80
  const jointProbability = Object.entries(attributes).reduce((acc, [_trait, { metrics }]) => {
68
81
  return acc * metrics.binomial.p;
69
82
  }, 1);
70
83
  const probabilityDensity = gaussianProbabilityDensity(jointProbability);
71
- const score = [probabilityDensity, maxProbabilityDensity];
84
+ const score = [
85
+ probabilityDensity,
86
+ maxProbabilityDensity
87
+ ];
72
88
  return (0, import_crypto_nft_score_model2.normalize)(score, maxScore2);
73
- };
89
+ }, "scoreTotalAttributes");
74
90
 
75
91
  // src/lib/rating/criteria/scoring/metadata/metadata.ts
76
92
  var attributeScoringCriteria = {
77
- "Metadata Attributes Individual": { score: scoreIndividualAttributes, weight: 2 },
78
- "Metadata Attributes Total": { score: scoreTotalAttributes, weight: 2 }
93
+ "Metadata Attributes Individual": {
94
+ score: scoreIndividualAttributes,
95
+ weight: 2
96
+ },
97
+ "Metadata Attributes Total": {
98
+ score: scoreTotalAttributes,
99
+ weight: 2
100
+ }
79
101
  };
80
102
  var scoreMetadata = {
81
103
  ...attributeScoringCriteria
@@ -87,76 +109,103 @@ var median = 810308398217352e-7;
87
109
  var defaultMu2 = Math.log(median);
88
110
  var defaultSigma2 = 3;
89
111
  var mode = Math.exp(defaultMu2 - Math.pow(defaultSigma2, 2));
90
- var logNormalProbabilityDensity = (x, mu = defaultMu2, sigma = defaultSigma2) => {
112
+ var logNormalProbabilityDensity = /* @__PURE__ */ __name((x, mu = defaultMu2, sigma = defaultSigma2) => {
91
113
  if (x <= 0) return 0;
92
114
  const logX = Math.log(x);
93
115
  return 1 / (x * sigma * Math.sqrt(2 * Math.PI)) * Math.exp(-0.5 * Math.pow((logX - mu) / sigma, 2));
94
- };
116
+ }, "logNormalProbabilityDensity");
95
117
  var maxProbabilityDensity2 = logNormalProbabilityDensity(mode);
96
118
  var maxScore3 = 10;
97
- var scoreTotal = (nft) => {
119
+ var scoreTotal = /* @__PURE__ */ __name((nft) => {
98
120
  const density = logNormalProbabilityDensity(nft.total);
99
- const score = [density, maxProbabilityDensity2];
121
+ const score = [
122
+ density,
123
+ maxProbabilityDensity2
124
+ ];
100
125
  return (0, import_crypto_nft_score_model3.normalize)(score, maxScore3);
101
- };
126
+ }, "scoreTotal");
102
127
 
103
128
  // src/lib/rating/criteria/index.ts
104
129
  var scoringCriteria = {
105
130
  ...scoreMetadata,
106
- Total: { score: scoreTotal, weight: 2 }
131
+ Total: {
132
+ score: scoreTotal,
133
+ weight: 2
134
+ }
107
135
  };
108
136
 
109
137
  // src/lib/rating/analyzeNftCollection.ts
110
- var analyzeNftCollection = async (nft) => {
111
- const result = Object.fromEntries(
112
- await Promise.all(
113
- Object.entries(scoringCriteria).map(async ([key, { score, weight }]) => {
114
- const rawScore = await score(nft);
115
- const weighted = rawScore.map((v) => v * weight);
116
- return [key, weighted];
117
- })
118
- )
119
- );
138
+ var analyzeNftCollection = /* @__PURE__ */ __name(async (nft) => {
139
+ const result = Object.fromEntries(await Promise.all(Object.entries(scoringCriteria).map(async ([key, { score, weight }]) => {
140
+ const rawScore = await score(nft);
141
+ const weighted = rawScore.map((v) => v * weight);
142
+ return [
143
+ key,
144
+ weighted
145
+ ];
146
+ })));
120
147
  return result;
121
- };
148
+ }, "analyzeNftCollection");
122
149
 
123
150
  // src/Diviner.ts
124
- var toNftCollectionScore = (nftCollectionInfo, scores) => {
151
+ var toNftCollectionScore = /* @__PURE__ */ __name((nftCollectionInfo, scores) => {
125
152
  const { name, symbol, address, chainId, type } = nftCollectionInfo;
126
- const metadata = { address, chainId, name, symbol, type };
127
- return { ...metadata, schema: import_crypto_nft_collection_payload_plugin.NftCollectionScoreSchema, scores };
128
- };
153
+ const metadata = {
154
+ address,
155
+ chainId,
156
+ name,
157
+ symbol,
158
+ type
159
+ };
160
+ return {
161
+ ...metadata,
162
+ schema: import_crypto_nft_collection_payload_plugin.NftCollectionScoreSchema,
163
+ scores
164
+ };
165
+ }, "toNftCollectionScore");
129
166
  var NftCollectionScoreDiviner = class extends import_diviner_abstract.AbstractDiviner {
130
- static configSchemas = [...super.configSchemas, import_crypto_nft_collection_payload_plugin.NftCollectionScoreDivinerConfigSchema];
167
+ static {
168
+ __name(this, "NftCollectionScoreDiviner");
169
+ }
170
+ static configSchemas = [
171
+ ...super.configSchemas,
172
+ import_crypto_nft_collection_payload_plugin.NftCollectionScoreDivinerConfigSchema
173
+ ];
131
174
  static defaultConfigSchema = import_crypto_nft_collection_payload_plugin.NftCollectionScoreDivinerConfigSchema;
132
- divineHandler = async (payloads) => {
175
+ divineHandler = /* @__PURE__ */ __name(async (payloads) => {
133
176
  const nftCollectionInfos = payloads?.filter(import_crypto_nft_collection_payload_plugin.isNftCollectionInfo) ?? [];
134
- const results = await Promise.all(
135
- nftCollectionInfos.map(async (nftCollectionInfo) => {
136
- const [score, sourceHash] = await Promise.all([
137
- // Get score
138
- toNftCollectionScore(nftCollectionInfo, await analyzeNftCollection(nftCollectionInfo)),
139
- // Hash sources
140
- import_payload_builder.PayloadBuilder.dataHash(nftCollectionInfo)
141
- ]);
142
- return { ...score, schema: import_crypto_nft_collection_payload_plugin.NftCollectionScoreSchema, sources: [sourceHash] };
143
- })
144
- );
177
+ const results = await Promise.all(nftCollectionInfos.map(async (nftCollectionInfo) => {
178
+ const [score, sourceHash] = await Promise.all([
179
+ // Get score
180
+ toNftCollectionScore(nftCollectionInfo, await analyzeNftCollection(nftCollectionInfo)),
181
+ // Hash sources
182
+ import_payload_builder.PayloadBuilder.dataHash(nftCollectionInfo)
183
+ ]);
184
+ return {
185
+ ...score,
186
+ schema: import_crypto_nft_collection_payload_plugin.NftCollectionScoreSchema,
187
+ sources: [
188
+ sourceHash
189
+ ]
190
+ };
191
+ }));
145
192
  return results;
146
- };
193
+ }, "divineHandler");
147
194
  };
148
195
 
149
196
  // src/Plugin.ts
150
197
  var import_crypto_nft_payload_plugin = require("@xyo-network/crypto-nft-payload-plugin");
151
198
  var import_payload_model = require("@xyo-network/payload-model");
152
199
  var import_payloadset_plugin = require("@xyo-network/payloadset-plugin");
153
- var NftCollectionScoreDivinerPlugin = () => (0, import_payloadset_plugin.createPayloadSetDivinerPlugin)(
154
- { required: { [import_crypto_nft_payload_plugin.NftSchema]: 1 }, schema: import_payload_model.PayloadSetSchema },
155
- {
156
- diviner: async (params) => {
157
- const result = await NftCollectionScoreDiviner.create(params);
158
- return result;
159
- }
160
- }
161
- );
200
+ var NftCollectionScoreDivinerPlugin = /* @__PURE__ */ __name(() => (0, import_payloadset_plugin.createPayloadSetDivinerPlugin)({
201
+ required: {
202
+ [import_crypto_nft_payload_plugin.NftSchema]: 1
203
+ },
204
+ schema: import_payload_model.PayloadSetSchema
205
+ }, {
206
+ diviner: /* @__PURE__ */ __name(async (params) => {
207
+ const result = await NftCollectionScoreDiviner.create(params);
208
+ return result;
209
+ }, "diviner")
210
+ }), "NftCollectionScoreDivinerPlugin");
162
211
  //# sourceMappingURL=index.cjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/index.ts","../../src/Diviner.ts","../../src/lib/rating/criteria/scoring/metadata/scoreIndividualAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/scoreTotalAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/metadata.ts","../../src/lib/rating/criteria/scoring/total.ts","../../src/lib/rating/criteria/index.ts","../../src/lib/rating/analyzeNftCollection.ts","../../src/Plugin.ts"],"sourcesContent":["export * from './Diviner.js'\nexport * from './lib/index.js'\n// eslint-disable-next-line import/no-default-export\nexport { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.js'\n","import {\n isNftCollectionInfo,\n NftCollectionInfo,\n NftCollectionMetadata,\n NftCollectionScore,\n NftCollectionScoreDivinerConfig,\n NftCollectionScoreDivinerConfigSchema,\n NftCollectionScoreSchema,\n} from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { DivinerParams } from '@xyo-network/diviner-model'\nimport { AnyConfigSchema } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload, Schema } from '@xyo-network/payload-model'\n\nimport { analyzeNftCollection, NftCollectionAnalysis } from './lib/index.js'\n\nexport type NftCollectionScoreDivinerParams = DivinerParams<AnyConfigSchema<NftCollectionScoreDivinerConfig>>\n\nconst toNftCollectionScore = (nftCollectionInfo: NftCollectionInfo, scores: NftCollectionAnalysis): NftCollectionScore => {\n const { name, symbol, address, chainId, type } = nftCollectionInfo\n const metadata: NftCollectionMetadata = { address, chainId, name, symbol, type }\n return { ...metadata, schema: NftCollectionScoreSchema, scores }\n}\n\nexport class NftCollectionScoreDiviner<\n TParams extends NftCollectionScoreDivinerParams = NftCollectionScoreDivinerParams,\n> extends AbstractDiviner<TParams> {\n static override readonly configSchemas: Schema[] = [...super.configSchemas, NftCollectionScoreDivinerConfigSchema]\n static override readonly defaultConfigSchema: Schema = NftCollectionScoreDivinerConfigSchema\n\n protected override divineHandler = async (payloads?: Payload[]): Promise<Payload[]> => {\n const nftCollectionInfos = payloads?.filter(isNftCollectionInfo) ?? []\n const results = await Promise.all(\n nftCollectionInfos.map<Promise<NftCollectionScore>>(async (nftCollectionInfo) => {\n const [score, sourceHash] = await Promise.all([\n // Get score\n toNftCollectionScore(nftCollectionInfo, await analyzeNftCollection(nftCollectionInfo)),\n // Hash sources\n PayloadBuilder.dataHash(nftCollectionInfo),\n ])\n return { ...score, schema: NftCollectionScoreSchema, sources: [sourceHash] } as NftCollectionScore\n }),\n )\n return results\n }\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\nexport const scoreIndividualAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n const entries = Object.entries(attributes)\n if (entries.length === 0) return [0, maxScore]\n const scores = Object.entries(attributes).flatMap(([_trait, { values }]) => {\n return Object.entries(values).map<Score>(([_traitValue, metrics]) => {\n const rarity = Math.min(Math.round((1 - metrics.binomial.p) * maxScore), maxScore)\n return [rarity, maxScore]\n })\n })\n // eslint-disable-next-line unicorn/no-array-reduce\n const total = scores.reduce<Score>(([a, b], [c, d]) => [a + c, b + d], [0, 0])\n return normalize(total, maxScore)\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\n/**\n * Mean: What value is the distribution centered around\n */\nconst defaultMu = 0.15\n\n/**\n * Standard Deviation: How spread out is the distribution\n */\nconst defaultSigma = 0.1\n\n/**\n * Calculates the Gaussian probability density\n * @param x\n * @param mu Mean\n * @param sigma Standard Deviation\n * @returns\n */\nconst gaussianProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n const sqrtTwoPi = Math.sqrt(2 * Math.PI)\n const denominator = sigma * sqrtTwoPi\n const power = -0.5 * Math.pow((x - mu) / sigma, 2)\n return (1 / denominator) * Math.exp(power)\n}\n\n/**\n * For a Gaussian distribution, the peak of the distribution is the mean\n */\nconst maxProbabilityDensity = gaussianProbabilityDensity(defaultMu)\n\n/**\n * We're working on some assumptions here:\n *\n * - If you have a 100% chance of getting a trait, everyone get's a trophy\n * - If you have a 50% chance of getting a trait, it's not rare\n * - If you have a 0% chance of getting a trait, it's not fun\n *\n * So we're looking for something Pareto-ish (somewhere between\n * 80/20 or 90/10) as that's a good & sustainable model for the\n * distribution of many traits in real life.\n * However, we also don't want to maximally reward collections\n * that have a lot of single attributes distributed uniformly\n * (basically a 0% trait probably) as that's perfectly entropic\n * but not very interesting (some overlap is desirable).\n * So we're using a Gaussian distribution to model the\n * probability density of the joint probability of all traits\n * centered around 15%.\n * @param info\n * @returns\n */\nexport const scoreTotalAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n // This has somewhat of a filtering function by causing anything with 100% probability to\n // add no value to the end score\n const jointProbability = Object.entries(attributes).reduce((acc, [_trait, { metrics }]) => {\n return acc * metrics.binomial.p\n }, 1)\n const probabilityDensity = gaussianProbabilityDensity(jointProbability)\n const score: Score = [probabilityDensity, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreIndividualAttributes } from './scoreIndividualAttributes.js'\nimport { scoreTotalAttributes } from './scoreTotalAttributes.js'\n\nexport const attributeScoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n 'Metadata Attributes Individual': { score: scoreIndividualAttributes, weight: 2 },\n 'Metadata Attributes Total': { score: scoreTotalAttributes, weight: 2 },\n}\n\nexport const scoreMetadata: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...attributeScoringCriteria,\n}\n","import { NftCollectionCount } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\n/**\n * This \"magic\" value was obtained using Solver in Excel\n * to find the median, with mu/sigma fixed, which maximizes\n * the distribution (the mode for lognormal) at 10,000\n */\nconst median = 81_030_839.821_735_2\nconst defaultMu = Math.log(median)\nconst defaultSigma = 3\nconst mode = Math.exp(defaultMu - Math.pow(defaultSigma, 2))\n/**\n * Calculates the log-normal probability density\n * @param x the value at which you want to calculate the probability density\n * @param mu mean of the associated normal distribution\n * @param sigma standard deviation of the associated normal distribution\n * @returns\n */\nconst logNormalProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n if (x <= 0) return 0\n const logX = Math.log(x)\n return (1 / (x * sigma * Math.sqrt(2 * Math.PI))) * Math.exp(-0.5 * Math.pow((logX - mu) / sigma, 2))\n}\n\n/**\n * For a lognormal distribution, the peak of the distribution is the mode\n */\nconst maxProbabilityDensity = logNormalProbabilityDensity(mode)\n\nconst maxScore = 10\n\n/**\n * We're working on some assumptions here:\n * - If there's < 1000 NFTs in your collection it starts becoming too niche\n * - If there's > 20,000 NFTs in your collection it starts becoming too broad\n * So there's a sweet spot somewhere between 2000 and 10,000\n * where a collection has enough NFTs to be interesting, but\n * not so many that it's teetering on a diluted money grab.\n * To model that we're using a log-normal distribution optimized\n * to maximally reward collections in the aforementioned range\n * @param nft\n * @returns\n */\nexport const scoreTotal = (nft: NftCollectionCount): Score => {\n const density = logNormalProbabilityDensity(nft.total)\n const score: Score = [density, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreMetadata, scoreTotal } from './scoring/index.js'\n\nexport const scoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...scoreMetadata,\n Total: { score: scoreTotal, weight: 2 },\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { Score } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoringCriteria } from './criteria/index.js'\n\nexport type ScoringCriteriaKey = keyof typeof scoringCriteria & PropertyKey\n\nexport type NftCollectionAnalysis = {\n [key in ScoringCriteriaKey]: Score\n}\n\nexport const analyzeNftCollection = async (\n /**\n * The NFT to evaluate\n */\n nft: NftCollectionInfo,\n): Promise<NftCollectionAnalysis> => {\n const result = Object.fromEntries(\n await Promise.all(\n Object.entries(scoringCriteria).map(async ([key, { score, weight }]) => {\n const rawScore = await score(nft)\n const weighted = rawScore.map((v) => v * weight) as Score\n return [key, weighted] as const\n }),\n ),\n ) as NftCollectionAnalysis\n return result\n}\n","import { NftSchema } from '@xyo-network/crypto-nft-payload-plugin'\nimport { PayloadSetSchema } from '@xyo-network/payload-model'\nimport { createPayloadSetDivinerPlugin } from '@xyo-network/payloadset-plugin'\n\nimport { NftCollectionScoreDiviner } from './Diviner.js'\n\nexport const NftCollectionScoreDivinerPlugin = () =>\n createPayloadSetDivinerPlugin<NftCollectionScoreDiviner>(\n { required: { [NftSchema]: 1 }, schema: PayloadSetSchema },\n {\n diviner: async (params) => {\n const result = await NftCollectionScoreDiviner.create(params)\n return result\n },\n },\n )\n"],"mappings":"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","names":["import_crypto_nft_score_model","maxScore","import_crypto_nft_score_model","defaultMu","defaultSigma","maxProbabilityDensity","maxScore"]}
1
+ {"version":3,"sources":["../../src/index.ts","../../src/Diviner.ts","../../src/lib/rating/criteria/scoring/metadata/scoreIndividualAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/scoreTotalAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/metadata.ts","../../src/lib/rating/criteria/scoring/total.ts","../../src/lib/rating/criteria/index.ts","../../src/lib/rating/analyzeNftCollection.ts","../../src/Plugin.ts"],"sourcesContent":["export * from './Diviner.ts'\nexport * from './lib/index.ts'\n// eslint-disable-next-line import/no-default-export\nexport { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.ts'\n","import {\n isNftCollectionInfo,\n NftCollectionInfo,\n NftCollectionMetadata,\n NftCollectionScore,\n NftCollectionScoreDivinerConfig,\n NftCollectionScoreDivinerConfigSchema,\n NftCollectionScoreSchema,\n} from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { DivinerParams } from '@xyo-network/diviner-model'\nimport { AnyConfigSchema } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload, Schema } from '@xyo-network/payload-model'\n\nimport { analyzeNftCollection, NftCollectionAnalysis } from './lib/index.ts'\n\nexport type NftCollectionScoreDivinerParams = DivinerParams<AnyConfigSchema<NftCollectionScoreDivinerConfig>>\n\nconst toNftCollectionScore = (nftCollectionInfo: NftCollectionInfo, scores: NftCollectionAnalysis): NftCollectionScore => {\n const { name, symbol, address, chainId, type } = nftCollectionInfo\n const metadata: NftCollectionMetadata = { address, chainId, name, symbol, type }\n return { ...metadata, schema: NftCollectionScoreSchema, scores }\n}\n\nexport class NftCollectionScoreDiviner<\n TParams extends NftCollectionScoreDivinerParams = NftCollectionScoreDivinerParams,\n> extends AbstractDiviner<TParams> {\n static override readonly configSchemas: Schema[] = [...super.configSchemas, NftCollectionScoreDivinerConfigSchema]\n static override readonly defaultConfigSchema: Schema = NftCollectionScoreDivinerConfigSchema\n\n protected override divineHandler = async (payloads?: Payload[]): Promise<Payload[]> => {\n const nftCollectionInfos = payloads?.filter(isNftCollectionInfo) ?? []\n const results = await Promise.all(\n nftCollectionInfos.map<Promise<NftCollectionScore>>(async (nftCollectionInfo) => {\n const [score, sourceHash] = await Promise.all([\n // Get score\n toNftCollectionScore(nftCollectionInfo, await analyzeNftCollection(nftCollectionInfo)),\n // Hash sources\n PayloadBuilder.dataHash(nftCollectionInfo),\n ])\n return { ...score, schema: NftCollectionScoreSchema, sources: [sourceHash] } as NftCollectionScore\n }),\n )\n return results\n }\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\nexport const scoreIndividualAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n const entries = Object.entries(attributes)\n if (entries.length === 0) return [0, maxScore]\n const scores = Object.entries(attributes).flatMap(([_trait, { values }]) => {\n return Object.entries(values).map<Score>(([_traitValue, metrics]) => {\n const rarity = Math.min(Math.round((1 - metrics.binomial.p) * maxScore), maxScore)\n return [rarity, maxScore]\n })\n })\n // eslint-disable-next-line unicorn/no-array-reduce\n const total = scores.reduce<Score>(([a, b], [c, d]) => [a + c, b + d], [0, 0])\n return normalize(total, maxScore)\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\n/**\n * Mean: What value is the distribution centered around\n */\nconst defaultMu = 0.15\n\n/**\n * Standard Deviation: How spread out is the distribution\n */\nconst defaultSigma = 0.1\n\n/**\n * Calculates the Gaussian probability density\n * @param x\n * @param mu Mean\n * @param sigma Standard Deviation\n * @returns\n */\nconst gaussianProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n const sqrtTwoPi = Math.sqrt(2 * Math.PI)\n const denominator = sigma * sqrtTwoPi\n const power = -0.5 * Math.pow((x - mu) / sigma, 2)\n return (1 / denominator) * Math.exp(power)\n}\n\n/**\n * For a Gaussian distribution, the peak of the distribution is the mean\n */\nconst maxProbabilityDensity = gaussianProbabilityDensity(defaultMu)\n\n/**\n * We're working on some assumptions here:\n *\n * - If you have a 100% chance of getting a trait, everyone get's a trophy\n * - If you have a 50% chance of getting a trait, it's not rare\n * - If you have a 0% chance of getting a trait, it's not fun\n *\n * So we're looking for something Pareto-ish (somewhere between\n * 80/20 or 90/10) as that's a good & sustainable model for the\n * distribution of many traits in real life.\n * However, we also don't want to maximally reward collections\n * that have a lot of single attributes distributed uniformly\n * (basically a 0% trait probably) as that's perfectly entropic\n * but not very interesting (some overlap is desirable).\n * So we're using a Gaussian distribution to model the\n * probability density of the joint probability of all traits\n * centered around 15%.\n * @param info\n * @returns\n */\nexport const scoreTotalAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n // This has somewhat of a filtering function by causing anything with 100% probability to\n // add no value to the end score\n const jointProbability = Object.entries(attributes).reduce((acc, [_trait, { metrics }]) => {\n return acc * metrics.binomial.p\n }, 1)\n const probabilityDensity = gaussianProbabilityDensity(jointProbability)\n const score: Score = [probabilityDensity, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreIndividualAttributes } from './scoreIndividualAttributes.ts'\nimport { scoreTotalAttributes } from './scoreTotalAttributes.ts'\n\nexport const attributeScoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n 'Metadata Attributes Individual': { score: scoreIndividualAttributes, weight: 2 },\n 'Metadata Attributes Total': { score: scoreTotalAttributes, weight: 2 },\n}\n\nexport const scoreMetadata: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...attributeScoringCriteria,\n}\n","import { NftCollectionCount } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\n/**\n * This \"magic\" value was obtained using Solver in Excel\n * to find the median, with mu/sigma fixed, which maximizes\n * the distribution (the mode for lognormal) at 10,000\n */\nconst median = 81_030_839.821_735_2\nconst defaultMu = Math.log(median)\nconst defaultSigma = 3\nconst mode = Math.exp(defaultMu - Math.pow(defaultSigma, 2))\n/**\n * Calculates the log-normal probability density\n * @param x the value at which you want to calculate the probability density\n * @param mu mean of the associated normal distribution\n * @param sigma standard deviation of the associated normal distribution\n * @returns\n */\nconst logNormalProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n if (x <= 0) return 0\n const logX = Math.log(x)\n return (1 / (x * sigma * Math.sqrt(2 * Math.PI))) * Math.exp(-0.5 * Math.pow((logX - mu) / sigma, 2))\n}\n\n/**\n * For a lognormal distribution, the peak of the distribution is the mode\n */\nconst maxProbabilityDensity = logNormalProbabilityDensity(mode)\n\nconst maxScore = 10\n\n/**\n * We're working on some assumptions here:\n * - If there's < 1000 NFTs in your collection it starts becoming too niche\n * - If there's > 20,000 NFTs in your collection it starts becoming too broad\n * So there's a sweet spot somewhere between 2000 and 10,000\n * where a collection has enough NFTs to be interesting, but\n * not so many that it's teetering on a diluted money grab.\n * To model that we're using a log-normal distribution optimized\n * to maximally reward collections in the aforementioned range\n * @param nft\n * @returns\n */\nexport const scoreTotal = (nft: NftCollectionCount): Score => {\n const density = logNormalProbabilityDensity(nft.total)\n const score: Score = [density, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreMetadata, scoreTotal } from './scoring/index.ts'\n\nexport const scoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...scoreMetadata,\n Total: { score: scoreTotal, weight: 2 },\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { Score } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoringCriteria } from './criteria/index.ts'\n\nexport type ScoringCriteriaKey = keyof typeof scoringCriteria & PropertyKey\n\nexport type NftCollectionAnalysis = {\n [key in ScoringCriteriaKey]: Score\n}\n\nexport const analyzeNftCollection = async (\n /**\n * The NFT to evaluate\n */\n nft: NftCollectionInfo,\n): Promise<NftCollectionAnalysis> => {\n const result = Object.fromEntries(\n await Promise.all(\n Object.entries(scoringCriteria).map(async ([key, { score, weight }]) => {\n const rawScore = await score(nft)\n const weighted = rawScore.map(v => v * weight) as Score\n return [key, weighted] as const\n }),\n ),\n ) as NftCollectionAnalysis\n return result\n}\n","import { NftSchema } from '@xyo-network/crypto-nft-payload-plugin'\nimport { PayloadSetSchema } from '@xyo-network/payload-model'\nimport { createPayloadSetDivinerPlugin } from '@xyo-network/payloadset-plugin'\n\nimport { NftCollectionScoreDiviner } from './Diviner.ts'\n\nexport const NftCollectionScoreDivinerPlugin = () =>\n createPayloadSetDivinerPlugin<NftCollectionScoreDiviner>(\n { required: { [NftSchema]: 1 }, schema: PayloadSetSchema },\n {\n diviner: async (params) => {\n const result = await NftCollectionScoreDiviner.create(params)\n return result\n },\n },\n )\n"],"mappings":"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","names":["maxScore","scoreIndividualAttributes","info","attributes","metrics","metadata","entries","Object","length","scores","flatMap","_trait","values","map","_traitValue","rarity","Math","min","round","binomial","p","total","reduce","a","b","c","d","normalize","import_crypto_nft_score_model","maxScore","defaultMu","defaultSigma","gaussianProbabilityDensity","x","mu","sigma","sqrtTwoPi","Math","sqrt","PI","denominator","power","pow","exp","maxProbabilityDensity","scoreTotalAttributes","info","attributes","metrics","metadata","jointProbability","Object","entries","reduce","acc","_trait","binomial","p","probabilityDensity","score","normalize","attributeScoringCriteria","score","scoreIndividualAttributes","weight","scoreTotalAttributes","scoreMetadata","import_crypto_nft_score_model","median","defaultMu","Math","log","defaultSigma","mode","exp","pow","logNormalProbabilityDensity","x","mu","sigma","logX","sqrt","PI","maxProbabilityDensity","maxScore","scoreTotal","nft","density","total","score","normalize","scoringCriteria","scoreMetadata","Total","score","scoreTotal","weight","analyzeNftCollection","nft","result","Object","fromEntries","Promise","all","entries","scoringCriteria","map","key","score","weight","rawScore","weighted","v","toNftCollectionScore","nftCollectionInfo","scores","name","symbol","address","chainId","type","metadata","schema","NftCollectionScoreSchema","NftCollectionScoreDiviner","AbstractDiviner","configSchemas","NftCollectionScoreDivinerConfigSchema","defaultConfigSchema","divineHandler","payloads","nftCollectionInfos","filter","isNftCollectionInfo","results","Promise","all","map","score","sourceHash","analyzeNftCollection","PayloadBuilder","dataHash","sources","NftCollectionScoreDivinerPlugin","createPayloadSetDivinerPlugin","required","NftSchema","schema","PayloadSetSchema","diviner","params","result","NftCollectionScoreDiviner","create"]}
@@ -1,4 +1,4 @@
1
- export * from './Diviner.js';
2
- export * from './lib/index.js';
3
- export { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.js';
1
+ export * from './Diviner.ts';
2
+ export * from './lib/index.ts';
3
+ export { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.ts';
4
4
  //# sourceMappingURL=index.d.ts.map
@@ -1,4 +1,4 @@
1
- export * from './Diviner.js';
2
- export * from './lib/index.js';
3
- export { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.js';
1
+ export * from './Diviner.ts';
2
+ export * from './lib/index.ts';
3
+ export { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.ts';
4
4
  //# sourceMappingURL=index.d.ts.map
@@ -1,4 +1,4 @@
1
- export * from './Diviner.js';
2
- export * from './lib/index.js';
3
- export { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.js';
1
+ export * from './Diviner.ts';
2
+ export * from './lib/index.ts';
3
+ export { NftCollectionScoreDivinerPlugin as default, NftCollectionScoreDivinerPlugin } from './Plugin.ts';
4
4
  //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1,190 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
3
+
4
+ // src/Diviner.ts
5
+ import { isNftCollectionInfo, NftCollectionScoreDivinerConfigSchema, NftCollectionScoreSchema } from "@xyo-network/crypto-nft-collection-payload-plugin";
6
+ import { AbstractDiviner } from "@xyo-network/diviner-abstract";
7
+ import { PayloadBuilder } from "@xyo-network/payload-builder";
8
+
9
+ // src/lib/rating/criteria/scoring/metadata/scoreIndividualAttributes.ts
10
+ import { normalize } from "@xyo-network/crypto-nft-score-model";
11
+ var maxScore = 10;
12
+ var scoreIndividualAttributes = /* @__PURE__ */ __name((info) => {
13
+ const { attributes } = info.metrics.metadata;
14
+ const entries = Object.entries(attributes);
15
+ if (entries.length === 0) return [
16
+ 0,
17
+ maxScore
18
+ ];
19
+ const scores = Object.entries(attributes).flatMap(([_trait, { values }]) => {
20
+ return Object.entries(values).map(([_traitValue, metrics]) => {
21
+ const rarity = Math.min(Math.round((1 - metrics.binomial.p) * maxScore), maxScore);
22
+ return [
23
+ rarity,
24
+ maxScore
25
+ ];
26
+ });
27
+ });
28
+ const total = scores.reduce(([a, b], [c, d]) => [
29
+ a + c,
30
+ b + d
31
+ ], [
32
+ 0,
33
+ 0
34
+ ]);
35
+ return normalize(total, maxScore);
36
+ }, "scoreIndividualAttributes");
37
+
38
+ // src/lib/rating/criteria/scoring/metadata/scoreTotalAttributes.ts
39
+ import { normalize as normalize2 } from "@xyo-network/crypto-nft-score-model";
40
+ var maxScore2 = 10;
41
+ var defaultMu = 0.15;
42
+ var defaultSigma = 0.1;
43
+ var gaussianProbabilityDensity = /* @__PURE__ */ __name((x, mu = defaultMu, sigma = defaultSigma) => {
44
+ const sqrtTwoPi = Math.sqrt(2 * Math.PI);
45
+ const denominator = sigma * sqrtTwoPi;
46
+ const power = -0.5 * Math.pow((x - mu) / sigma, 2);
47
+ return 1 / denominator * Math.exp(power);
48
+ }, "gaussianProbabilityDensity");
49
+ var maxProbabilityDensity = gaussianProbabilityDensity(defaultMu);
50
+ var scoreTotalAttributes = /* @__PURE__ */ __name((info) => {
51
+ const { attributes } = info.metrics.metadata;
52
+ const jointProbability = Object.entries(attributes).reduce((acc, [_trait, { metrics }]) => {
53
+ return acc * metrics.binomial.p;
54
+ }, 1);
55
+ const probabilityDensity = gaussianProbabilityDensity(jointProbability);
56
+ const score = [
57
+ probabilityDensity,
58
+ maxProbabilityDensity
59
+ ];
60
+ return normalize2(score, maxScore2);
61
+ }, "scoreTotalAttributes");
62
+
63
+ // src/lib/rating/criteria/scoring/metadata/metadata.ts
64
+ var attributeScoringCriteria = {
65
+ "Metadata Attributes Individual": {
66
+ score: scoreIndividualAttributes,
67
+ weight: 2
68
+ },
69
+ "Metadata Attributes Total": {
70
+ score: scoreTotalAttributes,
71
+ weight: 2
72
+ }
73
+ };
74
+ var scoreMetadata = {
75
+ ...attributeScoringCriteria
76
+ };
77
+
78
+ // src/lib/rating/criteria/scoring/total.ts
79
+ import { normalize as normalize3 } from "@xyo-network/crypto-nft-score-model";
80
+ var median = 810308398217352e-7;
81
+ var defaultMu2 = Math.log(median);
82
+ var defaultSigma2 = 3;
83
+ var mode = Math.exp(defaultMu2 - Math.pow(defaultSigma2, 2));
84
+ var logNormalProbabilityDensity = /* @__PURE__ */ __name((x, mu = defaultMu2, sigma = defaultSigma2) => {
85
+ if (x <= 0) return 0;
86
+ const logX = Math.log(x);
87
+ return 1 / (x * sigma * Math.sqrt(2 * Math.PI)) * Math.exp(-0.5 * Math.pow((logX - mu) / sigma, 2));
88
+ }, "logNormalProbabilityDensity");
89
+ var maxProbabilityDensity2 = logNormalProbabilityDensity(mode);
90
+ var maxScore3 = 10;
91
+ var scoreTotal = /* @__PURE__ */ __name((nft) => {
92
+ const density = logNormalProbabilityDensity(nft.total);
93
+ const score = [
94
+ density,
95
+ maxProbabilityDensity2
96
+ ];
97
+ return normalize3(score, maxScore3);
98
+ }, "scoreTotal");
99
+
100
+ // src/lib/rating/criteria/index.ts
101
+ var scoringCriteria = {
102
+ ...scoreMetadata,
103
+ Total: {
104
+ score: scoreTotal,
105
+ weight: 2
106
+ }
107
+ };
108
+
109
+ // src/lib/rating/analyzeNftCollection.ts
110
+ var analyzeNftCollection = /* @__PURE__ */ __name(async (nft) => {
111
+ const result = Object.fromEntries(await Promise.all(Object.entries(scoringCriteria).map(async ([key, { score, weight }]) => {
112
+ const rawScore = await score(nft);
113
+ const weighted = rawScore.map((v) => v * weight);
114
+ return [
115
+ key,
116
+ weighted
117
+ ];
118
+ })));
119
+ return result;
120
+ }, "analyzeNftCollection");
121
+
122
+ // src/Diviner.ts
123
+ var toNftCollectionScore = /* @__PURE__ */ __name((nftCollectionInfo, scores) => {
124
+ const { name, symbol, address, chainId, type } = nftCollectionInfo;
125
+ const metadata = {
126
+ address,
127
+ chainId,
128
+ name,
129
+ symbol,
130
+ type
131
+ };
132
+ return {
133
+ ...metadata,
134
+ schema: NftCollectionScoreSchema,
135
+ scores
136
+ };
137
+ }, "toNftCollectionScore");
138
+ var NftCollectionScoreDiviner = class extends AbstractDiviner {
139
+ static {
140
+ __name(this, "NftCollectionScoreDiviner");
141
+ }
142
+ static configSchemas = [
143
+ ...super.configSchemas,
144
+ NftCollectionScoreDivinerConfigSchema
145
+ ];
146
+ static defaultConfigSchema = NftCollectionScoreDivinerConfigSchema;
147
+ divineHandler = /* @__PURE__ */ __name(async (payloads) => {
148
+ const nftCollectionInfos = payloads?.filter(isNftCollectionInfo) ?? [];
149
+ const results = await Promise.all(nftCollectionInfos.map(async (nftCollectionInfo) => {
150
+ const [score, sourceHash] = await Promise.all([
151
+ // Get score
152
+ toNftCollectionScore(nftCollectionInfo, await analyzeNftCollection(nftCollectionInfo)),
153
+ // Hash sources
154
+ PayloadBuilder.dataHash(nftCollectionInfo)
155
+ ]);
156
+ return {
157
+ ...score,
158
+ schema: NftCollectionScoreSchema,
159
+ sources: [
160
+ sourceHash
161
+ ]
162
+ };
163
+ }));
164
+ return results;
165
+ }, "divineHandler");
166
+ };
167
+
168
+ // src/Plugin.ts
169
+ import { NftSchema } from "@xyo-network/crypto-nft-payload-plugin";
170
+ import { PayloadSetSchema } from "@xyo-network/payload-model";
171
+ import { createPayloadSetDivinerPlugin } from "@xyo-network/payloadset-plugin";
172
+ var NftCollectionScoreDivinerPlugin = /* @__PURE__ */ __name(() => createPayloadSetDivinerPlugin({
173
+ required: {
174
+ [NftSchema]: 1
175
+ },
176
+ schema: PayloadSetSchema
177
+ }, {
178
+ diviner: /* @__PURE__ */ __name(async (params) => {
179
+ const result = await NftCollectionScoreDiviner.create(params);
180
+ return result;
181
+ }, "diviner")
182
+ }), "NftCollectionScoreDivinerPlugin");
183
+ export {
184
+ NftCollectionScoreDiviner,
185
+ NftCollectionScoreDivinerPlugin,
186
+ analyzeNftCollection,
187
+ NftCollectionScoreDivinerPlugin as default,
188
+ scoringCriteria
189
+ };
190
+ //# sourceMappingURL=index.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/Diviner.ts","../../src/lib/rating/criteria/scoring/metadata/scoreIndividualAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/scoreTotalAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/metadata.ts","../../src/lib/rating/criteria/scoring/total.ts","../../src/lib/rating/criteria/index.ts","../../src/lib/rating/analyzeNftCollection.ts","../../src/Plugin.ts"],"sourcesContent":["import {\n isNftCollectionInfo,\n NftCollectionInfo,\n NftCollectionMetadata,\n NftCollectionScore,\n NftCollectionScoreDivinerConfig,\n NftCollectionScoreDivinerConfigSchema,\n NftCollectionScoreSchema,\n} from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { DivinerParams } from '@xyo-network/diviner-model'\nimport { AnyConfigSchema } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload, Schema } from '@xyo-network/payload-model'\n\nimport { analyzeNftCollection, NftCollectionAnalysis } from './lib/index.js'\n\nexport type NftCollectionScoreDivinerParams = DivinerParams<AnyConfigSchema<NftCollectionScoreDivinerConfig>>\n\nconst toNftCollectionScore = (nftCollectionInfo: NftCollectionInfo, scores: NftCollectionAnalysis): NftCollectionScore => {\n const { name, symbol, address, chainId, type } = nftCollectionInfo\n const metadata: NftCollectionMetadata = { address, chainId, name, symbol, type }\n return { ...metadata, schema: NftCollectionScoreSchema, scores }\n}\n\nexport class NftCollectionScoreDiviner<\n TParams extends NftCollectionScoreDivinerParams = NftCollectionScoreDivinerParams,\n> extends AbstractDiviner<TParams> {\n static override readonly configSchemas: Schema[] = [...super.configSchemas, NftCollectionScoreDivinerConfigSchema]\n static override readonly defaultConfigSchema: Schema = NftCollectionScoreDivinerConfigSchema\n\n protected override divineHandler = async (payloads?: Payload[]): Promise<Payload[]> => {\n const nftCollectionInfos = payloads?.filter(isNftCollectionInfo) ?? []\n const results = await Promise.all(\n nftCollectionInfos.map<Promise<NftCollectionScore>>(async (nftCollectionInfo) => {\n const [score, sourceHash] = await Promise.all([\n // Get score\n toNftCollectionScore(nftCollectionInfo, await analyzeNftCollection(nftCollectionInfo)),\n // Hash sources\n PayloadBuilder.dataHash(nftCollectionInfo),\n ])\n return { ...score, schema: NftCollectionScoreSchema, sources: [sourceHash] } as NftCollectionScore\n }),\n )\n return results\n }\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\nexport const scoreIndividualAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n const entries = Object.entries(attributes)\n if (entries.length === 0) return [0, maxScore]\n const scores = Object.entries(attributes).flatMap(([_trait, { values }]) => {\n return Object.entries(values).map<Score>(([_traitValue, metrics]) => {\n const rarity = Math.min(Math.round((1 - metrics.binomial.p) * maxScore), maxScore)\n return [rarity, maxScore]\n })\n })\n // eslint-disable-next-line unicorn/no-array-reduce\n const total = scores.reduce<Score>(([a, b], [c, d]) => [a + c, b + d], [0, 0])\n return normalize(total, maxScore)\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\n/**\n * Mean: What value is the distribution centered around\n */\nconst defaultMu = 0.15\n\n/**\n * Standard Deviation: How spread out is the distribution\n */\nconst defaultSigma = 0.1\n\n/**\n * Calculates the Gaussian probability density\n * @param x\n * @param mu Mean\n * @param sigma Standard Deviation\n * @returns\n */\nconst gaussianProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n const sqrtTwoPi = Math.sqrt(2 * Math.PI)\n const denominator = sigma * sqrtTwoPi\n const power = -0.5 * Math.pow((x - mu) / sigma, 2)\n return (1 / denominator) * Math.exp(power)\n}\n\n/**\n * For a Gaussian distribution, the peak of the distribution is the mean\n */\nconst maxProbabilityDensity = gaussianProbabilityDensity(defaultMu)\n\n/**\n * We're working on some assumptions here:\n *\n * - If you have a 100% chance of getting a trait, everyone get's a trophy\n * - If you have a 50% chance of getting a trait, it's not rare\n * - If you have a 0% chance of getting a trait, it's not fun\n *\n * So we're looking for something Pareto-ish (somewhere between\n * 80/20 or 90/10) as that's a good & sustainable model for the\n * distribution of many traits in real life.\n * However, we also don't want to maximally reward collections\n * that have a lot of single attributes distributed uniformly\n * (basically a 0% trait probably) as that's perfectly entropic\n * but not very interesting (some overlap is desirable).\n * So we're using a Gaussian distribution to model the\n * probability density of the joint probability of all traits\n * centered around 15%.\n * @param info\n * @returns\n */\nexport const scoreTotalAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n // This has somewhat of a filtering function by causing anything with 100% probability to\n // add no value to the end score\n const jointProbability = Object.entries(attributes).reduce((acc, [_trait, { metrics }]) => {\n return acc * metrics.binomial.p\n }, 1)\n const probabilityDensity = gaussianProbabilityDensity(jointProbability)\n const score: Score = [probabilityDensity, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreIndividualAttributes } from './scoreIndividualAttributes.js'\nimport { scoreTotalAttributes } from './scoreTotalAttributes.js'\n\nexport const attributeScoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n 'Metadata Attributes Individual': { score: scoreIndividualAttributes, weight: 2 },\n 'Metadata Attributes Total': { score: scoreTotalAttributes, weight: 2 },\n}\n\nexport const scoreMetadata: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...attributeScoringCriteria,\n}\n","import { NftCollectionCount } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\n/**\n * This \"magic\" value was obtained using Solver in Excel\n * to find the median, with mu/sigma fixed, which maximizes\n * the distribution (the mode for lognormal) at 10,000\n */\nconst median = 81_030_839.821_735_2\nconst defaultMu = Math.log(median)\nconst defaultSigma = 3\nconst mode = Math.exp(defaultMu - Math.pow(defaultSigma, 2))\n/**\n * Calculates the log-normal probability density\n * @param x the value at which you want to calculate the probability density\n * @param mu mean of the associated normal distribution\n * @param sigma standard deviation of the associated normal distribution\n * @returns\n */\nconst logNormalProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n if (x <= 0) return 0\n const logX = Math.log(x)\n return (1 / (x * sigma * Math.sqrt(2 * Math.PI))) * Math.exp(-0.5 * Math.pow((logX - mu) / sigma, 2))\n}\n\n/**\n * For a lognormal distribution, the peak of the distribution is the mode\n */\nconst maxProbabilityDensity = logNormalProbabilityDensity(mode)\n\nconst maxScore = 10\n\n/**\n * We're working on some assumptions here:\n * - If there's < 1000 NFTs in your collection it starts becoming too niche\n * - If there's > 20,000 NFTs in your collection it starts becoming too broad\n * So there's a sweet spot somewhere between 2000 and 10,000\n * where a collection has enough NFTs to be interesting, but\n * not so many that it's teetering on a diluted money grab.\n * To model that we're using a log-normal distribution optimized\n * to maximally reward collections in the aforementioned range\n * @param nft\n * @returns\n */\nexport const scoreTotal = (nft: NftCollectionCount): Score => {\n const density = logNormalProbabilityDensity(nft.total)\n const score: Score = [density, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreMetadata, scoreTotal } from './scoring/index.js'\n\nexport const scoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...scoreMetadata,\n Total: { score: scoreTotal, weight: 2 },\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { Score } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoringCriteria } from './criteria/index.js'\n\nexport type ScoringCriteriaKey = keyof typeof scoringCriteria & PropertyKey\n\nexport type NftCollectionAnalysis = {\n [key in ScoringCriteriaKey]: Score\n}\n\nexport const analyzeNftCollection = async (\n /**\n * The NFT to evaluate\n */\n nft: NftCollectionInfo,\n): Promise<NftCollectionAnalysis> => {\n const result = Object.fromEntries(\n await Promise.all(\n Object.entries(scoringCriteria).map(async ([key, { score, weight }]) => {\n const rawScore = await score(nft)\n const weighted = rawScore.map((v) => v * weight) as Score\n return [key, weighted] as const\n }),\n ),\n ) as NftCollectionAnalysis\n return result\n}\n","import { NftSchema } from '@xyo-network/crypto-nft-payload-plugin'\nimport { PayloadSetSchema } from '@xyo-network/payload-model'\nimport { createPayloadSetDivinerPlugin } from '@xyo-network/payloadset-plugin'\n\nimport { NftCollectionScoreDiviner } from './Diviner.js'\n\nexport const NftCollectionScoreDivinerPlugin = () =>\n createPayloadSetDivinerPlugin<NftCollectionScoreDiviner>(\n { required: { [NftSchema]: 1 }, schema: PayloadSetSchema },\n {\n diviner: async (params) => {\n const result = await NftCollectionScoreDiviner.create(params)\n return result\n },\n },\n )\n"],"mappings":"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","names":["normalize","maxScore","normalize","defaultMu","defaultSigma","maxProbabilityDensity","maxScore"]}
1
+ {"version":3,"sources":["../../src/Diviner.ts","../../src/lib/rating/criteria/scoring/metadata/scoreIndividualAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/scoreTotalAttributes.ts","../../src/lib/rating/criteria/scoring/metadata/metadata.ts","../../src/lib/rating/criteria/scoring/total.ts","../../src/lib/rating/criteria/index.ts","../../src/lib/rating/analyzeNftCollection.ts","../../src/Plugin.ts"],"sourcesContent":["import {\n isNftCollectionInfo,\n NftCollectionInfo,\n NftCollectionMetadata,\n NftCollectionScore,\n NftCollectionScoreDivinerConfig,\n NftCollectionScoreDivinerConfigSchema,\n NftCollectionScoreSchema,\n} from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { AbstractDiviner } from '@xyo-network/diviner-abstract'\nimport { DivinerParams } from '@xyo-network/diviner-model'\nimport { AnyConfigSchema } from '@xyo-network/module-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload, Schema } from '@xyo-network/payload-model'\n\nimport { analyzeNftCollection, NftCollectionAnalysis } from './lib/index.ts'\n\nexport type NftCollectionScoreDivinerParams = DivinerParams<AnyConfigSchema<NftCollectionScoreDivinerConfig>>\n\nconst toNftCollectionScore = (nftCollectionInfo: NftCollectionInfo, scores: NftCollectionAnalysis): NftCollectionScore => {\n const { name, symbol, address, chainId, type } = nftCollectionInfo\n const metadata: NftCollectionMetadata = { address, chainId, name, symbol, type }\n return { ...metadata, schema: NftCollectionScoreSchema, scores }\n}\n\nexport class NftCollectionScoreDiviner<\n TParams extends NftCollectionScoreDivinerParams = NftCollectionScoreDivinerParams,\n> extends AbstractDiviner<TParams> {\n static override readonly configSchemas: Schema[] = [...super.configSchemas, NftCollectionScoreDivinerConfigSchema]\n static override readonly defaultConfigSchema: Schema = NftCollectionScoreDivinerConfigSchema\n\n protected override divineHandler = async (payloads?: Payload[]): Promise<Payload[]> => {\n const nftCollectionInfos = payloads?.filter(isNftCollectionInfo) ?? []\n const results = await Promise.all(\n nftCollectionInfos.map<Promise<NftCollectionScore>>(async (nftCollectionInfo) => {\n const [score, sourceHash] = await Promise.all([\n // Get score\n toNftCollectionScore(nftCollectionInfo, await analyzeNftCollection(nftCollectionInfo)),\n // Hash sources\n PayloadBuilder.dataHash(nftCollectionInfo),\n ])\n return { ...score, schema: NftCollectionScoreSchema, sources: [sourceHash] } as NftCollectionScore\n }),\n )\n return results\n }\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\nexport const scoreIndividualAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n const entries = Object.entries(attributes)\n if (entries.length === 0) return [0, maxScore]\n const scores = Object.entries(attributes).flatMap(([_trait, { values }]) => {\n return Object.entries(values).map<Score>(([_traitValue, metrics]) => {\n const rarity = Math.min(Math.round((1 - metrics.binomial.p) * maxScore), maxScore)\n return [rarity, maxScore]\n })\n })\n // eslint-disable-next-line unicorn/no-array-reduce\n const total = scores.reduce<Score>(([a, b], [c, d]) => [a + c, b + d], [0, 0])\n return normalize(total, maxScore)\n}\n","import { NftCollectionAttributeMetrics } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\nconst maxScore = 10\n\n/**\n * Mean: What value is the distribution centered around\n */\nconst defaultMu = 0.15\n\n/**\n * Standard Deviation: How spread out is the distribution\n */\nconst defaultSigma = 0.1\n\n/**\n * Calculates the Gaussian probability density\n * @param x\n * @param mu Mean\n * @param sigma Standard Deviation\n * @returns\n */\nconst gaussianProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n const sqrtTwoPi = Math.sqrt(2 * Math.PI)\n const denominator = sigma * sqrtTwoPi\n const power = -0.5 * Math.pow((x - mu) / sigma, 2)\n return (1 / denominator) * Math.exp(power)\n}\n\n/**\n * For a Gaussian distribution, the peak of the distribution is the mean\n */\nconst maxProbabilityDensity = gaussianProbabilityDensity(defaultMu)\n\n/**\n * We're working on some assumptions here:\n *\n * - If you have a 100% chance of getting a trait, everyone get's a trophy\n * - If you have a 50% chance of getting a trait, it's not rare\n * - If you have a 0% chance of getting a trait, it's not fun\n *\n * So we're looking for something Pareto-ish (somewhere between\n * 80/20 or 90/10) as that's a good & sustainable model for the\n * distribution of many traits in real life.\n * However, we also don't want to maximally reward collections\n * that have a lot of single attributes distributed uniformly\n * (basically a 0% trait probably) as that's perfectly entropic\n * but not very interesting (some overlap is desirable).\n * So we're using a Gaussian distribution to model the\n * probability density of the joint probability of all traits\n * centered around 15%.\n * @param info\n * @returns\n */\nexport const scoreTotalAttributes = (info: NftCollectionAttributeMetrics): Score => {\n const { attributes } = info.metrics.metadata\n // This has somewhat of a filtering function by causing anything with 100% probability to\n // add no value to the end score\n const jointProbability = Object.entries(attributes).reduce((acc, [_trait, { metrics }]) => {\n return acc * metrics.binomial.p\n }, 1)\n const probabilityDensity = gaussianProbabilityDensity(jointProbability)\n const score: Score = [probabilityDensity, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreIndividualAttributes } from './scoreIndividualAttributes.ts'\nimport { scoreTotalAttributes } from './scoreTotalAttributes.ts'\n\nexport const attributeScoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n 'Metadata Attributes Individual': { score: scoreIndividualAttributes, weight: 2 },\n 'Metadata Attributes Total': { score: scoreTotalAttributes, weight: 2 },\n}\n\nexport const scoreMetadata: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...attributeScoringCriteria,\n}\n","import { NftCollectionCount } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { normalize, Score } from '@xyo-network/crypto-nft-score-model'\n\n/**\n * This \"magic\" value was obtained using Solver in Excel\n * to find the median, with mu/sigma fixed, which maximizes\n * the distribution (the mode for lognormal) at 10,000\n */\nconst median = 81_030_839.821_735_2\nconst defaultMu = Math.log(median)\nconst defaultSigma = 3\nconst mode = Math.exp(defaultMu - Math.pow(defaultSigma, 2))\n/**\n * Calculates the log-normal probability density\n * @param x the value at which you want to calculate the probability density\n * @param mu mean of the associated normal distribution\n * @param sigma standard deviation of the associated normal distribution\n * @returns\n */\nconst logNormalProbabilityDensity = (x: number, mu: number = defaultMu, sigma: number = defaultSigma): number => {\n if (x <= 0) return 0\n const logX = Math.log(x)\n return (1 / (x * sigma * Math.sqrt(2 * Math.PI))) * Math.exp(-0.5 * Math.pow((logX - mu) / sigma, 2))\n}\n\n/**\n * For a lognormal distribution, the peak of the distribution is the mode\n */\nconst maxProbabilityDensity = logNormalProbabilityDensity(mode)\n\nconst maxScore = 10\n\n/**\n * We're working on some assumptions here:\n * - If there's < 1000 NFTs in your collection it starts becoming too niche\n * - If there's > 20,000 NFTs in your collection it starts becoming too broad\n * So there's a sweet spot somewhere between 2000 and 10,000\n * where a collection has enough NFTs to be interesting, but\n * not so many that it's teetering on a diluted money grab.\n * To model that we're using a log-normal distribution optimized\n * to maximally reward collections in the aforementioned range\n * @param nft\n * @returns\n */\nexport const scoreTotal = (nft: NftCollectionCount): Score => {\n const density = logNormalProbabilityDensity(nft.total)\n const score: Score = [density, maxProbabilityDensity]\n return normalize(score, maxScore)\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { WeightedScoringCriteria } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoreMetadata, scoreTotal } from './scoring/index.ts'\n\nexport const scoringCriteria: { [key: string]: WeightedScoringCriteria<NftCollectionInfo> } = {\n ...scoreMetadata,\n Total: { score: scoreTotal, weight: 2 },\n}\n","import { NftCollectionInfo } from '@xyo-network/crypto-nft-collection-payload-plugin'\nimport { Score } from '@xyo-network/crypto-nft-score-model'\n\nimport { scoringCriteria } from './criteria/index.ts'\n\nexport type ScoringCriteriaKey = keyof typeof scoringCriteria & PropertyKey\n\nexport type NftCollectionAnalysis = {\n [key in ScoringCriteriaKey]: Score\n}\n\nexport const analyzeNftCollection = async (\n /**\n * The NFT to evaluate\n */\n nft: NftCollectionInfo,\n): Promise<NftCollectionAnalysis> => {\n const result = Object.fromEntries(\n await Promise.all(\n Object.entries(scoringCriteria).map(async ([key, { score, weight }]) => {\n const rawScore = await score(nft)\n const weighted = rawScore.map(v => v * weight) as Score\n return [key, weighted] as const\n }),\n ),\n ) as NftCollectionAnalysis\n return result\n}\n","import { NftSchema } from '@xyo-network/crypto-nft-payload-plugin'\nimport { PayloadSetSchema } from '@xyo-network/payload-model'\nimport { createPayloadSetDivinerPlugin } from '@xyo-network/payloadset-plugin'\n\nimport { NftCollectionScoreDiviner } from './Diviner.ts'\n\nexport const NftCollectionScoreDivinerPlugin = () =>\n createPayloadSetDivinerPlugin<NftCollectionScoreDiviner>(\n { required: { [NftSchema]: 1 }, schema: PayloadSetSchema },\n {\n diviner: async (params) => {\n const result = await NftCollectionScoreDiviner.create(params)\n return result\n },\n },\n )\n"],"mappings":"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","names":["isNftCollectionInfo","NftCollectionScoreDivinerConfigSchema","NftCollectionScoreSchema","AbstractDiviner","PayloadBuilder","normalize","maxScore","scoreIndividualAttributes","info","attributes","metrics","metadata","entries","Object","length","scores","flatMap","_trait","values","map","_traitValue","rarity","Math","min","round","binomial","p","total","reduce","a","b","c","d","normalize","normalize","maxScore","defaultMu","defaultSigma","gaussianProbabilityDensity","x","mu","sigma","sqrtTwoPi","Math","sqrt","PI","denominator","power","pow","exp","maxProbabilityDensity","scoreTotalAttributes","info","attributes","metrics","metadata","jointProbability","Object","entries","reduce","acc","_trait","binomial","p","probabilityDensity","score","normalize","attributeScoringCriteria","score","scoreIndividualAttributes","weight","scoreTotalAttributes","scoreMetadata","normalize","median","defaultMu","Math","log","defaultSigma","mode","exp","pow","logNormalProbabilityDensity","x","mu","sigma","logX","sqrt","PI","maxProbabilityDensity","maxScore","scoreTotal","nft","density","total","score","normalize","scoringCriteria","scoreMetadata","Total","score","scoreTotal","weight","analyzeNftCollection","nft","result","Object","fromEntries","Promise","all","entries","scoringCriteria","map","key","score","weight","rawScore","weighted","v","toNftCollectionScore","nftCollectionInfo","scores","name","symbol","address","chainId","type","metadata","schema","NftCollectionScoreSchema","NftCollectionScoreDiviner","AbstractDiviner","configSchemas","NftCollectionScoreDivinerConfigSchema","defaultConfigSchema","divineHandler","payloads","nftCollectionInfos","filter","isNftCollectionInfo","results","Promise","all","map","score","sourceHash","analyzeNftCollection","PayloadBuilder","dataHash","sources","NftSchema","PayloadSetSchema","createPayloadSetDivinerPlugin","NftCollectionScoreDivinerPlugin","createPayloadSetDivinerPlugin","required","NftSchema","schema","PayloadSetSchema","diviner","params","result","NftCollectionScoreDiviner","create"]}
@@ -1,2 +1,2 @@
1
- export * from './rating/index.js';
1
+ export * from './rating/index.ts';
2
2
  //# sourceMappingURL=index.d.ts.map