forked from cerc-io/laconic-faucet
Compare commits
2 Commits
pm-update-
...
main
Author | SHA1 | Date | |
---|---|---|---|
3649cfb2fa | |||
fd27cd141c |
@ -2,6 +2,8 @@ FROM node:20-bullseye
|
|||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
|
RUN apt-get update && apt-get install -y netcat
|
||||||
|
|
||||||
# Copy the application code
|
# Copy the application code
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
||||||
|
@ -1,14 +1,13 @@
|
|||||||
[registry]
|
[upstream]
|
||||||
rpcEndpoint = "http://localhost:26657"
|
rpcEndpoint = "http://localhost:26657"
|
||||||
chainId = "laconic_9000-1"
|
chainId = "laconic_9000-1"
|
||||||
denom = "photon"
|
denom = "alnt"
|
||||||
prefix = "laconic"
|
prefix = "laconic"
|
||||||
gasPrice = "0.01"
|
gasPrice = "1"
|
||||||
faucetKey = ""
|
faucetKey = ""
|
||||||
|
|
||||||
[server]
|
[server]
|
||||||
port = 3000
|
port = 3000
|
||||||
periodInSecs = 60
|
transferAmount = 10000000000 # 1 * 10^10 alnt
|
||||||
transferAmount = 1000000
|
periodTransferLimit = 30000000000 # 3 * 10^10 alnt
|
||||||
periodTransferLimit = 3000000
|
|
||||||
dbDir = "db"
|
dbDir = "db"
|
||||||
|
54
src/index.ts
54
src/index.ts
@ -12,10 +12,10 @@ import KeyvSqlite from '@keyv/sqlite';
|
|||||||
|
|
||||||
const CONFIG_PATH = 'environments/local.toml';
|
const CONFIG_PATH = 'environments/local.toml';
|
||||||
const FAUCET_DATA_FILE = 'faucet_data.sqlite';
|
const FAUCET_DATA_FILE = 'faucet_data.sqlite';
|
||||||
const TTL_MULTIPLIER = 1.5; // 1.5 times the configured period
|
const FAUCET_DATA_TTL = 86400000; // 24 hrs
|
||||||
|
|
||||||
interface Config {
|
interface Config {
|
||||||
registry: {
|
upstream: {
|
||||||
rpcEndpoint: string
|
rpcEndpoint: string
|
||||||
chainId: string
|
chainId: string
|
||||||
denom: string
|
denom: string
|
||||||
@ -25,18 +25,12 @@ interface Config {
|
|||||||
},
|
},
|
||||||
server: {
|
server: {
|
||||||
port: number
|
port: number
|
||||||
periodInSecs: number
|
|
||||||
transferAmount: string
|
transferAmount: string
|
||||||
periodTransferLimit: string
|
periodTransferLimit: string
|
||||||
dbDir: string
|
dbDir: string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
interface FaucetStoreValue {
|
|
||||||
periodStart: string
|
|
||||||
amountSent: string
|
|
||||||
}
|
|
||||||
|
|
||||||
async function main (): Promise<void> {
|
async function main (): Promise<void> {
|
||||||
// Read and parse the configuration
|
// Read and parse the configuration
|
||||||
const configFile = fs.readFileSync(CONFIG_PATH, 'utf-8');
|
const configFile = fs.readFileSync(CONFIG_PATH, 'utf-8');
|
||||||
@ -56,32 +50,18 @@ async function main (): Promise<void> {
|
|||||||
return res.status(400).json({ error: 'address is required' });
|
return res.status(400).json({ error: 'address is required' });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isValidAddress(accountAddress, config.registry.prefix)) {
|
if (!isValidAddress(accountAddress, config.upstream.prefix)) {
|
||||||
return res.status(400).json({ error: 'invalid address' });
|
return res.status(400).json({ error: 'invalid address' });
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentTimeInSeconds = Math.floor(Date.now() / 1000);
|
|
||||||
let amountSentToAccount = BigInt(0);
|
|
||||||
let accountPeriodStart = currentTimeInSeconds;
|
|
||||||
|
|
||||||
// Get accounts data from faucet data store
|
|
||||||
const accountFaucetData: FaucetStoreValue = await faucetDataStore.get(accountAddress);
|
|
||||||
|
|
||||||
// Check rate limit
|
// Check rate limit
|
||||||
if (accountFaucetData !== undefined) {
|
const now = Date.now();
|
||||||
const elapsedTime = currentTimeInSeconds - Number(accountFaucetData.periodStart);
|
const today = new Date(now).toISOString().split('T')[0];
|
||||||
|
const faucetStoreKey = `${accountAddress}:${today}`;
|
||||||
|
const amountSentToAddress = await faucetDataStore.get(faucetStoreKey) || '0';
|
||||||
|
|
||||||
// Check if saved entry is still within configured period
|
if (BigInt(amountSentToAddress) + BigInt(config.server.transferAmount) > BigInt(config.server.periodTransferLimit)) {
|
||||||
// Else, create a fresh entry
|
return res.status(429).json({ error: 'Limit exceeded' });
|
||||||
if (elapsedTime < config.server.periodInSecs) {
|
|
||||||
amountSentToAccount = BigInt(accountFaucetData.amountSent);
|
|
||||||
accountPeriodStart = Number(accountFaucetData.periodStart);
|
|
||||||
|
|
||||||
// Check if limit has been reached
|
|
||||||
if (amountSentToAccount + BigInt(config.server.transferAmount) > BigInt(config.server.periodTransferLimit)) {
|
|
||||||
return res.status(429).json({ error: 'Limit exceeded' });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@ -89,11 +69,7 @@ async function main (): Promise<void> {
|
|||||||
console.log(`Sent tokens to address: ${accountAddress}, txHash: ${txHash}`);
|
console.log(`Sent tokens to address: ${accountAddress}, txHash: ${txHash}`);
|
||||||
|
|
||||||
// Update rate limit
|
// Update rate limit
|
||||||
const updatedAccountFaucetData: FaucetStoreValue = {
|
await faucetDataStore.set(faucetStoreKey, (BigInt(amountSentToAddress) + BigInt(config.server.transferAmount)).toString(), FAUCET_DATA_TTL);
|
||||||
periodStart: accountPeriodStart.toString(),
|
|
||||||
amountSent: amountSentToAccount.toString()
|
|
||||||
};
|
|
||||||
await faucetDataStore.set(accountAddress, updatedAccountFaucetData, config.server.periodInSecs * TTL_MULTIPLIER * 1000);
|
|
||||||
|
|
||||||
res.json({ success: true, txHash });
|
res.json({ success: true, txHash });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -132,27 +108,27 @@ async function initKVStore (dbDir: string): Promise<Keyv> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function sendTokens (config: Config, recipientAddress: string, amount: string): Promise<string> {
|
async function sendTokens (config: Config, recipientAddress: string, amount: string): Promise<string> {
|
||||||
let faucetKey = config.registry.faucetKey;
|
let faucetKey = config.upstream.faucetKey;
|
||||||
if (faucetKey.startsWith('0x')) {
|
if (faucetKey.startsWith('0x')) {
|
||||||
faucetKey = faucetKey.slice(2);
|
faucetKey = faucetKey.slice(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
const wallet = await DirectSecp256k1Wallet.fromKey(
|
const wallet = await DirectSecp256k1Wallet.fromKey(
|
||||||
Buffer.from(faucetKey, 'hex'),
|
Buffer.from(faucetKey, 'hex'),
|
||||||
config.registry.prefix
|
config.upstream.prefix
|
||||||
);
|
);
|
||||||
const [faucetAccount] = await wallet.getAccounts();
|
const [faucetAccount] = await wallet.getAccounts();
|
||||||
|
|
||||||
const client = await SigningStargateClient.connectWithSigner(
|
const client = await SigningStargateClient.connectWithSigner(
|
||||||
config.registry.rpcEndpoint,
|
config.upstream.rpcEndpoint,
|
||||||
wallet,
|
wallet,
|
||||||
{ gasPrice: GasPrice.fromString(`${config.registry.gasPrice}${config.registry.denom}`) }
|
{ gasPrice: GasPrice.fromString(`${config.upstream.gasPrice}${config.upstream.denom}`) }
|
||||||
);
|
);
|
||||||
|
|
||||||
const result = await client.sendTokens(
|
const result = await client.sendTokens(
|
||||||
faucetAccount.address,
|
faucetAccount.address,
|
||||||
recipientAddress,
|
recipientAddress,
|
||||||
[{ denom: config.registry.denom, amount: amount }],
|
[{ denom: config.upstream.denom, amount: amount }],
|
||||||
'auto',
|
'auto',
|
||||||
'Faucet transfer'
|
'Faucet transfer'
|
||||||
);
|
);
|
||||||
|
Loading…
Reference in New Issue
Block a user