Skip to content

Commit

Permalink
Merge pull request #121 from Synthetixio/dev
Browse files Browse the repository at this point in the history
* add new endpoints

* chore(deps): Bump actions/checkout from 4.1.7 to 4.2.2 (#118)

Bumps [actions/checkout](https://github.com/actions/checkout) from 4.1.7 to 4.2.2.
- [Release notes](https://github.com/actions/checkout/releases)
- [Changelog](https://github.com/actions/checkout/blob/main/CHANGELOG.md)
- [Commits](actions/checkout@692973e...11bd719)

---
updated-dependencies:
- dependency-name: actions/checkout
  dependency-type: direct:production
  update-type: version-update:semver-minor
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>

* chore(deps): Bump actions/cache from 4.0.2 to 4.1.2 (#119)

Bumps [actions/cache](https://github.com/actions/cache) from 4.0.2 to 4.1.2.
- [Release notes](https://github.com/actions/cache/releases)
- [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md)
- [Commits](actions/cache@0c45773...6849a64)

---
updated-dependencies:
- dependency-name: actions/cache
  dependency-type: direct:production
  update-type: version-update:semver-minor
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>

---------

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: Troy <troy.burmeister@gmail.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
  • Loading branch information
3 people authored Nov 21, 2024
2 parents 00eff78 + 69b5498 commit 1e7c3e3
Show file tree
Hide file tree
Showing 6 changed files with 424 additions and 4 deletions.
6 changes: 3 additions & 3 deletions .github/workflows/audit_and_lint.yml
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ jobs:

steps:
- name: Checkout
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683

- name: Audit dependencies
run: audit-ci --critical --report-type full
Expand All @@ -36,14 +36,14 @@ jobs:

steps:
- name: Checkout
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683

- name: Set pnpm cache directory
run: pnpm config set store-dir .pnpm-store
continue-on-error: true

- name: Setup cache
uses: actions/cache@0c45773b623bea8c8e75f6c82b208c3cf94ea4f9 # pin@v2
uses: actions/cache@6849a6489940f00c2f30c0fb92c6274307ccb58a # pin@v2
with:
path: |
.pnpm-store
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/codeql.yml
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ jobs:

steps:
- name: Checkout repository
uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683

- name: Initialize CodeQL
uses: github/codeql-action/init@99c9897648dded3fe63d6f328c46089dd57735ca
Expand Down
137 changes: 137 additions & 0 deletions src/routes/stats/perps-volume.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,137 @@
const express = require('express');
const router = express.Router();
const { log, postgresClient, getCache, setCache } = require('../../utils');

const cacheKey = 'perps-volume';

fetchDataFromPostgres();
const cacheTime =
((process.env.CACHE_TIME =
typeof process.env.CACHE_TIME === 'string'
? parseInt(process.env.CACHE_TIME)
: process.env.CACHE_TIME) -
30) *
1000;
setInterval(fetchDataFromPostgres, cacheTime < 30000 ? 30000 : cacheTime);

/**
* @openapi
* /stats/perps-volume:
* get:
* tags:
* - stats
* description: Returns total volume figures across all perps deployments.
* responses:
* 200:
* description: Successful response.
* content:
* application/json:
* schema:
* type: object
* properties:
* timestamp:
* type: string
* format: date-time
* example: '2024-05-23T14:00:00.000Z'
* volume_7d:
* type: number
* example: 123456789.123456789
* volume_24h:
* type: number
* example: 12345678.123456789
* 401:
* description: Unauthorized.
* 403:
* description: You have been banned by WAF.
* 429:
* description: Too many requests, you're being rate-limited.
* 5XX:
* description: Service unavailable.
* default:
* description: Unexpected error.
*/
router.get('/', async (req, res, next) => {
try {
log.debug('Checking cache..');
const cachedResponse = await getCache(cacheKey);
if (cachedResponse) {
log.debug('Cache found');
res.json(cachedResponse);
} else {
log.debug('Cache not found, executing..');
const responseData = await fetchDataFromPostgres();
res.json(responseData);
}
} catch (error) {
log.error(`[statsPerpsVolume] Error: ${error.message}`);
next(error);
}
});

module.exports = router;

async function fetchDataFromPostgres() {
log.debug('[statsPerpsVolume] Fetching data from postgres..');
const queryResult = await postgresClient.query(
`WITH volume AS (
SELECT ts,
'volume_24h' AS label,
volume
FROM prod_base_mainnet.fct_perp_stats_hourly_base_mainnet
WHERE ts >= NOW() - INTERVAL '24 HOURS'
UNION ALL
SELECT ts,
'volume_24h' AS label,
volume
FROM prod_arbitrum_mainnet.fct_perp_stats_hourly_arbitrum_mainnet
WHERE ts >= NOW() - INTERVAL '24 HOURS'
UNION ALL
select ts,
'volume_24h' AS label,
volume
from prod_optimism_mainnet.fct_v2_stats_hourly_optimism_mainnet
where ts >= NOW() - INTERVAL '24 HOURS'
UNION ALL
SELECT ts,
'volume_7d' AS label,
volume
FROM prod_base_mainnet.fct_perp_stats_hourly_base_mainnet
WHERE ts >= NOW() - INTERVAL '7 DAYS'
UNION ALL
SELECT ts,
'volume_7d' AS label,
volume
FROM prod_arbitrum_mainnet.fct_perp_stats_hourly_arbitrum_mainnet
WHERE ts >= NOW() - INTERVAL '7 DAYS'
UNION ALL
select ts,
'volume_7d' AS label,
volume
from prod_optimism_mainnet.fct_v2_stats_hourly_optimism_mainnet
where ts >= NOW() - INTERVAL '7 DAYS'
)
SELECT label,
round(SUM(volume), 2) AS volume
FROM volume
GROUP BY label;`,
);

const volume24h = queryResult.rows.find(
(row) => row.label === 'volume_24h',
).volume;
const volume7d = queryResult.rows.find(
(row) => row.label === 'volume_7d',
).volume;

const volume24hUsd = parseFloat(volume24h);
const volume7dUsd = parseFloat(volume7d);

const responseData = {
timestamp: new Date().toISOString(),
volume_24h: volume24hUsd,
volume_7d: volume7dUsd,
};
log.debug('[statsPerpsVolume] Setting cache..');
await setCache(cacheKey, responseData, 60);
return responseData;
}
155 changes: 155 additions & 0 deletions src/routes/v3/top-asset.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,155 @@
const express = require('express');
const router = express.Router();
const { log, postgresClient, getCache, setCache } = require('../../utils');

const cacheKey = 'v3-top-asset';

fetchDataFromPostgres();
const cacheTime =
((process.env.CACHE_TIME =
typeof process.env.CACHE_TIME === 'string'
? parseInt(process.env.CACHE_TIME)
: process.env.CACHE_TIME) -
30) *
1000;
setInterval(fetchDataFromPostgres, cacheTime < 30000 ? 30000 : cacheTime);

/**
* @openapi
* /v3/top-asset:
* get:
* tags:
* - v3
* description: Returns the current top performing asset and it's APR and APY estimated from past 7 day performance.
* responses:
* 200:
* description: Successful response.
* content:
* application/json:
* schema:
* type: object
* properties:
* timestamp:
* type: string
* format: date-time
* example: '2024-05-23T14:00:00.000Z'
* chain:
* type: string
* example: 'base'
* token_symbol:
* type: string
* example: 'USDC'
* apr:
* type: number
* example: 0.123456789
* apy:
* type: number
* example: 0.123456789
* 401:
* description: Unauthorized.
* 403:
* description: You have been banned by WAF.
* 429:
* description: Too many requests, you're being rate-limited.
* 5XX:
* description: Service unavailable.
* default:
* description: Unexpected error.
*/
router.get('/', async (req, res, next) => {
try {
log.debug('Checking cache..');
const cachedResponse = await getCache(cacheKey);
if (cachedResponse) {
log.debug('Cache found');
res.json(cachedResponse);
} else {
log.debug('Cache not found, executing..');
const responseData = await fetchDataFromPostgres();
res.json(responseData);
}
} catch (error) {
log.error(`[v3TopAsset] Error: ${error.message}`);
next(error);
}
});

module.exports = router;

async function fetchDataFromPostgres() {
log.debug('[v3TopAsset] Fetching data from postgres..');
const queryResult = await postgresClient.query(
`with base as (
SELECT 'base' as chain,
t.token_symbol,
apr.apy_7d,
apr.apr_7d,
ROW_NUMBER() OVER (
PARTITION BY collateral_type
ORDER BY ts DESC
) AS rn
FROM prod_base_mainnet.fct_core_apr_base_mainnet apr
join prod_seeds.base_mainnet_tokens t on lower(apr.collateral_type) = lower(t.token_address)
),
arb as (
SELECT 'arbitrum' as chain,
t.token_symbol,
apr.apy_7d,
apr.apr_7d,
ROW_NUMBER() OVER (
PARTITION BY collateral_type
ORDER BY ts DESC
) AS rn
FROM prod_arbitrum_mainnet.fct_core_apr_arbitrum_mainnet apr
join prod_seeds.arbitrum_mainnet_tokens t on lower(apr.collateral_type) = lower(t.token_address)
),
eth as (
SELECT 'ethereum' as chain,
t.token_symbol,
apr.apy_7d,
apr.apr_7d,
ROW_NUMBER() OVER (
PARTITION BY collateral_type
ORDER BY ts DESC
) AS rn
FROM prod_eth_mainnet.fct_core_apr_eth_mainnet apr
join prod_seeds.eth_mainnet_tokens t on lower(apr.collateral_type) = lower(t.token_address)
),
combined as (
select *
from base
where rn = 1
union all
select *
from arb
where rn = 1
union all
select *
from eth
where rn = 1
)
select chain,
token_symbol,
round(apy_7d, 8) as apy,
round(apr_7d, 8) as apr
from combined
order by apy_7d desc
limit 1;`,
);

const chain = queryResult.rows[0].chain;
const tokenSymbol = queryResult.rows[0].token_symbol;
const apy = parseFloat(queryResult.rows[0].apy);
const apr = parseFloat(queryResult.rows[0].apr);

const responseData = {
timestamp: new Date().toISOString(),
chain,
token_symbol: tokenSymbol,
apr,
apy,
};
log.debug('[v3TopAsset] Setting cache..');
await setCache(cacheKey, responseData, 60);
return responseData;
}
Loading

0 comments on commit 1e7c3e3

Please sign in to comment.