Skip to content
This repository was archived by the owner on Aug 12, 2023. It is now read-only.

Commit 7b5c19e

Browse files
authored
Introduce app definition sync + new definitions (#432)
* Introduce app definition sync + new definitions * Add api to allowed app categories * Add portfolio-manager to allow app categories
1 parent 2076e93 commit 7b5c19e

11 files changed

+474
-3
lines changed

src/app/initialise.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -4,13 +4,18 @@ const { QUEUE } = require('../constants');
44
const { initQueues } = require('../queues');
55
const configure = require('./configure');
66
const populateRelayersCollection = require('../relayers/populate-relayers-collection');
7+
const syncAppDefinitions = require('../apps/sync-app-definitions');
78
const tokenCache = require('../tokens/token-cache');
89

910
const initialise = async () => {
1011
await configure();
1112
initQueues(Object.values(QUEUE), config.get('queues'));
1213

13-
await Promise.all([tokenCache.initialise(), populateRelayersCollection()]);
14+
await Promise.all([
15+
populateRelayersCollection(),
16+
syncAppDefinitions(),
17+
tokenCache.initialise(),
18+
]);
1419
};
1520

1621
module.exports = initialise;

src/apps/definitions/0x-api.json

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
{
2+
"id": "052b4862-2142-4532-bdc0-416814b0a5fe",
3+
"name": "0x API",
4+
"logoUrl": "https://resources.0xtracker.com/logos/0x.png",
5+
"urlSlug": "0x-api",
6+
"websiteUrl": "https://0x.org/api",
7+
"mappings": [
8+
{
9+
"type": "relayer",
10+
"feeRecipientAddress": "0x1000000000000000000000000000000000000011"
11+
}
12+
],
13+
"categories": [
14+
"api"
15+
]
16+
}

src/apps/definitions/defi-saver.json

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
{
2+
"id": "b986d2ba-77bf-420b-99e3-28e592d476e2",
3+
"name": "DeFi Saver",
4+
"logoUrl": "https://resources.0xtracker.com/logos/defi-saver.png",
5+
"urlSlug": "defi-saver",
6+
"websiteUrl": "https://defisaver.com/",
7+
"mappings": [
8+
{
9+
"affiliateAddress": "0x322d58b9e75a6918f7e7849aee0ff09369977e08",
10+
"type": "consumer"
11+
}
12+
],
13+
"categories": [
14+
"portfolio-manager"
15+
]
16+
}

src/apps/definitions/paraswap.json

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
{
2+
"id": "4c3854b7-95c3-4714-83ab-a8ed53d2967e",
3+
"name": "Paraswap",
4+
"logoUrl": "https://resources.0xtracker.com/logos/paraswap.png",
5+
"urlSlug": "paraswap",
6+
"websiteUrl": "https://paraswap.io",
7+
"mappings": [
8+
{
9+
"type": "consumer",
10+
"feeRecipientAddress": "0x72338b82800400f5488eca2b5a37270ba3b7a111"
11+
},
12+
{
13+
"type": "consumer",
14+
"feeRecipientAddress": "0xf92c1ad75005e6436b4ee84e88cb23ed8a290988"
15+
},
16+
{
17+
"type": "consumer",
18+
"feeRecipientAddress": "0xffef2d75b3d266f90108e82b3b476a7f08fe5385"
19+
}
20+
],
21+
"categories": [
22+
"dex-aggregator"
23+
]
24+
}

src/apps/index.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
11
const resolveApps = require('./resolve-apps');
2+
const syncAppDefinitions = require('./sync-app-definitions');
23

3-
module.exports = { resolveApps };
4+
module.exports = { resolveApps, syncAppDefinitions };

src/apps/sync-app-definitions.js

+116
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,116 @@
1+
const _ = require('lodash');
2+
const Bluebird = require('bluebird');
3+
4+
const { APP_TYPE } = require('../constants');
5+
const { getModel } = require('../model');
6+
// const { JOB, QUEUE } = require('../constants');
7+
// const { publishJob } = require('../queues');
8+
const getAppDefinitions = require('./get-app-definitions');
9+
10+
// TODO: Introduce and test in future PR
11+
// const scheduleBackfill = async appId => {
12+
// await publishJob(
13+
// QUEUE.APP_PROCESSING,
14+
// JOB.BACKFILL_APP,
15+
// {
16+
// id: appId,
17+
// },
18+
// {
19+
// jobId: `backfill-app-${appId}`,
20+
// removeOnComplete: false,
21+
// },
22+
// );
23+
// };
24+
25+
const transformMappings = mappings =>
26+
mappings.map(m => ({
27+
...m,
28+
type: APP_TYPE[m.type.toUpperCase()],
29+
}));
30+
31+
const createApp = async definition => {
32+
const App = getModel('App');
33+
const app = {
34+
..._.omit(definition, 'id', 'mappings'),
35+
_id: definition.id,
36+
mappings: transformMappings(definition.mappings),
37+
};
38+
39+
await App.create(app);
40+
// await scheduleBackfill(definition.id);
41+
};
42+
43+
const compareMappings = (currentMappings, definitionMappings) => {
44+
const newMappings = _.differenceWith(
45+
definitionMappings,
46+
currentMappings,
47+
_.isEqual,
48+
);
49+
50+
const mappingsModified = newMappings.length > 0;
51+
52+
return mappingsModified;
53+
};
54+
55+
const updateApp = async (app, definition) => {
56+
const metadata = _.omit(definition, 'id', 'mappings');
57+
58+
Object.keys(metadata).forEach(metadataKey => {
59+
app.set(metadataKey, metadata[metadataKey]);
60+
});
61+
62+
const currentMappings = app.mappings.map(m =>
63+
_.pickBy(
64+
_.pick(
65+
m,
66+
'affiliateAddress',
67+
'feeRecipientAddress',
68+
'takerAddress',
69+
'type',
70+
),
71+
value => value !== undefined,
72+
),
73+
);
74+
75+
const nextMappings = transformMappings(definition.mappings);
76+
const mappingsModified = compareMappings(currentMappings, nextMappings);
77+
78+
if (mappingsModified) {
79+
app.set('mappings', nextMappings);
80+
}
81+
82+
if (app.isModified()) {
83+
await app.save();
84+
}
85+
86+
// if (mappingsModified) {
87+
// await scheduleBackfill(definition.id);
88+
// }
89+
};
90+
91+
/**
92+
* Sync current app definitions with MongoDB making sure that any differences
93+
* in metadata or mappings are reflected in MongoDB.
94+
*
95+
* Any changes to definition mappings will trigger an attributions backfill
96+
* for the app in which the mappings changed.
97+
*
98+
* NOTE: Removal of apps or mappings must be handled manually. The sync process
99+
* is not currently built to automate this since it's unlikely to occur.
100+
*/
101+
const syncAppDefinitions = async () => {
102+
const App = getModel('App');
103+
const definitions = getAppDefinitions();
104+
105+
await Bluebird.each(definitions, async definition => {
106+
const app = await App.findById(definition.id);
107+
108+
if (app === null) {
109+
await createApp(definition);
110+
} else {
111+
await updateApp(app, definition);
112+
}
113+
});
114+
};
115+
116+
module.exports = syncAppDefinitions;

0 commit comments

Comments
 (0)