Compare commits
18 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 0e72d68d94 | |||
| 170d0c0ec9 | |||
| 5548c6de53 | |||
| 56ea8f3792 | |||
| 4d15a85e70 | |||
| 494e6aa059 | |||
| 5d33cf7e90 | |||
| f4ff9bc8cb | |||
| acc0039b14 | |||
| 57d2f65800 | |||
| be85a60436 | |||
| 1f0941edd8 | |||
| 626fd45757 | |||
| dd7fc1aa05 | |||
| f5ccc22d2c | |||
| 22f9cf0bca | |||
| 8f3052679a | |||
| 0d0da4af7b |
@@ -7,6 +7,8 @@ const varuint = require('varuint-bitcoin');
|
|||||||
const crypto = require('crypto');
|
const crypto = require('crypto');
|
||||||
const fastMerkleRoot = require('merkle-lib/fastRoot');
|
const fastMerkleRoot = require('merkle-lib/fastRoot');
|
||||||
|
|
||||||
|
const rtm = require('cryptoforknote-util/rtm');
|
||||||
|
|
||||||
function scriptCompile(addrHash) {
|
function scriptCompile(addrHash) {
|
||||||
return bitcoin.script.compile([
|
return bitcoin.script.compile([
|
||||||
bitcoin.opcodes.OP_DUP,
|
bitcoin.opcodes.OP_DUP,
|
||||||
@@ -18,7 +20,7 @@ function scriptCompile(addrHash) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function reverseBuffer(buff) {
|
function reverseBuffer(buff) {
|
||||||
let reversed = new Buffer(buff.length);
|
let reversed = Buffer.alloc(buff.length);
|
||||||
for (var i = buff.length - 1; i >= 0; i--) reversed[buff.length - i - 1] = buff[i];
|
for (var i = buff.length - 1; i >= 0; i--) reversed[buff.length - i - 1] = buff[i];
|
||||||
return reversed;
|
return reversed;
|
||||||
}
|
}
|
||||||
@@ -45,7 +47,7 @@ function hash256(buffer) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
function getMerkleRoot(transactions) {
|
function getMerkleRoot(transactions) {
|
||||||
if (transactions.length === 0) return new Buffer('0000000000000000000000000000000000000000000000000000000000000000', 'hex')
|
if (transactions.length === 0) return Buffer.from('0000000000000000000000000000000000000000000000000000000000000000', 'hex')
|
||||||
const forWitness = txesHaveWitnessCommit(transactions);
|
const forWitness = txesHaveWitnessCommit(transactions);
|
||||||
const hashes = transactions.map(transaction => transaction.getHash(forWitness));
|
const hashes = transactions.map(transaction => transaction.getHash(forWitness));
|
||||||
const rootHash = fastMerkleRoot(hashes, hash256);
|
const rootHash = fastMerkleRoot(hashes, hash256);
|
||||||
@@ -75,27 +77,27 @@ module.exports.RavenBlockTemplate = function(rpcData, poolAddress) {
|
|||||||
bytesHeight = Math.ceil((rpcData.height << 1).toString(2).length / 8);
|
bytesHeight = Math.ceil((rpcData.height << 1).toString(2).length / 8);
|
||||||
const lengthDiff = blockHeightSerial.length/2 - bytesHeight;
|
const lengthDiff = blockHeightSerial.length/2 - bytesHeight;
|
||||||
for (let i = 0; i < lengthDiff; i++) blockHeightSerial = blockHeightSerial + '00';
|
for (let i = 0; i < lengthDiff; i++) blockHeightSerial = blockHeightSerial + '00';
|
||||||
const serializedBlockHeight = new Buffer.concat([
|
const serializedBlockHeight = Buffer.concat([
|
||||||
new Buffer('0' + bytesHeight, 'hex'),
|
Buffer.from('0' + bytesHeight, 'hex'),
|
||||||
reverseBuffer(new Buffer(blockHeightSerial, 'hex')),
|
reverseBuffer(Buffer.from(blockHeightSerial, 'hex')),
|
||||||
new Buffer('00', 'hex') // OP_0
|
Buffer.from('00', 'hex') // OP_0
|
||||||
]);
|
]);
|
||||||
|
|
||||||
txCoinbase.addInput(
|
txCoinbase.addInput(
|
||||||
// will be used for our reserved_offset extra_nonce
|
// will be used for our reserved_offset extra_nonce
|
||||||
new Buffer('0000000000000000000000000000000000000000000000000000000000000000', 'hex'),
|
Buffer.from('0000000000000000000000000000000000000000000000000000000000000000', 'hex'),
|
||||||
0xFFFFFFFF, 0xFFFFFFFF,
|
0xFFFFFFFF, 0xFFFFFFFF,
|
||||||
new Buffer.concat([serializedBlockHeight, Buffer('CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC', 'hex')]) // 17 bytes
|
Buffer.concat([serializedBlockHeight, Buffer.alloc(17, 0xCC)]) // 17 bytes
|
||||||
);
|
);
|
||||||
|
|
||||||
txCoinbase.addOutput(scriptCompile(poolAddrHash), Math.floor(rpcData.coinbasevalue));
|
txCoinbase.addOutput(scriptCompile(poolAddrHash), Math.floor(rpcData.coinbasevalue));
|
||||||
|
|
||||||
if (rpcData.default_witness_commitment) {
|
if (rpcData.default_witness_commitment) {
|
||||||
txCoinbase.addOutput(new Buffer(rpcData.default_witness_commitment, 'hex'), 0);
|
txCoinbase.addOutput(Buffer.from(rpcData.default_witness_commitment, 'hex'), 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let header = new Buffer(80);
|
let header = Buffer.alloc(80);
|
||||||
{ let position = 0;
|
{ let position = 0;
|
||||||
header.writeUInt32BE(rpcData.height, position, 4); // height 42-46
|
header.writeUInt32BE(rpcData.height, position, 4); // height 42-46
|
||||||
header.write(rpcData.bits, position += 4, 4, 'hex'); // bits 47-50
|
header.write(rpcData.bits, position += 4, 4, 'hex'); // bits 47-50
|
||||||
@@ -106,17 +108,17 @@ module.exports.RavenBlockTemplate = function(rpcData, poolAddress) {
|
|||||||
header = reverseBuffer(header);
|
header = reverseBuffer(header);
|
||||||
}
|
}
|
||||||
|
|
||||||
let blob = new Buffer.concat([
|
let blob = Buffer.concat([
|
||||||
header, // 80 bytes
|
header, // 80 bytes
|
||||||
new Buffer('AAAAAAAAAAAAAAAA', 'hex'), // 8 bytes
|
Buffer.from('AAAAAAAAAAAAAAAA', 'hex'), // 8 bytes
|
||||||
new Buffer('BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB', 'hex'), // 32 bytes
|
Buffer.from('BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB', 'hex'), // 32 bytes
|
||||||
varuint.encode(rpcData.transactions.length + 1, new Buffer(varuint.encodingLength(rpcData.transactions.length + 1)), 0)
|
varuint.encode(rpcData.transactions.length + 1, Buffer.alloc(varuint.encodingLength(rpcData.transactions.length + 1)), 0)
|
||||||
]);
|
]);
|
||||||
const offset1 = blob.length;
|
const offset1 = blob.length;
|
||||||
blob = new Buffer.concat([ blob, new Buffer(txCoinbase.toHex(), 'hex') ]);
|
blob = Buffer.concat([ blob, Buffer.from(txCoinbase.toHex(), 'hex') ]);
|
||||||
|
|
||||||
rpcData.transactions.forEach(function (value) {
|
rpcData.transactions.forEach(function (value) {
|
||||||
blob = new Buffer.concat([ blob, new Buffer(value.data, 'hex') ]);
|
blob = Buffer.concat([ blob, Buffer.from(value.data, 'hex') ]);
|
||||||
});
|
});
|
||||||
|
|
||||||
const EPOCH_LENGTH = 7500;
|
const EPOCH_LENGTH = 7500;
|
||||||
@@ -126,7 +128,7 @@ module.exports.RavenBlockTemplate = function(rpcData, poolAddress) {
|
|||||||
if (last_epoch_number && last_epoch_number + 1 === epoch_number) {
|
if (last_epoch_number && last_epoch_number + 1 === epoch_number) {
|
||||||
last_seed_hash = sha3.update(last_seed_hash).digest();
|
last_seed_hash = sha3.update(last_seed_hash).digest();
|
||||||
} else {
|
} else {
|
||||||
last_seed_hash = new Buffer(32, 0);
|
last_seed_hash = Buffer.alloc(32, 0);
|
||||||
for (let i = 0; i < epoch_number; i++) {
|
for (let i = 0; i < epoch_number; i++) {
|
||||||
last_seed_hash = sha3.update(last_seed_hash).digest();
|
last_seed_hash = sha3.update(last_seed_hash).digest();
|
||||||
sha3.reset();
|
sha3.reset();
|
||||||
@@ -150,27 +152,30 @@ module.exports.RavenBlockTemplate = function(rpcData, poolAddress) {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
function update_merkle_root_hash(blob_in, blob_out) {
|
function update_merkle_root_hash(offset, payload, blob_in, blob_out) {
|
||||||
let offset = 80 + 8 + 32;
|
|
||||||
const nTransactions = varuint.decode(blob_in, offset);
|
const nTransactions = varuint.decode(blob_in, offset);
|
||||||
offset += varuint.decode.bytes;
|
offset += varuint.decode.bytes;
|
||||||
let transactions = [];
|
let transactions = [];
|
||||||
for (let i = 0; i < nTransactions; ++i) {
|
for (let i = 0; i < nTransactions; ++i) {
|
||||||
const tx = bitcoin.Transaction.fromBuffer(blob_in.slice(offset), true);
|
const tx = bitcoin.Transaction.fromBuffer(blob_in.slice(offset), true, payload && i == 0);
|
||||||
transactions.push(tx);
|
transactions.push(tx);
|
||||||
offset += tx.byteLength();
|
offset += tx.byteLength();
|
||||||
}
|
}
|
||||||
getMerkleRoot(transactions).copy(blob_out, 4 + 32);
|
getMerkleRoot(transactions).copy(blob_out, 4 + 32);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
module.exports.blockHashBuff = function(blobBuffer) {
|
||||||
|
return reverseBuffer(hash256(blobBuffer));
|
||||||
|
};
|
||||||
|
|
||||||
module.exports.convertRavenBlob = function(blobBuffer) {
|
module.exports.convertRavenBlob = function(blobBuffer) {
|
||||||
let header = blobBuffer.slice(0, 80);
|
let header = blobBuffer.slice(0, 80);
|
||||||
update_merkle_root_hash(blobBuffer, header);
|
update_merkle_root_hash(80 + 8 + 32, false, blobBuffer, header);
|
||||||
return reverseBuffer(hash256(header));
|
return module.exports.blockHashBuff(header);
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports.constructNewRavenBlob = function(blockTemplate, nonceBuff, mixhashBuff) {
|
module.exports.constructNewRavenBlob = function(blockTemplate, nonceBuff, mixhashBuff) {
|
||||||
update_merkle_root_hash(blockTemplate, blockTemplate);
|
update_merkle_root_hash(80 + 8 + 32, false, blockTemplate, blockTemplate);
|
||||||
nonceBuff.copy (blockTemplate, 80, 0, 8);
|
nonceBuff.copy (blockTemplate, 80, 0, 8);
|
||||||
mixhashBuff.copy(blockTemplate, 88, 0, 32);
|
mixhashBuff.copy(blockTemplate, 88, 0, 32);
|
||||||
return blockTemplate;
|
return blockTemplate;
|
||||||
@@ -199,4 +204,20 @@ module.exports.ErgBlockTemplate = function(rpcData) {
|
|||||||
difficulty: difficulty,
|
difficulty: difficulty,
|
||||||
height: parseInt(rpcData.h)
|
height: parseInt(rpcData.h)
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
module.exports.RtmBlockTemplate = function(rpcData, poolAddress) {
|
||||||
|
return rtm.RtmBlockTemplate(rpcData, poolAddress);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports.convertRtmBlob = function(blobBuffer) {
|
||||||
|
let header = blobBuffer.slice(0, 80);
|
||||||
|
update_merkle_root_hash(80, true, blobBuffer, header);
|
||||||
|
return header;
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports.constructNewRtmBlob = function(blockTemplate, nonceBuff) {
|
||||||
|
update_merkle_root_hash(80, true, blockTemplate, blockTemplate);
|
||||||
|
nonceBuff.copy(blockTemplate, 76, 0, 4);
|
||||||
|
return blockTemplate;
|
||||||
|
};
|
||||||
|
|||||||
+4
-2
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "cryptoforknote-util",
|
"name": "cryptoforknote-util",
|
||||||
"version": "10.0.3",
|
"version": "11.1.0",
|
||||||
"main": "cryptoforknote-util",
|
"main": "cryptoforknote-util",
|
||||||
"author": {
|
"author": {
|
||||||
"name": "LucasJones",
|
"name": "LucasJones",
|
||||||
@@ -16,8 +16,10 @@
|
|||||||
"nan": "^2.14.2",
|
"nan": "^2.14.2",
|
||||||
"bignum": "^0.13.1",
|
"bignum": "^0.13.1",
|
||||||
"sha3": "*",
|
"sha3": "*",
|
||||||
|
"base58-native": "*",
|
||||||
"varuint-bitcoin": "^1.0.4",
|
"varuint-bitcoin": "^1.0.4",
|
||||||
"bitcoinjs-lib": "git+https://github.com/bitcoinjs/bitcoinjs-lib.git#533d6c2e6d0aa4111f7948b1c12003cf6ef83137"
|
"merkle-lib": "^2.0.10",
|
||||||
|
"bitcoinjs-lib": "git+https://github.com/MoneroOcean/bitcoinjs-lib.git"
|
||||||
},
|
},
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"cryptonight",
|
"cryptonight",
|
||||||
|
|||||||
@@ -0,0 +1,267 @@
|
|||||||
|
const bignum = require('bignum');
|
||||||
|
const base58 = require('base58-native');
|
||||||
|
|
||||||
|
const diff1 = 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF;
|
||||||
|
|
||||||
|
function reverseBuffer(buff) {
|
||||||
|
let reversed = Buffer.alloc(buff.length);
|
||||||
|
for (let i = buff.length - 1; i >= 0; i--) reversed[buff.length - i - 1] = buff[i];
|
||||||
|
return reversed;
|
||||||
|
}
|
||||||
|
|
||||||
|
function reverseByteOrder(buff) {
|
||||||
|
for (let i = 0; i < 8; i++) buff.writeUInt32LE(buff.readUInt32BE(i * 4), i * 4);
|
||||||
|
return reverseBuffer(buff);
|
||||||
|
}
|
||||||
|
|
||||||
|
function packInt32LE(num) {
|
||||||
|
let buff = Buffer.alloc(4);
|
||||||
|
buff.writeInt32LE(num, 0);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
|
||||||
|
function packInt32BE(num) {
|
||||||
|
let buff = Buffer.alloc(4);
|
||||||
|
buff.writeInt32BE(num, 0);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
|
||||||
|
function packUInt16LE(num) {
|
||||||
|
let buff = Buffer.alloc(2);
|
||||||
|
buff.writeUInt16LE(num, 0);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
|
||||||
|
function packUInt32LE(num) {
|
||||||
|
let buff = Buffer.alloc(4);
|
||||||
|
buff.writeUInt32LE(num, 0);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
|
||||||
|
function packUInt32BE(num) {
|
||||||
|
let buff = Buffer.alloc(4);
|
||||||
|
buff.writeUInt32BE(num, 0);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
|
||||||
|
function packInt64LE(num){
|
||||||
|
let buff = Buffer.alloc(8);
|
||||||
|
buff.writeUInt32LE(num % Math.pow(2, 32), 0);
|
||||||
|
buff.writeUInt32LE(Math.floor(num / Math.pow(2, 32)), 4);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Defined in bitcoin protocol here:
|
||||||
|
// https://en.bitcoin.it/wiki/Protocol_specification#Variable_length_integer
|
||||||
|
function varIntBuffer(n) {
|
||||||
|
if (n < 0xfd) {
|
||||||
|
return Buffer.from([n]);
|
||||||
|
} else if (n <= 0xffff) {
|
||||||
|
let buff = Buffer.alloc(3);
|
||||||
|
buff[0] = 0xfd;
|
||||||
|
buff.writeUInt16LE(n, 1);
|
||||||
|
return buff;
|
||||||
|
} else if (n <= 0xffffffff) {
|
||||||
|
let buff = Buffer.alloc(5);
|
||||||
|
buff[0] = 0xfe;
|
||||||
|
buff.writeUInt32LE(n, 1);
|
||||||
|
return buff;
|
||||||
|
} else{
|
||||||
|
let buff = Buffer.alloc(9);
|
||||||
|
buff[0] = 0xff;
|
||||||
|
packUInt16LE(n).copy(buff, 1);
|
||||||
|
return buff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// "serialized CScript" formatting as defined here:
|
||||||
|
// https://github.com/bitcoin/bips/blob/master/bip-0034.mediawiki#specification
|
||||||
|
// Used to format height and date when putting into script signature:
|
||||||
|
// https://en.bitcoin.it/wiki/Script
|
||||||
|
function serializeNumber(n) {
|
||||||
|
// New version from TheSeven
|
||||||
|
if (n >= 1 && n <= 16) return Buffer.from([0x50 + n]);
|
||||||
|
var l = 1;
|
||||||
|
var buff = Buffer.alloc(9);
|
||||||
|
while (n > 0x7f) {
|
||||||
|
buff.writeUInt8(n & 0xff, l++);
|
||||||
|
n >>= 8;
|
||||||
|
}
|
||||||
|
buff.writeUInt8(l, 0);
|
||||||
|
buff.writeUInt8(n, l++);
|
||||||
|
return buff.slice(0, l);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Used for serializing strings used in script signature
|
||||||
|
function serializeString(s) {
|
||||||
|
if (s.length < 253) {
|
||||||
|
return Buffer.concat([ Buffer.from([s.length]), Buffer.from(s) ]);
|
||||||
|
} else if (s.length < 0x10000) {
|
||||||
|
return Buffer.concat([ Buffer.from([253]), packUInt16LE(s.length), Buffer.from(s) ]);
|
||||||
|
} else if (s.length < 0x100000000) {
|
||||||
|
return Buffer.concat([ Buffer.from([254]), packUInt32LE(s.length), Buffer.from(s) ]);
|
||||||
|
} else {
|
||||||
|
return Buffer.concat([ Buffer.from([255]), packUInt16LE(s.length), Buffer.from(s) ]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// An exact copy of python's range feature. Written by Tadeck:
|
||||||
|
// http://stackoverflow.com/a/8273091
|
||||||
|
function range(start, stop, step) {
|
||||||
|
if (typeof stop === 'undefined') {
|
||||||
|
stop = start;
|
||||||
|
start = 0;
|
||||||
|
}
|
||||||
|
if (typeof step === 'undefined') {
|
||||||
|
step = 1;
|
||||||
|
}
|
||||||
|
if ((step > 0 && start >= stop) || (step < 0 && start <= stop)) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
let result = [];
|
||||||
|
for (let i = start; step > 0 ? i < stop : i > stop; i += step) {
|
||||||
|
result.push(i);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
function uint256BufferFromHash(hex) {
|
||||||
|
let fromHex = Buffer.from(hex, 'hex');
|
||||||
|
if (fromHex.length != 32) {
|
||||||
|
let empty = Buffer.alloc(32);
|
||||||
|
empty.fill(0);
|
||||||
|
fromHex.copy(empty);
|
||||||
|
fromHex = empty;
|
||||||
|
}
|
||||||
|
return reverseBuffer(fromHex);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTransactionBuffers(txs) {
|
||||||
|
let txHashes = txs.map(function(tx) {
|
||||||
|
if (tx.txid !== undefined) {
|
||||||
|
return uint256BufferFromHash(tx.txid);
|
||||||
|
}
|
||||||
|
return uint256BufferFromHash(tx.hash);
|
||||||
|
});
|
||||||
|
return [null].concat(txHashes);
|
||||||
|
}
|
||||||
|
|
||||||
|
function addressToScript(addr) {
|
||||||
|
const decoded = base58.decode(addr);
|
||||||
|
if (decoded.length != 25) throw new Error('Invalid address length for ' + addr);
|
||||||
|
if (!decoded) throw new Error('Base58 decode failed for ' + addr);
|
||||||
|
const pubkey = decoded.slice(1, -4);
|
||||||
|
return Buffer.concat([Buffer.from([0x76, 0xa9, 0x14]), pubkey, Buffer.from([0x88, 0xac])]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function createOutputTransaction(amount, payee, rewardToPool, reward, txOutputBuffers, payeeScript) {
|
||||||
|
const payeeReward = amount;
|
||||||
|
if (!payeeScript) payeeScript = addressToScript(payee);
|
||||||
|
txOutputBuffers.push(Buffer.concat([
|
||||||
|
packInt64LE(payeeReward),
|
||||||
|
varIntBuffer(payeeScript.length),
|
||||||
|
payeeScript
|
||||||
|
]));
|
||||||
|
return { reward: reward - amount, rewardToPool: rewardToPool - amount };
|
||||||
|
}
|
||||||
|
|
||||||
|
function generateOutputTransactions(rpcData, poolAddress) {
|
||||||
|
let reward = rpcData.coinbasevalue;
|
||||||
|
let rewardToPool = reward;
|
||||||
|
let txOutputBuffers = [];
|
||||||
|
|
||||||
|
if (rpcData.smartnode) {
|
||||||
|
if (rpcData.smartnode.payee) {
|
||||||
|
const rewards = createOutputTransaction(rpcData.smartnode.amount, rpcData.smartnode.payee, rewardToPool, reward, txOutputBuffers);
|
||||||
|
reward = rewards.reward;
|
||||||
|
rewardToPool = rewards.rewardToPool;
|
||||||
|
} else if (Array.isArray(rpcData.smartnode)) {
|
||||||
|
for (let i in rpcData.smartnode) {
|
||||||
|
const rewards = createOutputTransaction(rpcData.smartnode[i].amount, rpcData.smartnode[i].payee, rewardToPool, reward, txOutputBuffers);
|
||||||
|
reward = rewards.reward;
|
||||||
|
rewardToPool = rewards.rewardToPool;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rpcData.superblock) {
|
||||||
|
for (let i in rpcData.superblock) {
|
||||||
|
const rewards = createOutputTransaction(rpcData.superblock[i].amount, rpcData.superblock[i].payee, rewardToPool, reward, txOutputBuffers);
|
||||||
|
reward = rewards.reward;
|
||||||
|
rewardToPool = rewards.rewardToPool;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rpcData.founder_payments_started && rpcData.founder) {
|
||||||
|
const founderReward = rpcData.founder.amount || 0;
|
||||||
|
const rewards = createOutputTransaction(founderReward, rpcData.founder.payee, rewardToPool, reward, txOutputBuffers);
|
||||||
|
reward = rewards.reward;
|
||||||
|
rewardToPool = rewards.rewardToPool;
|
||||||
|
}
|
||||||
|
|
||||||
|
createOutputTransaction(rewardToPool, null, rewardToPool, reward, txOutputBuffers, Buffer.from(addressToScript(poolAddress), "hex"));
|
||||||
|
|
||||||
|
if (rpcData.default_witness_commitment !== undefined) {
|
||||||
|
const witness_commitment = Buffer.from(rpcData.default_witness_commitment, 'hex');
|
||||||
|
txOutputBuffers.unshift(Buffer.concat([
|
||||||
|
packInt64LE(0),
|
||||||
|
varIntBuffer(witness_commitment.length),
|
||||||
|
witness_commitment
|
||||||
|
]));
|
||||||
|
}
|
||||||
|
|
||||||
|
return Buffer.concat([ varIntBuffer(txOutputBuffers.length), Buffer.concat(txOutputBuffers)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports.RtmBlockTemplate = function(rpcData, poolAddress) {
|
||||||
|
const extraNoncePlaceholderLength = 17;
|
||||||
|
const coinbaseVersion = Buffer.concat([packUInt16LE(3), packUInt16LE(5)]);
|
||||||
|
|
||||||
|
const scriptSigPart1 = Buffer.concat([
|
||||||
|
serializeNumber(rpcData.height),
|
||||||
|
Buffer.from(rpcData.coinbaseaux.flags, 'hex'),
|
||||||
|
serializeNumber(Date.now() / 1000 | 0),
|
||||||
|
Buffer.from([extraNoncePlaceholderLength])
|
||||||
|
]);
|
||||||
|
|
||||||
|
const scriptSigPart2 = serializeString('/nodeStratum/');
|
||||||
|
|
||||||
|
const blob1 = Buffer.concat([
|
||||||
|
coinbaseVersion,
|
||||||
|
// transaction input
|
||||||
|
varIntBuffer(1), // txInputsCount
|
||||||
|
uint256BufferFromHash(""), // txInPrevOutHash
|
||||||
|
packUInt32LE(Math.pow(2, 32) - 1), // txInPrevOutIndex
|
||||||
|
varIntBuffer(scriptSigPart1.length + extraNoncePlaceholderLength + scriptSigPart2.length),
|
||||||
|
scriptSigPart1
|
||||||
|
]);
|
||||||
|
|
||||||
|
let blob2 = Buffer.concat([
|
||||||
|
scriptSigPart2,
|
||||||
|
packUInt32LE(0), // txInSequence
|
||||||
|
// end transaction input
|
||||||
|
// transaction output
|
||||||
|
generateOutputTransactions(rpcData, poolAddress),
|
||||||
|
// end transaction ouput
|
||||||
|
packUInt32LE(0), // txLockTime
|
||||||
|
varIntBuffer(rpcData.coinbase_payload.length / 2),
|
||||||
|
Buffer.from(rpcData.coinbase_payload, 'hex')
|
||||||
|
]);
|
||||||
|
|
||||||
|
const prev_hash = reverseBuffer(Buffer.from(rpcData.previousblockhash, 'hex')).toString('hex');
|
||||||
|
const version = packInt32LE(rpcData.version).toString('hex');
|
||||||
|
const curtime = packUInt32LE(rpcData.curtime).toString('hex');
|
||||||
|
let bits = Buffer.from(rpcData.bits, 'hex');
|
||||||
|
bits.writeUInt32LE(bits.readUInt32BE());
|
||||||
|
const txn = varIntBuffer(rpcData.transactions.length + 1);
|
||||||
|
|
||||||
|
return {
|
||||||
|
difficulty: parseFloat((diff1 / bignum(rpcData.target, 16).toNumber()).toFixed(9)),
|
||||||
|
height: rpcData.height,
|
||||||
|
prev_hash: prev_hash,
|
||||||
|
blocktemplate_blob: version + prev_hash + Buffer.alloc(32, 0).toString('hex') + curtime + bits.toString('hex') + Buffer.alloc(4, 0).toString('hex') +
|
||||||
|
txn.toString('hex') + blob1.toString('hex') + Buffer.alloc(extraNoncePlaceholderLength, 0xCC).toString('hex') + blob2.toString('hex') +
|
||||||
|
Buffer.concat(rpcData.transactions.map(function(tx) { return Buffer.from(tx.data, 'hex'); })).toString('hex'),
|
||||||
|
reserved_offset: 80 + txn.length + blob1.length
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -37,6 +37,7 @@
|
|||||||
#include "portable_storage_val_converters.h"
|
#include "portable_storage_val_converters.h"
|
||||||
#include "span.h"
|
#include "span.h"
|
||||||
#include "int-util.h"
|
#include "int-util.h"
|
||||||
|
#include <boost/mpl/contains.hpp>
|
||||||
|
|
||||||
namespace epee
|
namespace epee
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,8 +1,20 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#define CURRENT_TRANSACTION_VERSION 1
|
#define CURRENT_TRANSACTION_VERSION 1
|
||||||
|
#define POU_TRANSACTION_VERSION 6
|
||||||
#define OFFSHORE_TRANSACTION_VERSION 3
|
#define OFFSHORE_TRANSACTION_VERSION 3
|
||||||
#define HF_VERSION_XASSET_FEES_V2 17
|
#define HF_VERSION_XASSET_FEES_V2 17
|
||||||
|
#define HF_VERSION_HAVEN2 18
|
||||||
|
|
||||||
|
// UNLOCK TIMES
|
||||||
|
#define TX_V6_OFFSHORE_UNLOCK_BLOCKS 21*720 // 21 day unlock time
|
||||||
|
#define TX_V6_ONSHORE_UNLOCK_BLOCKS 360 // 12 hour unlock time
|
||||||
|
#define TX_V6_XASSET_UNLOCK_BLOCKS 1440 // 2 day unlock time
|
||||||
|
#define TX_V6_OFFSHORE_UNLOCK_BLOCKS_TESTNET 60 // 2 hour unlock time - FOR TESTING ONLY
|
||||||
|
#define TX_V6_ONSHORE_UNLOCK_BLOCKS_TESTNET 30 // 1 hour unlock time - FOR TESTING ONLY
|
||||||
|
#define TX_V6_XASSET_UNLOCK_BLOCKS_TESTNET 60 // 2 hour unlock time - FOR TESTING ONLY
|
||||||
|
|
||||||
|
#define PRICING_RECORD_VALID_TIME_DIFF_FROM_BLOCK 120 // seconds
|
||||||
|
|
||||||
enum BLOB_TYPE {
|
enum BLOB_TYPE {
|
||||||
BLOB_TYPE_CRYPTONOTE = 0,
|
BLOB_TYPE_CRYPTONOTE = 0,
|
||||||
|
|||||||
@@ -258,7 +258,8 @@ namespace cryptonote
|
|||||||
if (version == loki_version_3_per_output_unlock_times)
|
if (version == loki_version_3_per_output_unlock_times)
|
||||||
FIELD(is_deregister)
|
FIELD(is_deregister)
|
||||||
}
|
}
|
||||||
VARINT_FIELD(unlock_time)
|
if (blob_type != BLOB_TYPE_CRYPTONOTE_XHV || version < POU_TRANSACTION_VERSION)
|
||||||
|
VARINT_FIELD(unlock_time)
|
||||||
FIELD(vin)
|
FIELD(vin)
|
||||||
FIELD(vout)
|
FIELD(vout)
|
||||||
if (blob_type == BLOB_TYPE_CRYPTONOTE_LOKI || blob_type == BLOB_TYPE_CRYPTONOTE_XTNC)
|
if (blob_type == BLOB_TYPE_CRYPTONOTE_LOKI || blob_type == BLOB_TYPE_CRYPTONOTE_XTNC)
|
||||||
@@ -273,7 +274,13 @@ namespace cryptonote
|
|||||||
}
|
}
|
||||||
if (blob_type == BLOB_TYPE_CRYPTONOTE_XHV && version >= OFFSHORE_TRANSACTION_VERSION) {
|
if (blob_type == BLOB_TYPE_CRYPTONOTE_XHV && version >= OFFSHORE_TRANSACTION_VERSION) {
|
||||||
VARINT_FIELD(pricing_record_height)
|
VARINT_FIELD(pricing_record_height)
|
||||||
FIELD(offshore_data)
|
if (version < 5)
|
||||||
|
FIELD(offshore_data)
|
||||||
|
if (version >= POU_TRANSACTION_VERSION)
|
||||||
|
{
|
||||||
|
FIELD(output_unlock_times)
|
||||||
|
}
|
||||||
|
if (version >= POU_TRANSACTION_VERSION && vout.size() != output_unlock_times.size()) return false;
|
||||||
VARINT_FIELD(amount_burnt)
|
VARINT_FIELD(amount_burnt)
|
||||||
VARINT_FIELD(amount_minted)
|
VARINT_FIELD(amount_minted)
|
||||||
}
|
}
|
||||||
@@ -388,6 +395,7 @@ namespace cryptonote
|
|||||||
offshore_data.clear();
|
offshore_data.clear();
|
||||||
amount_burnt = 0;
|
amount_burnt = 0;
|
||||||
amount_minted = 0;
|
amount_minted = 0;
|
||||||
|
output_unlock_times.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
inline
|
inline
|
||||||
|
|||||||
@@ -85,25 +85,40 @@ namespace cryptonote
|
|||||||
{
|
{
|
||||||
uint64_t amount_in = 0;
|
uint64_t amount_in = 0;
|
||||||
uint64_t amount_out = 0;
|
uint64_t amount_out = 0;
|
||||||
if ((tx.blob_type == BLOB_TYPE_CRYPTONOTE_XHV) && (tx.version > 1))
|
if (tx.blob_type == BLOB_TYPE_CRYPTONOTE_XHV)
|
||||||
{
|
{
|
||||||
// This is the correct way to get the fee for Haven, because outs may be in different currencies to ins
|
// This is the correct way to get the fee for Haven, because outs may be in different currencies to ins
|
||||||
fee = tx.rct_signatures.txnFee + tx.rct_signatures.txnOffshoreFee;
|
switch (tx.version) {
|
||||||
|
case 6:
|
||||||
|
case 5:
|
||||||
|
fee = tx.rct_signatures.txnFee + tx.rct_signatures.txnOffshoreFee;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
case 3:
|
||||||
|
if (tx.vin[0].type() == typeid(txin_to_key)) {
|
||||||
|
fee = tx.rct_signatures.txnFee + tx.rct_signatures.txnOffshoreFee;
|
||||||
|
} else if (tx.vin[0].type() == typeid(txin_offshore)) {
|
||||||
|
fee = tx.rct_signatures.txnFee_usd + tx.rct_signatures.txnOffshoreFee_usd;
|
||||||
|
} else if (tx.vin[0].type() == typeid(txin_onshore)) {
|
||||||
|
fee = tx.rct_signatures.txnFee_usd + tx.rct_signatures.txnOffshoreFee_usd;
|
||||||
|
} else if (tx.vin[0].type() == typeid(txin_xasset)) {
|
||||||
|
fee = tx.rct_signatures.txnFee_xasset + tx.rct_signatures.txnOffshoreFee_xasset;
|
||||||
|
} else {
|
||||||
|
CHECK_AND_ASSERT_MES(false, false, "unexpected type id in transaction");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
case 1:
|
||||||
|
fee = tx.rct_signatures.txnFee;
|
||||||
|
break;
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
BOOST_FOREACH(auto& in, tx.vin)
|
BOOST_FOREACH(auto& in, tx.vin)
|
||||||
{
|
{
|
||||||
if (tx.blob_type != BLOB_TYPE_CRYPTONOTE_XHV) {
|
CHECK_AND_ASSERT_MES(in.type() == typeid(txin_to_key), 0, "unexpected type id in transaction");
|
||||||
CHECK_AND_ASSERT_MES(in.type() == typeid(txin_to_key), 0, "unexpected type id in transaction");
|
amount_in += boost::get<txin_to_key>(in).amount;
|
||||||
amount_in += boost::get<txin_to_key>(in).amount;
|
|
||||||
} else {
|
|
||||||
CHECK_AND_ASSERT_MES(in.type() == typeid(txin_to_key) || in.type() == typeid(txin_offshore) || in.type() == typeid(txin_onshore) || in.type() == typeid(txin_xasset), 0, "unexpected type id in transaction");
|
|
||||||
amount_in +=
|
|
||||||
in.type() == typeid(txin_to_key) ? boost::get<txin_to_key>(in).amount :
|
|
||||||
in.type() == typeid(txin_onshore) ? boost::get<txin_onshore>(in).amount :
|
|
||||||
in.type() == typeid(txin_offshore) ? boost::get<txin_offshore>(in).amount :
|
|
||||||
boost::get<txin_xasset>(in).amount;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
BOOST_FOREACH(auto& o, tx.vout)
|
BOOST_FOREACH(auto& o, tx.vout)
|
||||||
amount_out += o.amount;
|
amount_out += o.amount;
|
||||||
@@ -268,6 +283,10 @@ namespace cryptonote
|
|||||||
//-----------------------------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------------------------
|
||||||
bool check_outs_valid(const transaction& tx)
|
bool check_outs_valid(const transaction& tx)
|
||||||
{
|
{
|
||||||
|
if (tx.blob_type == BLOB_TYPE_CRYPTONOTE_XHV && tx.version >= POU_TRANSACTION_VERSION)
|
||||||
|
{
|
||||||
|
CHECK_AND_ASSERT_MES(tx.vout.size() == tx.output_unlock_times.size(), false, "tx version 6+ must have equal number of output unlock times and outputs");
|
||||||
|
}
|
||||||
BOOST_FOREACH(const tx_out& out, tx.vout)
|
BOOST_FOREACH(const tx_out& out, tx.vout)
|
||||||
{
|
{
|
||||||
if (tx.blob_type != BLOB_TYPE_CRYPTONOTE_XHV) {
|
if (tx.blob_type != BLOB_TYPE_CRYPTONOTE_XHV) {
|
||||||
|
|||||||
+110
-44
@@ -32,6 +32,7 @@
|
|||||||
#include "serialization/keyvalue_serialization.h"
|
#include "serialization/keyvalue_serialization.h"
|
||||||
#include "storages/portable_storage.h"
|
#include "storages/portable_storage.h"
|
||||||
|
|
||||||
|
#include "string_tools.h"
|
||||||
namespace offshore
|
namespace offshore
|
||||||
{
|
{
|
||||||
|
|
||||||
@@ -127,8 +128,8 @@ namespace offshore
|
|||||||
unused3 = in.unused3;
|
unused3 = in.unused3;
|
||||||
timestamp = in.timestamp;
|
timestamp = in.timestamp;
|
||||||
for (unsigned int i = 0; i < in.signature.length(); i += 2) {
|
for (unsigned int i = 0; i < in.signature.length(); i += 2) {
|
||||||
std::string byteString = in.signature.substr(i, 2);
|
std::string byteString = in.signature.substr(i, 2);
|
||||||
signature[i>>1] = (char) strtol(byteString.c_str(), NULL, 16);
|
signature[i>>1] = (char) strtol(byteString.c_str(), NULL, 16);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -194,7 +195,7 @@ namespace offshore
|
|||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t pricing_record::operator[](const std::string asset_type) const
|
uint64_t pricing_record::operator[](const std::string& asset_type) const
|
||||||
{
|
{
|
||||||
if (asset_type == "XHV") {
|
if (asset_type == "XHV") {
|
||||||
return 1000000000000;
|
return 1000000000000;
|
||||||
@@ -251,17 +252,26 @@ namespace offshore
|
|||||||
!::memcmp(signature, other.signature, sizeof(signature)));
|
!::memcmp(signature, other.signature, sizeof(signature)));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool pricing_record::is_empty() const noexcept
|
bool pricing_record::empty() const noexcept
|
||||||
{
|
{
|
||||||
const pricing_record empty_pr = offshore::pricing_record();
|
const pricing_record empty_pr = offshore::pricing_record();
|
||||||
return (*this).equal(empty_pr);
|
return (*this).equal(empty_pr);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool pricing_record::verifySignature(EVP_PKEY* public_key) const noexcept
|
bool pricing_record::verifySignature(const std::string& public_key) const
|
||||||
{
|
{
|
||||||
// Sanity check - accept empty pricing records
|
CHECK_AND_ASSERT_THROW_MES(!public_key.empty(), "Pricing record verification failed. NULL public key. PK Size: " << public_key.size()); // TODO: is this necessary or the one below already covers this case, meannin it will produce empty pubkey?
|
||||||
if ((*this).is_empty())
|
|
||||||
return true;
|
// extract the key
|
||||||
|
EVP_PKEY* pubkey;
|
||||||
|
BIO* bio = BIO_new_mem_buf(public_key.c_str(), public_key.size());
|
||||||
|
if (!bio) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
pubkey = PEM_read_bio_PUBKEY(bio, NULL, NULL, NULL);
|
||||||
|
BIO_free(bio);
|
||||||
|
CHECK_AND_ASSERT_THROW_MES(pubkey != NULL, "Pricing record verification failed. NULL public key.");
|
||||||
|
|
||||||
|
|
||||||
// Convert our internal 64-byte binary representation into 128-byte hex string
|
// Convert our internal 64-byte binary representation into 128-byte hex string
|
||||||
std::string sig_hex;
|
std::string sig_hex;
|
||||||
@@ -330,50 +340,23 @@ namespace offshore
|
|||||||
compact += (byte);
|
compact += (byte);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we have been passed a public key to use
|
|
||||||
EVP_PKEY* pubkey = NULL;
|
|
||||||
if (public_key) {
|
|
||||||
|
|
||||||
// Take a copy for local use
|
|
||||||
pubkey = public_key;
|
|
||||||
|
|
||||||
} else {
|
|
||||||
|
|
||||||
// No public key provided - failover to embedded key
|
|
||||||
static const char public_key[] = "-----BEGIN PUBLIC KEY-----\n"
|
|
||||||
"MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE5YBxWx1AZCA9jTUk8Pr2uZ9jpfRt\n"
|
|
||||||
"KWv3Vo1/Gny+1vfaxsXhBQiG1KlHkafNGarzoL0WHW4ocqaaqF5iv8i35A==\n"
|
|
||||||
"-----END PUBLIC KEY-----\n";
|
|
||||||
|
|
||||||
BIO* bio = BIO_new_mem_buf(public_key, (int)sizeof(public_key));
|
|
||||||
if (!bio) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
pubkey = PEM_read_bio_PUBKEY(bio, NULL, NULL, NULL);
|
|
||||||
BIO_free(bio);
|
|
||||||
}
|
|
||||||
assert(pubkey != NULL);
|
|
||||||
|
|
||||||
// Create a verify digest from the message
|
// Create a verify digest from the message
|
||||||
EVP_MD_CTX *ctx = EVP_MD_CTX_create();
|
EVP_MD_CTX *ctx = EVP_MD_CTX_create();
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
if (ctx) {
|
if (ctx) {
|
||||||
ret=EVP_DigestVerifyInit(ctx, NULL, EVP_sha256(), NULL, pubkey);
|
ret = EVP_DigestVerifyInit(ctx, NULL, EVP_sha256(), NULL, pubkey);
|
||||||
if (ret == 1) {
|
if (ret == 1) {
|
||||||
ret=EVP_DigestVerifyUpdate(ctx, message.data(), message.length());
|
ret = EVP_DigestVerifyUpdate(ctx, message.data(), message.length());
|
||||||
if (ret == 1) {
|
if (ret == 1) {
|
||||||
ret=EVP_DigestVerifyFinal(ctx, (const unsigned char *)compact.data(), compact.length());
|
ret = EVP_DigestVerifyFinal(ctx, (const unsigned char *)compact.data(), compact.length());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cleanup the context we created
|
// Cleanup the context we created
|
||||||
EVP_MD_CTX_destroy(ctx);
|
EVP_MD_CTX_destroy(ctx);
|
||||||
|
// Cleanup the openssl stuff
|
||||||
// Was the key provided by the caller?
|
EVP_PKEY_free(pubkey);
|
||||||
if (pubkey != public_key) {
|
|
||||||
// Cleanup the openssl stuff
|
|
||||||
EVP_PKEY_free(pubkey);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret == 1)
|
if (ret == 1)
|
||||||
return true;
|
return true;
|
||||||
@@ -383,4 +366,87 @@ namespace offshore
|
|||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
void pricing_record::set_for_height_821428() {
|
||||||
|
const std::string pr_821428 = "9b3f6f2f8f0000003d620e1202000000be71be2555120000b8627010000000000000000000000000ea0885b2270d00000000000000000000f797ff9be00b0000ddbdb005270a0000fc90cfe02b01060000000000000000000000000000000000d0a28224000e000000d643be960e0000002e8bb6a40e000000f8a817f80d00002f5d27d45cdbfbac3d0f6577103f68de30895967d7562fbd56c161ae90130f54301b1ea9d5fd062f37dac75c3d47178bc6f149d21da1ff0e8430065cb762b93a";
|
||||||
|
this->xAG = 614976143259;
|
||||||
|
this->xAU = 8892867133;
|
||||||
|
this->xAUD = 20156914758078;
|
||||||
|
this->xBTC = 275800760;
|
||||||
|
this->xCAD = 0;
|
||||||
|
this->xCHF = 14464149948650;
|
||||||
|
this->xCNY = 0;
|
||||||
|
this->xEUR = 13059317798903;
|
||||||
|
this->xGBP = 11162715471325;
|
||||||
|
this->xJPY = 1690137827184892;
|
||||||
|
this->xNOK = 0;
|
||||||
|
this->xNZD = 0;
|
||||||
|
this->xUSD = 15393775330000;
|
||||||
|
this->unused1 = 16040600000000;
|
||||||
|
this->unused2 = 16100600000000;
|
||||||
|
this->unused3 = 15359200000000;
|
||||||
|
this->timestamp = 0;
|
||||||
|
std::string sig = "2f5d27d45cdbfbac3d0f6577103f68de30895967d7562fbd56c161ae90130f54301b1ea9d5fd062f37dac75c3d47178bc6f149d21da1ff0e8430065cb762b93a";
|
||||||
|
int j=0;
|
||||||
|
for (unsigned int i = 0; i < sig.size(); i += 2) {
|
||||||
|
std::string byteString = sig.substr(i, 2);
|
||||||
|
this->signature[j++] = (char) strtol(byteString.c_str(), NULL, 16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// overload for pr validation for block
|
||||||
|
bool pricing_record::valid(uint32_t hf_version, uint64_t bl_timestamp, uint64_t last_bl_timestamp) const
|
||||||
|
{
|
||||||
|
// check for empty pr
|
||||||
|
if (hf_version >= HF_VERSION_XASSET_FEES_V2) {
|
||||||
|
if (this->empty())
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
unsigned char test_sig[64];
|
||||||
|
std::memset(test_sig, 0, sizeof(test_sig));
|
||||||
|
if (std::memcmp(test_sig, this->signature, sizeof(this->signature)) == 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Oracle public keys
|
||||||
|
std::string const mainnet_public_key = "-----BEGIN PUBLIC KEY-----\n"
|
||||||
|
"MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE5YBxWx1AZCA9jTUk8Pr2uZ9jpfRt\n"
|
||||||
|
"KWv3Vo1/Gny+1vfaxsXhBQiG1KlHkafNGarzoL0WHW4ocqaaqF5iv8i35A==\n"
|
||||||
|
"-----END PUBLIC KEY-----\n";
|
||||||
|
std::string const testnet_public_key = "-----BEGIN PUBLIC KEY-----\n"
|
||||||
|
"MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEtWqvQh7OdXrdgXcDeBMRVfLWTW3F\n"
|
||||||
|
"wByeoVJFBfZymScJIJl46j66xG6ngnyj4ai4/QPFnSZ1I9jjMRlTWC4EPA==\n"
|
||||||
|
"-----END PUBLIC KEY-----\n";
|
||||||
|
std::string const stagenet_public_key = "-----BEGIN PUBLIC KEY-----\n"
|
||||||
|
"MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEtWqvQh7OdXrdgXcDeBMRVfLWTW3F\n"
|
||||||
|
"wByeoVJFBfZymScJIJl46j66xG6ngnyj4ai4/QPFnSZ1I9jjMRlTWC4EPA==\n"
|
||||||
|
"-----END PUBLIC KEY-----\n";
|
||||||
|
|
||||||
|
// Comment out all but 1 of the following lines to select the correct Oracle PK
|
||||||
|
std::string const public_key = mainnet_public_key;
|
||||||
|
//std::string const public_key = testnet_public_key;
|
||||||
|
//std::string const public_key = stagenet_public_key;
|
||||||
|
|
||||||
|
// verify the signature
|
||||||
|
if (!verifySignature(public_key)) {
|
||||||
|
LOG_ERROR("Invalid pricing record signature.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// valiadte the timestmap
|
||||||
|
if (hf_version >= HF_VERSION_XASSET_FEES_V2) {
|
||||||
|
if (this->timestamp > bl_timestamp + PRICING_RECORD_VALID_TIME_DIFF_FROM_BLOCK) {
|
||||||
|
LOG_ERROR("Pricing record timestamp is too far in the future.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (this->timestamp <= last_bl_timestamp) {
|
||||||
|
LOG_ERROR("Pricing record timestamp is too old.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -43,6 +43,9 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
||||||
|
#include "cryptonote_config.h"
|
||||||
|
#include "crypto/hash.h"
|
||||||
|
|
||||||
namespace epee
|
namespace epee
|
||||||
{
|
{
|
||||||
namespace serialization
|
namespace serialization
|
||||||
@@ -80,47 +83,44 @@ namespace offshore
|
|||||||
class pricing_record
|
class pricing_record
|
||||||
{
|
{
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
// Fields
|
// Fields
|
||||||
uint64_t xAG;
|
uint64_t xAG;
|
||||||
uint64_t xAU;
|
uint64_t xAU;
|
||||||
uint64_t xAUD;
|
uint64_t xAUD;
|
||||||
uint64_t xBTC;
|
uint64_t xBTC;
|
||||||
uint64_t xCAD;
|
uint64_t xCAD;
|
||||||
uint64_t xCHF;
|
uint64_t xCHF;
|
||||||
uint64_t xCNY;
|
uint64_t xCNY;
|
||||||
uint64_t xEUR;
|
uint64_t xEUR;
|
||||||
uint64_t xGBP;
|
uint64_t xGBP;
|
||||||
uint64_t xJPY;
|
uint64_t xJPY;
|
||||||
uint64_t xNOK;
|
uint64_t xNOK;
|
||||||
uint64_t xNZD;
|
uint64_t xNZD;
|
||||||
uint64_t xUSD;
|
uint64_t xUSD;
|
||||||
uint64_t unused1;
|
uint64_t unused1;
|
||||||
uint64_t unused2;
|
uint64_t unused2;
|
||||||
uint64_t unused3;
|
uint64_t unused3;
|
||||||
uint64_t timestamp;
|
uint64_t timestamp;
|
||||||
unsigned char signature[64];
|
unsigned char signature[64];
|
||||||
|
|
||||||
// Default c'tor
|
// Default c'tor
|
||||||
pricing_record() noexcept;
|
pricing_record() noexcept;
|
||||||
|
//! Load from epee p2p format
|
||||||
//! Load from epee p2p format
|
bool _load(epee::serialization::portable_storage& src, epee::serialization::section* hparent);
|
||||||
bool _load(epee::serialization::portable_storage& src, epee::serialization::section* hparent);
|
//! Store in epee p2p format
|
||||||
|
bool store(epee::serialization::portable_storage& dest, epee::serialization::section* hparent) const;
|
||||||
//! Store in epee p2p format
|
pricing_record(const pricing_record& orig) noexcept;
|
||||||
bool store(epee::serialization::portable_storage& dest, epee::serialization::section* hparent) const;
|
~pricing_record() = default;
|
||||||
pricing_record(const pricing_record& orig) noexcept;
|
void set_for_height_821428();
|
||||||
~pricing_record() = default;
|
bool equal(const pricing_record& other) const noexcept;
|
||||||
pricing_record& operator=(const pricing_record& orig) noexcept;
|
bool empty() const noexcept;
|
||||||
|
bool verifySignature(const std::string& public_key) const;
|
||||||
|
bool valid(uint32_t hf_version, uint64_t bl_timestamp, uint64_t last_bl_timestamp) const;
|
||||||
|
|
||||||
uint64_t operator[](const std::string asset_type) const;
|
pricing_record& operator=(const pricing_record& orig) noexcept;
|
||||||
|
uint64_t operator[](const std::string& asset_type) const;
|
||||||
bool equal(const pricing_record& other) const noexcept;
|
|
||||||
|
|
||||||
bool is_empty() const noexcept;
|
|
||||||
|
|
||||||
bool verifySignature(EVP_PKEY* public_key = NULL) const noexcept;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
inline bool operator==(const pricing_record& a, const pricing_record& b) noexcept
|
inline bool operator==(const pricing_record& a, const pricing_record& b) noexcept
|
||||||
|
|||||||
+123
-102
@@ -48,6 +48,7 @@ extern "C" {
|
|||||||
|
|
||||||
#include "hex.h"
|
#include "hex.h"
|
||||||
#include "span.h"
|
#include "span.h"
|
||||||
|
#include "memwipe.h"
|
||||||
#include "serialization/vector.h"
|
#include "serialization/vector.h"
|
||||||
#include "serialization/debug_archive.h"
|
#include "serialization/debug_archive.h"
|
||||||
#include "serialization/binary_archive.h"
|
#include "serialization/binary_archive.h"
|
||||||
@@ -106,6 +107,8 @@ namespace rct {
|
|||||||
key L;
|
key L;
|
||||||
key R;
|
key R;
|
||||||
key ki;
|
key ki;
|
||||||
|
|
||||||
|
~multisig_kLRki() { memwipe(&k, sizeof(k)); }
|
||||||
};
|
};
|
||||||
|
|
||||||
struct multisig_out {
|
struct multisig_out {
|
||||||
@@ -255,6 +258,7 @@ namespace rct {
|
|||||||
RCTTypeBulletproof2 = 4,
|
RCTTypeBulletproof2 = 4,
|
||||||
RCTTypeCLSAG = 5,
|
RCTTypeCLSAG = 5,
|
||||||
RCTTypeCLSAGN = 6,
|
RCTTypeCLSAGN = 6,
|
||||||
|
RCTTypeHaven2 = 7, // Add public mask sum terms, remove extraneous fields (txnFee_usd,txnFee_xasset,txnOffshoreFee_usd,txnOffshoreFee_xasset)
|
||||||
};
|
};
|
||||||
enum RangeProofType { RangeProofBorromean, RangeProofBulletproof, RangeProofMultiOutputBulletproof, RangeProofPaddedBulletproof };
|
enum RangeProofType { RangeProofBorromean, RangeProofBulletproof, RangeProofMultiOutputBulletproof, RangeProofPaddedBulletproof };
|
||||||
struct RCTConfig {
|
struct RCTConfig {
|
||||||
@@ -262,108 +266,125 @@ namespace rct {
|
|||||||
int bp_version;
|
int bp_version;
|
||||||
};
|
};
|
||||||
struct rctSigBase {
|
struct rctSigBase {
|
||||||
uint8_t type;
|
uint8_t type;
|
||||||
key message;
|
key message;
|
||||||
ctkeyM mixRing; //the set of all pubkeys / copy
|
ctkeyM mixRing; //the set of all pubkeys / copy
|
||||||
//pairs that you mix with
|
//pairs that you mix with
|
||||||
keyV pseudoOuts; //C - for simple rct
|
keyV pseudoOuts; //C - for simple rct
|
||||||
std::vector<ecdhTuple> ecdhInfo;
|
std::vector<ecdhTuple> ecdhInfo;
|
||||||
ctkeyV outPk;
|
ctkeyV outPk;
|
||||||
ctkeyV outPk_usd;
|
ctkeyV outPk_usd;
|
||||||
ctkeyV outPk_xasset;
|
ctkeyV outPk_xasset;
|
||||||
xmr_amount txnFee; // contains b
|
xmr_amount txnFee = 0; // contains b
|
||||||
xmr_amount txnFee_usd;
|
xmr_amount txnFee_usd = 0;
|
||||||
xmr_amount txnFee_xasset;
|
xmr_amount txnFee_xasset = 0;
|
||||||
xmr_amount txnOffshoreFee;
|
xmr_amount txnOffshoreFee = 0;
|
||||||
xmr_amount txnOffshoreFee_usd;
|
xmr_amount txnOffshoreFee_usd = 0;
|
||||||
xmr_amount txnOffshoreFee_xasset;
|
xmr_amount txnOffshoreFee_xasset = 0;
|
||||||
|
keyV maskSums; // contains 2 elements. 1. is the sum of masks of inputs. 2. is the sum of masks of changes.
|
||||||
|
|
||||||
template<bool W, template <bool> class Archive>
|
template<bool W, template <bool> class Archive>
|
||||||
bool serialize_rctsig_base(Archive<W> &ar, size_t inputs, size_t outputs)
|
bool serialize_rctsig_base(Archive<W> &ar, size_t inputs, size_t outputs)
|
||||||
|
{
|
||||||
|
FIELD(type)
|
||||||
|
if (type == RCTTypeNull)
|
||||||
|
return ar.stream().good();
|
||||||
|
if (type != RCTTypeFull && type != RCTTypeSimple && type != RCTTypeBulletproof && type != RCTTypeBulletproof2 && type != RCTTypeCLSAG && type != RCTTypeCLSAGN && type != RCTTypeHaven2)
|
||||||
|
return false;
|
||||||
|
VARINT_FIELD(txnFee)
|
||||||
|
if (type == RCTTypeHaven2) {
|
||||||
|
// serialize offshore fee
|
||||||
|
VARINT_FIELD(txnOffshoreFee)
|
||||||
|
} else if (type == RCTTypeCLSAG || type == RCTTypeCLSAGN) {
|
||||||
|
VARINT_FIELD(txnFee_usd)
|
||||||
|
if (type == RCTTypeCLSAGN)
|
||||||
|
{
|
||||||
|
VARINT_FIELD(txnFee_xasset)
|
||||||
|
}
|
||||||
|
VARINT_FIELD(txnOffshoreFee)
|
||||||
|
VARINT_FIELD(txnOffshoreFee_usd)
|
||||||
|
if (type == RCTTypeCLSAGN)
|
||||||
|
{
|
||||||
|
VARINT_FIELD(txnOffshoreFee_xasset)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
txnFee_usd = 0;
|
||||||
|
txnFee_xasset = 0;
|
||||||
|
txnOffshoreFee = 0;
|
||||||
|
txnOffshoreFee_usd = 0;
|
||||||
|
txnOffshoreFee_xasset = 0;
|
||||||
|
}
|
||||||
|
// inputs/outputs not saved, only here for serialization help
|
||||||
|
// FIELD(message) - not serialized, it can be reconstructed
|
||||||
|
// FIELD(mixRing) - not serialized, it can be reconstructed
|
||||||
|
if (type == RCTTypeSimple) // moved to prunable with bulletproofs
|
||||||
{
|
{
|
||||||
FIELD(type)
|
ar.tag("pseudoOuts");
|
||||||
if (type == RCTTypeNull)
|
|
||||||
return ar.stream().good();
|
|
||||||
if (type != RCTTypeFull && type != RCTTypeSimple && type != RCTTypeBulletproof && type != RCTTypeBulletproof2 && type != RCTTypeCLSAG && type != RCTTypeCLSAGN)
|
|
||||||
return false;
|
|
||||||
VARINT_FIELD(txnFee)
|
|
||||||
if ((type == RCTTypeCLSAG) || (type == RCTTypeCLSAGN))
|
|
||||||
{
|
|
||||||
VARINT_FIELD(txnFee_usd)
|
|
||||||
if (type == RCTTypeCLSAGN)
|
|
||||||
{
|
|
||||||
VARINT_FIELD(txnFee_xasset)
|
|
||||||
}
|
|
||||||
VARINT_FIELD(txnOffshoreFee)
|
|
||||||
VARINT_FIELD(txnOffshoreFee_usd)
|
|
||||||
if (type == RCTTypeCLSAGN)
|
|
||||||
{
|
|
||||||
VARINT_FIELD(txnOffshoreFee_xasset)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
txnFee_usd = 0;
|
|
||||||
txnFee_xasset = 0;
|
|
||||||
txnOffshoreFee = 0;
|
|
||||||
txnOffshoreFee_usd = 0;
|
|
||||||
txnOffshoreFee_xasset = 0;
|
|
||||||
}
|
|
||||||
// inputs/outputs not saved, only here for serialization help
|
|
||||||
// FIELD(message) - not serialized, it can be reconstructed
|
|
||||||
// FIELD(mixRing) - not serialized, it can be reconstructed
|
|
||||||
if (type == RCTTypeSimple) // moved to prunable with bulletproofs
|
|
||||||
{
|
|
||||||
ar.tag("pseudoOuts");
|
|
||||||
ar.begin_array();
|
|
||||||
PREPARE_CUSTOM_VECTOR_SERIALIZATION(inputs, pseudoOuts);
|
|
||||||
if (pseudoOuts.size() != inputs)
|
|
||||||
return false;
|
|
||||||
for (size_t i = 0; i < inputs; ++i)
|
|
||||||
{
|
|
||||||
FIELDS(pseudoOuts[i])
|
|
||||||
if (inputs - i > 1)
|
|
||||||
ar.delimit_array();
|
|
||||||
}
|
|
||||||
ar.end_array();
|
|
||||||
}
|
|
||||||
|
|
||||||
ar.tag("ecdhInfo");
|
|
||||||
ar.begin_array();
|
ar.begin_array();
|
||||||
PREPARE_CUSTOM_VECTOR_SERIALIZATION(outputs, ecdhInfo);
|
PREPARE_CUSTOM_VECTOR_SERIALIZATION(inputs, pseudoOuts);
|
||||||
if (ecdhInfo.size() != outputs)
|
if (pseudoOuts.size() != inputs)
|
||||||
return false;
|
return false;
|
||||||
for (size_t i = 0; i < outputs; ++i)
|
for (size_t i = 0; i < inputs; ++i)
|
||||||
{
|
{
|
||||||
if (type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN)
|
FIELDS(pseudoOuts[i])
|
||||||
{
|
if (inputs - i > 1)
|
||||||
ar.begin_object();
|
|
||||||
if (!typename Archive<W>::is_saving())
|
|
||||||
memset(ecdhInfo[i].amount.bytes, 0, sizeof(ecdhInfo[i].amount.bytes));
|
|
||||||
crypto::hash8 &amount = (crypto::hash8&)ecdhInfo[i].amount;
|
|
||||||
FIELD(amount);
|
|
||||||
ar.end_object();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
FIELDS(ecdhInfo[i])
|
|
||||||
}
|
|
||||||
if (outputs - i > 1)
|
|
||||||
ar.delimit_array();
|
ar.delimit_array();
|
||||||
}
|
}
|
||||||
ar.end_array();
|
ar.end_array();
|
||||||
|
}
|
||||||
|
|
||||||
ar.tag("outPk");
|
ar.tag("ecdhInfo");
|
||||||
ar.begin_array();
|
ar.begin_array();
|
||||||
PREPARE_CUSTOM_VECTOR_SERIALIZATION(outputs, outPk);
|
PREPARE_CUSTOM_VECTOR_SERIALIZATION(outputs, ecdhInfo);
|
||||||
if (outPk.size() != outputs)
|
if (ecdhInfo.size() != outputs)
|
||||||
return false;
|
return false;
|
||||||
for (size_t i = 0; i < outputs; ++i)
|
for (size_t i = 0; i < outputs; ++i)
|
||||||
|
{
|
||||||
|
if (type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN || type == RCTTypeHaven2)
|
||||||
{
|
{
|
||||||
FIELDS(outPk[i].mask)
|
ar.begin_object();
|
||||||
|
if (!typename Archive<W>::is_saving())
|
||||||
|
memset(ecdhInfo[i].amount.bytes, 0, sizeof(ecdhInfo[i].amount.bytes));
|
||||||
|
crypto::hash8 &amount = (crypto::hash8&)ecdhInfo[i].amount;
|
||||||
|
FIELD(amount);
|
||||||
|
ar.end_object();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
FIELDS(ecdhInfo[i])
|
||||||
|
}
|
||||||
|
if (outputs - i > 1)
|
||||||
|
ar.delimit_array();
|
||||||
|
}
|
||||||
|
ar.end_array();
|
||||||
|
|
||||||
|
ar.tag("outPk");
|
||||||
|
ar.begin_array();
|
||||||
|
PREPARE_CUSTOM_VECTOR_SERIALIZATION(outputs, outPk);
|
||||||
|
if (outPk.size() != outputs)
|
||||||
|
return false;
|
||||||
|
for (size_t i = 0; i < outputs; ++i)
|
||||||
|
{
|
||||||
|
FIELDS(outPk[i].mask)
|
||||||
if (outputs - i > 1)
|
if (outputs - i > 1)
|
||||||
ar.delimit_array();
|
ar.delimit_array();
|
||||||
}
|
}
|
||||||
|
ar.end_array();
|
||||||
|
|
||||||
|
if (type == RCTTypeHaven2) {
|
||||||
|
|
||||||
|
ar.tag("maskSums");
|
||||||
|
ar.begin_array();
|
||||||
|
PREPARE_CUSTOM_VECTOR_SERIALIZATION(2, maskSums);
|
||||||
|
if (maskSums.size() != 2)
|
||||||
|
return false;
|
||||||
|
FIELDS(maskSums[0])
|
||||||
|
ar.delimit_array();
|
||||||
|
FIELDS(maskSums[1])
|
||||||
ar.end_array();
|
ar.end_array();
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
if ((type == RCTTypeCLSAG) || (type == RCTTypeCLSAGN))
|
if ((type == RCTTypeCLSAG) || (type == RCTTypeCLSAGN))
|
||||||
{
|
{
|
||||||
ar.tag("outPk_usd");
|
ar.tag("outPk_usd");
|
||||||
@@ -374,8 +395,8 @@ namespace rct {
|
|||||||
for (size_t i = 0; i < outputs; ++i)
|
for (size_t i = 0; i < outputs; ++i)
|
||||||
{
|
{
|
||||||
FIELDS(outPk_usd[i].mask)
|
FIELDS(outPk_usd[i].mask)
|
||||||
if (outputs - i > 1)
|
if (outputs - i > 1)
|
||||||
ar.delimit_array();
|
ar.delimit_array();
|
||||||
}
|
}
|
||||||
ar.end_array();
|
ar.end_array();
|
||||||
}
|
}
|
||||||
@@ -389,14 +410,14 @@ namespace rct {
|
|||||||
for (size_t i = 0; i < outputs; ++i)
|
for (size_t i = 0; i < outputs; ++i)
|
||||||
{
|
{
|
||||||
FIELDS(outPk_xasset[i].mask)
|
FIELDS(outPk_xasset[i].mask)
|
||||||
if (outputs - i > 1)
|
if (outputs - i > 1)
|
||||||
ar.delimit_array();
|
ar.delimit_array();
|
||||||
}
|
}
|
||||||
ar.end_array();
|
ar.end_array();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return ar.stream().good();
|
return ar.stream().good();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
struct rctSigPrunable {
|
struct rctSigPrunable {
|
||||||
std::vector<rangeSig> rangeSigs;
|
std::vector<rangeSig> rangeSigs;
|
||||||
@@ -411,12 +432,12 @@ namespace rct {
|
|||||||
{
|
{
|
||||||
if (type == RCTTypeNull)
|
if (type == RCTTypeNull)
|
||||||
return ar.stream().good();
|
return ar.stream().good();
|
||||||
if (type != RCTTypeFull && type != RCTTypeSimple && type != RCTTypeBulletproof && type != RCTTypeBulletproof2 && type != RCTTypeCLSAG && type != RCTTypeCLSAGN)
|
if (type != RCTTypeFull && type != RCTTypeSimple && type != RCTTypeBulletproof && type != RCTTypeBulletproof2 && type != RCTTypeCLSAG && type != RCTTypeCLSAGN && type != RCTTypeHaven2)
|
||||||
return false;
|
return false;
|
||||||
if (type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN)
|
if (type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN || type == RCTTypeHaven2)
|
||||||
{
|
{
|
||||||
uint32_t nbp = bulletproofs.size();
|
uint32_t nbp = bulletproofs.size();
|
||||||
if (type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN)
|
if (type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN || type == RCTTypeHaven2)
|
||||||
VARINT_FIELD(nbp)
|
VARINT_FIELD(nbp)
|
||||||
else
|
else
|
||||||
FIELD(nbp)
|
FIELD(nbp)
|
||||||
@@ -451,7 +472,7 @@ namespace rct {
|
|||||||
ar.end_array();
|
ar.end_array();
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((type == RCTTypeCLSAG) || (type == RCTTypeCLSAGN))
|
if ((type == RCTTypeCLSAG) || (type == RCTTypeCLSAGN) || (type == RCTTypeHaven2))
|
||||||
{
|
{
|
||||||
ar.tag("CLSAGs");
|
ar.tag("CLSAGs");
|
||||||
ar.begin_array();
|
ar.begin_array();
|
||||||
@@ -542,7 +563,7 @@ namespace rct {
|
|||||||
}
|
}
|
||||||
ar.end_array();
|
ar.end_array();
|
||||||
}
|
}
|
||||||
if (type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN)
|
if (type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN || type == RCTTypeHaven2)
|
||||||
{
|
{
|
||||||
ar.tag("pseudoOuts");
|
ar.tag("pseudoOuts");
|
||||||
ar.begin_array();
|
ar.begin_array();
|
||||||
@@ -566,12 +587,12 @@ namespace rct {
|
|||||||
|
|
||||||
keyV& get_pseudo_outs()
|
keyV& get_pseudo_outs()
|
||||||
{
|
{
|
||||||
return type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN ? p.pseudoOuts : pseudoOuts;
|
return type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN || type == RCTTypeHaven2 ? p.pseudoOuts : pseudoOuts;
|
||||||
}
|
}
|
||||||
|
|
||||||
keyV const& get_pseudo_outs() const
|
keyV const& get_pseudo_outs() const
|
||||||
{
|
{
|
||||||
return type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN ? p.pseudoOuts : pseudoOuts;
|
return type == RCTTypeBulletproof || type == RCTTypeBulletproof2 || type == RCTTypeCLSAG || type == RCTTypeCLSAGN || type == RCTTypeHaven2 ? p.pseudoOuts : pseudoOuts;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user