-
Notifications
You must be signed in to change notification settings - Fork 8
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
7 changed files
with
167 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,72 @@ | ||
import * as promiseUtilsModule from '@api3/promise-utils'; | ||
|
||
import { config, parseHeartbeatLog } from '../../test/fixtures'; | ||
import { logger } from '../logger'; | ||
import * as stateModule from '../state'; | ||
import { loadRawConfig } from '../validation/config'; | ||
|
||
import { initiateHeartbeat, logHeartbeat, createHash } from '.'; | ||
|
||
// eslint-disable-next-line jest/no-hooks | ||
beforeEach(() => { | ||
jest.useFakeTimers().setSystemTime(new Date('2023-01-20')); | ||
}); | ||
|
||
afterEach(() => { | ||
jest.useRealTimers(); | ||
}); | ||
|
||
describe(logHeartbeat.name, () => { | ||
const expectedLogMessage = [ | ||
'0xbF3137b0a7574563a23a8fC8badC6537F98197CC', | ||
'test', | ||
'0.1.0', | ||
'1674172803', | ||
'1674172800', | ||
'0x6d4306f70c5fe9d8608b4e0c1d72e06a366e6f60b8461a6a9a0833a7401f5778', | ||
'0x6e6379a42b89bdc78286efd6f6ad94142765930c4843784f49ba955ea95c1cb64c432b41aecc1b220dc90d616332bbf7bdc7242a16eb0e31306dac96a6d6ee821b', | ||
].join(' - '); | ||
|
||
it('sends the correct heartbeat log', async () => { | ||
const state = stateModule.getInitialState(config); | ||
jest.spyOn(stateModule, 'getState').mockReturnValue(state); | ||
jest.spyOn(logger, 'info'); | ||
jest.advanceTimersByTime(1000 * 3); // Advance time by 3 seconds to ensure the timestamp of the log is different from deployment timestamp. | ||
|
||
await logHeartbeat(); | ||
|
||
expect(logger.info).toHaveBeenCalledWith(expectedLogMessage); | ||
}); | ||
|
||
it('the heartbeat log can be parsed', () => { | ||
const rawConfig = loadRawConfig(); | ||
const expectedHeartbeatPayload = { | ||
airnodeAddress: '0xbF3137b0a7574563a23a8fC8badC6537F98197CC', | ||
stage: 'test', | ||
nodeVersion: '0.1.0', | ||
heartbeatTimestamp: '1674172803', | ||
deploymentTimestamp: '1674172800', | ||
configHash: '0x6d4306f70c5fe9d8608b4e0c1d72e06a366e6f60b8461a6a9a0833a7401f5778', | ||
signature: | ||
'0x6e6379a42b89bdc78286efd6f6ad94142765930c4843784f49ba955ea95c1cb64c432b41aecc1b220dc90d616332bbf7bdc7242a16eb0e31306dac96a6d6ee821b', | ||
}; | ||
|
||
const heartbeatPayload = parseHeartbeatLog(expectedLogMessage); | ||
|
||
expect(heartbeatPayload).toStrictEqual(expectedHeartbeatPayload); | ||
expect(heartbeatPayload.configHash).toBe(createHash(JSON.stringify(rawConfig))); | ||
}); | ||
}); | ||
|
||
test('sends heartbeat payload every minute', async () => { | ||
// We would ideally want to assert that the logHeartbeat function is called, but spying on functions that are called | ||
// from the same module is annoying. See: https://jestjs.io/docs/mock-functions#mocking-partials. | ||
// | ||
// Instead we spyOn the "go" which is a third party module that wraps the logHeartbeat call. | ||
jest.spyOn(promiseUtilsModule, 'go'); | ||
|
||
initiateHeartbeat(); | ||
|
||
await jest.advanceTimersByTimeAsync(1000 * 60 * 8); | ||
expect(promiseUtilsModule.go).toHaveBeenCalledTimes(8); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
import { go } from '@api3/promise-utils'; | ||
import { ethers } from 'ethers'; | ||
|
||
import { logger } from '../logger'; | ||
import { getState } from '../state'; | ||
import { loadRawConfig } from '../validation/config'; | ||
|
||
export const initiateHeartbeat = () => { | ||
logger.debug('Initiating heartbeat loop'); | ||
setInterval(async () => { | ||
const goLogHeartbeat = await go(logHeartbeat); | ||
if (!goLogHeartbeat.success) logger.error('Failed to log heartbeat', goLogHeartbeat.error); | ||
}, 1000 * 60); // Frequency is hardcoded to 1 minute. | ||
}; | ||
|
||
export const signHeartbeat = async (airnodeWallet: ethers.Wallet, heartbeatPayload: unknown[]) => { | ||
logger.debug('Signing heartbeat payload'); | ||
const signaturePayload = ethers.utils.arrayify(createHash(JSON.stringify(heartbeatPayload))); | ||
return airnodeWallet.signMessage(signaturePayload); | ||
}; | ||
|
||
export const createHash = (value: string) => ethers.utils.keccak256(ethers.utils.toUtf8Bytes(value)); | ||
|
||
export const logHeartbeat = async () => { | ||
logger.debug('Creating heartbeat log'); | ||
|
||
const rawConfig = loadRawConfig(); // We want to log the raw config, not the one with interpolated secrets. | ||
const rawConfigHash = createHash(JSON.stringify(rawConfig)); | ||
const { | ||
airnodeWallet, | ||
deploymentTimestamp, | ||
config: { | ||
nodeSettings: { stage, nodeVersion }, | ||
}, | ||
} = getState(); | ||
|
||
logger.debug('Creating heartbeat payload'); | ||
const currentTimestamp = Math.floor(Date.now() / 1000); | ||
const heartbeatPayload = [ | ||
airnodeWallet.address, | ||
stage, | ||
nodeVersion, | ||
currentTimestamp.toString(), | ||
deploymentTimestamp.toString(), | ||
rawConfigHash, | ||
]; | ||
const heartbeatSignature = await signHeartbeat(airnodeWallet, heartbeatPayload); | ||
const heartbeatLog = [...heartbeatPayload, heartbeatSignature].join(' - '); | ||
|
||
// We ensure in config validation that INFO level logs are not disabled. The logs are sent to API3 for validation | ||
// (that the data provider deployed deployed the correct configuration) and monitoring purposes (whether the instance | ||
// is running). | ||
logger.info(heartbeatLog); | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export * from './heartbeat'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters