/*
* Copyright (c) [2016] [ <ether.camp> ]
* This file is part of the ethereumJ library.
*
* The ethereumJ library is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* The ethereumJ library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with the ethereumJ library. If not, see <http://www.gnu.org/licenses/>.
*/
package org.ethereum.listener;
import org.ethereum.core.BlockSummary;
import org.ethereum.core.Transaction;
import org.ethereum.core.TransactionExecutionSummary;
import org.ethereum.util.ByteUtil;
import java.util.Arrays;
/**
* Calculates a 'reasonable' Gas price based on statistics of the latest transaction's Gas prices
*
* Normally the price returned should be sufficient to execute a transaction since ~25% of the latest
* transactions were executed at this or lower price.
*
* Created by Anton Nashatyrev on 22.09.2015.
*/
public class GasPriceTracker extends EthereumListenerAdapter {
private static final long defaultPrice = 70_000_000_000L;
private long[] window = new long[512];
private int idx = window.length - 1;
private boolean filled = false;
private long lastVal;
@Override
public void onBlock(BlockSummary blockSummary) {
for (Transaction tx : blockSummary.getBlock().getTransactionsList()) {
onTransaction(tx);
}
}
public void onTransaction(Transaction tx) {
if (idx == -1) {
idx = window.length - 1;
filled = true;
lastVal = 0; // recalculate only 'sometimes'
}
window[idx--] = ByteUtil.byteArrayToLong(tx.getGasPrice());
}
public long getGasPrice() {
if (!filled) {
return defaultPrice;
} else {
if (lastVal == 0) {
long[] longs = Arrays.copyOf(window, window.length);
Arrays.sort(longs);
lastVal = longs[longs.length / 4]; // 25% percentile
}
return lastVal;
}
}
}