Skip to main content

Example Remote Nonce Cache Provider

This test code utilises Redis for remote caching.


danger

This is test code, use it at your own risk.

const { NonceType } = require('avn-api');
const Redis = require('ioredis');
const Redlock = require('redlock');

class TestRedisNonceCacheProvider {
SLOT_PREFIX = '{avnApi}:';
constructor() {
this.nonceMap = {};
this.redisClient;
}

async connect() {
this.redisClient = new Redis();

this.lockTtlMs = 5000;
this.redlock = new Redlock([this.redisClient], {
driftFactor: 0.01,
retryCount: 10,
retryDelay: 500,
retryJitter: 200
});

console.info(
'Connected to Redis database:\n',
(await this.redisClient.hello()).map((e, i) => (i % 2 == 0 ? e + ':' : e + ', ')).join('')
);

return this;
}

async initUserNonceCache(signerAddress) {
const signerKey = this.getSignerKey(signerAddress);
const isUserSetup = await this.redisClient.get(signerKey);
if (isUserSetup !== 'true' && isUserSetup !== true) {
// Set the signer
await this.redisClient.set(this.getSignerKey(signerAddress), true);
}
}

async getNonceData(signerAddress, nonceId) {
const nonceData = await this.readNonceDatafromRedis(signerAddress, nonceId);
return nonceData;
}

async getNonceAndLock(signerAddress, nonceId) {
const lock = await this.redlock.acquire([`redlock-${signerAddress}${nonceId}`], this.lockTtlMs);
try {
let nonceData = await this.readNonceDatafromRedis(signerAddress, nonceId);
if (nonceData.locked === true) {
return { lockAquired: false, data: nonceData };
}

const lockId = this.getLockId(signerAddress, nonceId, nonceData.nonce);
nonceData.locked = true;
nonceData.lockId = lockId;

await this.saveNonceDataToRedis(signerAddress, nonceId, nonceData);
return { lockAquired: true, data: nonceData };
} finally {
await lock.unlock();
}
}

async incrementNonce(lockId, signerAddress, nonceId, updateLastUpdate) {
let nonceData = await this.readNonceDatafromRedis(signerAddress, nonceId);
if (nonceData.locked !== true || nonceData.lockId !== lockId) {
throw new Error(
`Invalid attempt to increment nonce. Current lock: ${nonceData.lockId} LockId: ${lockId}, signerAddress: ${signerAddress}, nonceId: ${nonceId}`
);
}

nonceData.nonce = parseInt(nonceData.nonce) + 1;
if (updateLastUpdate === true) {
nonceData.lastUpdated = Date.now();
}

await this.saveNonceDataToRedis(signerAddress, nonceId, nonceData);
return nonceData;
}

async unlockNonce(lockId, signerAddress, nonceId) {
let nonceData = await this.readNonceDatafromRedis(signerAddress, nonceId);
if (nonceData.locked !== true || nonceData.lockId !== lockId) {
throw new Error(
`Invalid attempt to unlock nonce. Current lock: ${nonceData.lockId}. LockId: ${lockId}, signerAddress: ${signerAddress}, nonceId: ${nonceId}`
);
}
nonceData.locked = false;
nonceData.lockId = undefined;
await this.saveNonceDataToRedis(signerAddress, nonceId, nonceData);
}

async setNonce(lockId, signerAddress, nonceId, nonce) {
let nonceData = await this.readNonceDatafromRedis(signerAddress, nonceId);
if (nonceData.locked !== true || nonceData.lockId !== lockId) {
throw new Error(
`Invalid attempt to set nonce. Current lock: ${nonceData.lockId}. LockId: ${lockId}, signerAddress: ${signerAddress}, nonceId: ${nonceId}`
);
}

nonceData = { nonce: nonce, lastUpdated: Date.now() };
await this.saveNonceDataToRedis(signerAddress, nonceId, nonceData);
}

getLockId(signerAddress, nonceId, nonce) {
return `${Date.now()}-${nonce}-${signerAddress}-${nonceId}`;
}

getNonceKey(signerAddress, nonceId) {
return `${this.getSignerKey(signerAddress)}-${nonceId}`;
}

getSignerKey(signerAddress) {
return `${this.SLOT_PREFIX}-${signerAddress}`;
}

async readNonceDatafromRedis(signerAddress, nonceId) {
const nonceKey = this.getNonceKey(signerAddress, nonceId);
const nonceData = await this.redisClient.hgetall(nonceKey);
// deal with boolean values
nonceData.locked = nonceData.locked === 'true';
return nonceData;
}

async saveNonceDataToRedis(signerAddress, nonceId, nonceData) {
const nonceKey = this.getNonceKey(signerAddress, nonceId);
await this.redisClient.hset(nonceKey, nonceData);
}
}

module.exports = TestRedisNonceCacheProvider;