Skip to content

Commit

Permalink
merge with dev
Browse files Browse the repository at this point in the history
  • Loading branch information
gowthamsundaresan committed Nov 5, 2024
2 parents 2cf4d5e + 601615c commit e83a30f
Show file tree
Hide file tree
Showing 4 changed files with 429 additions and 0 deletions.
228 changes: 228 additions & 0 deletions packages/api/src/routes/operators/operatorController.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import { SortByQuerySchema } from '../../schema/zod/schemas/sortByQuery'
import { SearchByTextQuerySchema } from '../../schema/zod/schemas/searchByTextQuery'
import { WithRewardsQuerySchema } from '../../schema/zod/schemas/withRewardsQuery'
import { RequestHeadersSchema } from '../../schema/zod/schemas/auth'
import { OperatorEventQuerySchema } from '../../schema/zod/schemas/operatorEvents'
import { handleAndReturnErrorResponse } from '../../schema/errors'
import {
getStrategiesWithShareUnderlying,
Expand All @@ -18,6 +19,23 @@ import Prisma from '@prisma/client'
import prisma from '../../utils/prismaClient'
import { fetchTokenPrices } from '../../utils/tokenPrices'

type EventRecordArgs = {
staker: string
strategy?: string
shares?: number
}

type EventRecord = {
type: 'SHARES_INCREASED' | 'SHARES_DECREASED' | 'DELEGATION' | 'UNDELEGATION'
tx: string
blockNumber: number
blockTime: Date
args: EventRecordArgs
underlyingToken?: string
underlyingValue?: number
ethValue?: number
}

/**
* Function for route /operators
* Returns a list of all Operators with optional sorts & text search
Expand Down Expand Up @@ -371,6 +389,105 @@ export async function getOperatorRewards(req: Request, res: Response) {
}
}

/**
* Function for route /operators/:address/events
* Fetches and returns a list of events for a specific operator with optional filters
*
* @param req
* @param res
*/
export async function getOperatorEvents(req: Request, res: Response) {
const result = OperatorEventQuerySchema.and(PaginationQuerySchema).safeParse(req.query)
if (!result.success) {
return handleAndReturnErrorResponse(req, res, result.error)
}

try {
const {
type,
stakerAddress,
strategyAddress,
txHash,
startAt,
endAt,
withTokenData,
withEthValue,
skip,
take
} = result.data
const { address } = req.params

const baseFilterQuery = {
operator: {
contains: address,
mode: 'insensitive'
},
...(stakerAddress && {
staker: {
contains: stakerAddress,
mode: 'insensitive'
}
}),
...(strategyAddress && {
strategy: {
contains: strategyAddress,
mode: 'insensitive'
}
}),
...(txHash && {
transactionHash: {
contains: txHash,
mode: 'insensitive'
}
}),
blockTime: {
gte: new Date(startAt),
...(endAt ? { lte: new Date(endAt) } : {})
}
}

let eventRecords: EventRecord[] = []
let eventCount = 0

const eventTypesToFetch = type
? [type]
: strategyAddress
? ['SHARES_INCREASED', 'SHARES_DECREASED']
: ['SHARES_INCREASED', 'SHARES_DECREASED', 'DELEGATION', 'UNDELEGATION']

const fetchEventsForTypes = async (types: string[]) => {
const results = await Promise.all(
types.map((eventType) =>
fetchAndMapEvents(eventType, baseFilterQuery, withTokenData, withEthValue, skip, take)
)
)
return results
}

const results = await fetchEventsForTypes(eventTypesToFetch)

eventRecords = results.flatMap((result) => result.eventRecords)
eventRecords = eventRecords
.sort((a, b) => (b.blockNumber > a.blockNumber ? 1 : -1))
.slice(0, take)

eventCount = results.reduce((acc, result) => acc + result.eventCount, 0)

const response = {
data: eventRecords,
meta: {
total: eventCount,
skip,
take
}
}

res.send(response)
} catch (error) {
handleAndReturnErrorResponse(req, res, error)
}
}

/**
* Function for route /operators/:address/invalidate-metadata
* Protected route to invalidate the metadata of a given Operator
Expand Down Expand Up @@ -568,3 +685,114 @@ async function calculateOperatorApy(operator: any) {
return response
} catch {}
}

/**
* Utility function to fetch and map event records from the database.
*
* @param eventType
* @param baseFilterQuery
* @param skip
* @param take
* @returns
*/
async function fetchAndMapEvents(
eventType: string,
baseFilterQuery: any,
withTokenData: boolean,
withEthValue: boolean,
skip: number,
take: number
): Promise<{ eventRecords: EventRecord[]; eventCount: number }> {
const modelName = (() => {
switch (eventType) {
case 'SHARES_INCREASED':
return 'eventLogs_OperatorSharesIncreased'
case 'SHARES_DECREASED':
return 'eventLogs_OperatorSharesDecreased'
case 'DELEGATION':
return 'eventLogs_StakerDelegated'
case 'UNDELEGATION':
return 'eventLogs_StakerUndelegated'
default:
throw new Error(`Unknown event type: ${eventType}`)
}
})()

const model = prisma[modelName] as any

const eventCount = await model.count({
where: baseFilterQuery
})

const eventLogs = await model.findMany({
where: baseFilterQuery,
skip,
take,
orderBy: { blockNumber: 'desc' }
})

const strategiesWithSharesUnderlying = withTokenData
? await getStrategiesWithShareUnderlying()
: undefined

const eventRecords = await Promise.all(
eventLogs.map(async (event) => {
let underlyingToken: string | undefined
let underlyingValue: number | undefined
let ethValue: number | undefined

if (
withTokenData &&
(eventType === 'SHARES_INCREASED' || eventType === 'SHARES_DECREASED') &&
event.strategy
) {
const strategy = await prisma.strategies.findUnique({
where: {
address: event.strategy.toLowerCase()
}
})

if (strategy && strategiesWithSharesUnderlying) {
underlyingToken = strategy.underlyingToken

const sharesUnderlying = strategiesWithSharesUnderlying.find(
(s) => s.strategyAddress.toLowerCase() === event.strategy.toLowerCase()
)

if (sharesUnderlying) {
underlyingValue =
Number(
(BigInt(event.shares) * BigInt(sharesUnderlying.sharesToUnderlying)) / BigInt(1e18)
) / 1e18

if (withEthValue && sharesUnderlying.ethPrice) {
ethValue = underlyingValue * sharesUnderlying.ethPrice
}
}
}
}

return {
type: eventType,
tx: event.transactionHash,
blockNumber: event.blockNumber,
blockTime: event.blockTime,
args: {
staker: event.staker.toLowerCase(),
strategy: event.strategy?.toLowerCase(),
shares: event.shares
},
...(withTokenData && {
underlyingToken: underlyingToken?.toLowerCase(),
underlyingValue
}),
...(withEthValue && { ethValue })
}
})
)

return {
eventRecords,
eventCount
}
}
3 changes: 3 additions & 0 deletions packages/api/src/routes/operators/operatorRoutes.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import {
getOperator,
getAllOperatorAddresses,
getOperatorRewards,
getOperatorEvents,
invalidateMetadata
} from './operatorController'

Expand Down Expand Up @@ -107,6 +108,8 @@ router.get('/:address', routeCache.cacheSeconds(120), getOperator)

router.get('/:address/rewards', routeCache.cacheSeconds(120), getOperatorRewards)

router.get('/:address/events/delegation', routeCache.cacheSeconds(120), getOperatorEvents)

// Protected routes
router.get('/:address/invalidate-metadata', routeCache.cacheSeconds(120), invalidateMetadata)

Expand Down
Loading

0 comments on commit e83a30f

Please sign in to comment.