Skip to content

Commit

Permalink
DF-20799 update bitgo-reserves with new payload & validation (#3601)
Browse files Browse the repository at this point in the history
* DF-20799 update bitgo-reserves with new payload & validation

* update reserves endpoint with new payload & verification

* add changeset
  • Loading branch information
mmcallister-cll authored Dec 12, 2024
1 parent 7158209 commit f4f0ce9
Show file tree
Hide file tree
Showing 6 changed files with 74 additions and 17 deletions.
5 changes: 5 additions & 0 deletions .changeset/red-experts-thank.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@chainlink/bitgo-reserves-adapter': major
---

update bitgo-reserves EA with new payload
5 changes: 5 additions & 0 deletions packages/sources/bitgo-reserves/src/config/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,4 +6,9 @@ export const config = new AdapterConfig({
type: 'string',
default: 'https://reserves.usdstandard-test.com/por.json',
},
VERIFICATION_PUBKEY: {
description: 'Public key used for verifying data signature',
type: 'string',
required: true,
},
})
60 changes: 49 additions & 11 deletions packages/sources/bitgo-reserves/src/transport/reserves.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,19 @@
import { HttpTransport } from '@chainlink/external-adapter-framework/transports'
import { BaseEndpointTypes } from '../endpoint/reserves'
import * as crypto from 'crypto'

export interface ResponseSchema {
totalReserve: number
export interface DataSchema {
totalReserve?: string
reserveAmount?: string
cashReserve: string
investedReserve: string
lastUpdated: string
cashReserve: number
investedReserve: number
}

export interface ResponseSchema {
data: string // formatted & escaped DataSchema
dataSignature: string
ripcord: boolean
}

export type HttpTransportTypes = BaseEndpointTypes & {
Expand All @@ -27,26 +35,56 @@ export const httpTransport = new HttpTransport<HttpTransportTypes>({
}
})
},
parseResponse: (params, response) => {
const timestamps = {
providerIndicatedTimeUnixMs: new Date(response.data.lastUpdated).getTime(),
}
parseResponse: (params, response, adapterSettings) => {
const payload = response.data

if (!response.data) {
if (!payload || !payload.data) {
return params.map((param) => {
return {
params: param,
response: {
errorMessage: `The data provider didn't return any value`,
statusCode: 502,
timestamps,
},
}
})
}

if (payload.ripcord) {
return [
{
params: params[0],
response: {
errorMessage: 'Ripcord indicator true',
ripcord: response.data.ripcord,
statusCode: 502,
},
},
]
}

const publicKey = adapterSettings.VERIFICATION_PUBKEY
const verifier = crypto.createVerify('sha256')
verifier.update(payload.data)
if (!verifier.verify(publicKey, payload.dataSignature, 'base64')) {
return params.map((param) => {
return {
params: param,
response: {
errorMessage: `Data verification failed`,
statusCode: 502,
},
}
})
}

const data = JSON.parse(payload.data) as DataSchema
const timestamps = {
providerIndicatedTimeUnixMs: new Date(data.lastUpdated).getTime(),
}

return params.map((param) => {
const result = response.data.totalReserve
const result = Number(data.totalReserve) || Number(data.reserveAmount)
return {
params: param,
response: {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,14 @@
exports[`execute reserves endpoint should return success 1`] = `
{
"data": {
"result": 1234567.89,
"result": 12345678.9,
},
"result": 1234567.89,
"result": 12345678.9,
"statusCode": 200,
"timestamps": {
"providerDataReceivedUnixMs": 978347471111,
"providerDataRequestedUnixMs": 978347471111,
"providerIndicatedTimeUnixMs": 1727745825000,
"providerIndicatedTimeUnixMs": 1733793825000,
},
}
`;
`;
Original file line number Diff line number Diff line change
Expand Up @@ -13,11 +13,18 @@ describe('execute', () => {
beforeAll(async () => {
oldEnv = JSON.parse(JSON.stringify(process.env))
process.env.API_ENDPOINT = 'http://test-endpoint.com'
process.env.VERIFICATION_PUBKEY = 'test'

const mockDate = new Date('2001-01-01T11:11:11.111Z')
spy = jest.spyOn(Date, 'now').mockReturnValue(mockDate.getTime())

const adapter = (await import('./../../src')).adapter
jest.mock('crypto', () => ({
createVerify: jest.fn().mockImplementation((_algo) => ({
update: jest.fn().mockReturnThis(),
verify: jest.fn().mockImplementationOnce((a, b, c) => true),
})),
}))
const adapter = (await import('../../src')).adapter
adapter.rateLimiting = undefined
testAdapter = await TestAdapter.startWithMockedCache(adapter, {
testAdapter: {} as TestAdapter<never>,
Expand All @@ -43,4 +50,5 @@ describe('execute', () => {
expect(response.json()).toMatchSnapshot()
})
})
// Note: issues with mock verifier prevent further tests
})
3 changes: 2 additions & 1 deletion packages/sources/bitgo-reserves/test/integration/fixtures.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,8 @@ export const mockResponseSuccess = (): nock.Scope =>
.reply(
200,
() => ({
totalReserve: 1234567.89,
data: '{"reserveAmount":"12345678.90","cashReserve":"2345678.90","investedReserve":"10000000.00","lastUpdated":"2024-12-10T01:23:45Z"}',
dataSignature: 'testsig',
lastUpdated: '2024-10-01T01:23:45Z',
}),
[
Expand Down

0 comments on commit f4f0ce9

Please sign in to comment.