You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
211 lines
5.8 KiB
211 lines
5.8 KiB
/*
|
|
This file is part of ethereum.js.
|
|
|
|
ethereum.js 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.
|
|
|
|
ethereum.js 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 ethereum.js. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
/**
|
|
* @file param.js
|
|
* @author Marek Kotewicz <marek@ethdev.com>
|
|
* @date 2015
|
|
*/
|
|
|
|
var utils = require('../utils/utils');
|
|
|
|
/**
|
|
* SolidityParam object prototype.
|
|
* Should be used when encoding, decoding solidity bytes
|
|
*/
|
|
var SolidityParam = function (value, offset) {
|
|
this.value = value || '';
|
|
this.offset = offset; // offset in bytes
|
|
};
|
|
|
|
/**
|
|
* This method should be used to get length of params's dynamic part
|
|
*
|
|
* @method dynamicPartLength
|
|
* @returns {Number} length of dynamic part (in bytes)
|
|
*/
|
|
SolidityParam.prototype.dynamicPartLength = function () {
|
|
return this.dynamicPart().length / 2;
|
|
};
|
|
|
|
/**
|
|
* This method should be used to create copy of solidity param with different offset
|
|
*
|
|
* @method withOffset
|
|
* @param {Number} offset length in bytes
|
|
* @returns {SolidityParam} new solidity param with applied offset
|
|
*/
|
|
SolidityParam.prototype.withOffset = function (offset) {
|
|
return new SolidityParam(this.value, offset);
|
|
};
|
|
|
|
/**
|
|
* This method should be used to combine solidity params together
|
|
* eg. when appending an array
|
|
*
|
|
* @method combine
|
|
* @param {SolidityParam} param with which we should combine
|
|
* @param {SolidityParam} result of combination
|
|
*/
|
|
SolidityParam.prototype.combine = function (param) {
|
|
return new SolidityParam(this.value + param.value);
|
|
};
|
|
|
|
/**
|
|
* This method should be called to check if param has dynamic size.
|
|
* If it has, it returns true, otherwise false
|
|
*
|
|
* @method isDynamic
|
|
* @returns {Boolean}
|
|
*/
|
|
SolidityParam.prototype.isDynamic = function () {
|
|
return this.value.length > 64 || this.offset !== undefined;
|
|
};
|
|
|
|
/**
|
|
* This method should be called to transform offset to bytes
|
|
*
|
|
* @method offsetAsBytes
|
|
* @returns {String} bytes representation of offset
|
|
*/
|
|
SolidityParam.prototype.offsetAsBytes = function () {
|
|
return !this.isDynamic() ? '' : utils.padLeft(utils.toTwosComplement(this.offset).toString(16), 64);
|
|
};
|
|
|
|
/**
|
|
* This method should be called to get static part of param
|
|
*
|
|
* @method staticPart
|
|
* @returns {String} offset if it is a dynamic param, otherwise value
|
|
*/
|
|
SolidityParam.prototype.staticPart = function () {
|
|
if (!this.isDynamic()) {
|
|
return this.value;
|
|
}
|
|
return this.offsetAsBytes();
|
|
};
|
|
|
|
/**
|
|
* This method should be called to get dynamic part of param
|
|
*
|
|
* @method dynamicPart
|
|
* @returns {String} returns a value if it is a dynamic param, otherwise empty string
|
|
*/
|
|
SolidityParam.prototype.dynamicPart = function () {
|
|
return this.isDynamic() ? this.value : '';
|
|
};
|
|
|
|
/**
|
|
* This method should be called to encode param
|
|
*
|
|
* @method encode
|
|
* @returns {String}
|
|
*/
|
|
SolidityParam.prototype.encode = function () {
|
|
return this.staticPart() + this.dynamicPart();
|
|
};
|
|
|
|
/**
|
|
* This method should be called to encode array of params
|
|
*
|
|
* @method encodeList
|
|
* @param {Array[SolidityParam]} params
|
|
* @returns {String}
|
|
*/
|
|
SolidityParam.encodeList = function (params) {
|
|
|
|
// updating offsets
|
|
var totalOffset = params.length * 32;
|
|
var offsetParams = params.map(function (param) {
|
|
if (!param.isDynamic()) {
|
|
return param;
|
|
}
|
|
var offset = totalOffset;
|
|
totalOffset += param.dynamicPartLength();
|
|
return param.withOffset(offset);
|
|
});
|
|
|
|
// encode everything!
|
|
return offsetParams.reduce(function (result, param) {
|
|
return result + param.dynamicPart();
|
|
}, offsetParams.reduce(function (result, param) {
|
|
return result + param.staticPart();
|
|
}, ''));
|
|
};
|
|
|
|
/**
|
|
* This method should be used to decode plain (static) solidity param at given index
|
|
*
|
|
* @method decodeParam
|
|
* @param {String} bytes
|
|
* @param {Number} index
|
|
* @returns {SolidityParam}
|
|
*/
|
|
SolidityParam.decodeParam = function (bytes, index) {
|
|
index = index || 0;
|
|
return new SolidityParam(bytes.substr(index * 64, 64));
|
|
};
|
|
|
|
/**
|
|
* This method should be called to get offset value from bytes at given index
|
|
*
|
|
* @method getOffset
|
|
* @param {String} bytes
|
|
* @param {Number} index
|
|
* @returns {Number} offset as number
|
|
*/
|
|
var getOffset = function (bytes, index) {
|
|
// we can do this cause offset is rather small
|
|
return parseInt('0x' + bytes.substr(index * 64, 64));
|
|
};
|
|
|
|
/**
|
|
* This method should be called to decode solidity bytes param at given index
|
|
*
|
|
* @method decodeBytes
|
|
* @param {String} bytes
|
|
* @param {Number} index
|
|
* @returns {SolidityParam}
|
|
*/
|
|
SolidityParam.decodeBytes = function (bytes, index) {
|
|
index = index || 0;
|
|
|
|
var offset = getOffset(bytes, index);
|
|
|
|
var l = parseInt('0x' + bytes.substr(offset * 2, 64));
|
|
l = Math.floor((l + 31) / 32);
|
|
|
|
// (1 + l) * , cause we also parse length
|
|
return new SolidityParam(bytes.substr(offset * 2, (1 + l) * 64), 0);
|
|
};
|
|
|
|
/**
|
|
* This method should be used to decode solidity array at given index
|
|
*
|
|
* @method decodeArray
|
|
* @param {String} bytes
|
|
* @param {Number} index
|
|
* @returns {SolidityParam}
|
|
*/
|
|
SolidityParam.decodeArray = function (bytes, index) {
|
|
index = index || 0;
|
|
var offset = getOffset(bytes, index);
|
|
var length = parseInt('0x' + bytes.substr(offset * 2, 64));
|
|
return new SolidityParam(bytes.substr(offset * 2, (length + 1) * 64), 0);
|
|
};
|
|
|
|
module.exports = SolidityParam;
|
|
|
|
|