Here and after we will call credits.com Blockchain Network Node simply the "node" or "Credits node", and any external client software simply the "client" or "external client".
Credits node is a piece of software which provides access to replicated across a peer-to-peer verifiable append only database — the blockchain. Transport layer of this network is beyond the scope of this document, while the network service for external clients certainly is. Attentive reader may have concluded that these two networks are separate.
For relative simplicity of implementing external clients in a plethora of different programming languages Credits node uses Remote Procedure Call (RPC) Interface Description Language (IDL) called Apache Thrift (https://thrift.apache.org/). The choice of this technology was guided by it's rich type system and extensive language support while also implementing widely helpful language features such as exceptions.
For now, Credits node handles Thrift requests arriving to two TCP ports: 8081
for web-browser clients, for which Thrift officially supports only JSON-based serialization over HTTP transport as it seems after reading generated sources, and 9090
with TBinaryProtocol
and TSocket
transport for clients supporting it.
Our complete API specification is placed in the end of current document. Further we will refer to it as api.thrift
.
Our complete API specification is placed in the end of current document. Further we will refer to it as api.thrift
.
Quoting Apache Thrift homepage, you can generate sources for your language using command like
$ thrift --gen <language> <path where you stored it>/api.thrift
where $
denotes your command prompt invitation. Beware! If you use this command generated sources will be placed under your current working directory. See output of thrift --help
for more options and language-specific help.
Further we will describe everything either not covered by comments in api.thrift
at all or what was beyond ethical comment line size. If you feel something is missing, please contact our technical support and we will add it here or to the to-be-written FAQ section.
These are the points of attention which your author remembered at the time of writing:
- Node's API blocks in
TransactionFlow
call when you deploy smart contract with it till the moment it is actually deployed i.e. is put to the blockchain meaning it arrives as a result of a consensus and considered valid. On the other hand, execution of smart contract is blocked in case of previous execution of the same contract has not (yet?) concluded successfully being it requested from the same node. - For reference: to deploy smart contract you need to set it's aimed address as a transaction target, put itself with it's bytecode and all the other stuff into
smartContract
field of transaction (consider checking Thrift docs that you do it right, it may be not so obvious to arrange foroptional
Thrift field to be set in your language of choice) , and make suremethod
field of that structure is set to empty string (""
). On the other hand to execute smart,method
argument needs to be filled. - Public key which now plays a role of an wallet or smart contract address has length of 32 bytes and is generated with it's private counterpart using
ed25519
.
Side note: Auxiliary file variant.thrift
is in section after this one.
include 'variant.thrift'
namespace csharp NodeApi
namespace netcore NodeApi
namespace java com.credits.leveldb.client.thrift
namespace cpp api
// Currency code, 1 for mainline one (Credits, 'CS')
typedef i8 Currency
// Wallet or smart contract address
typedef binary Address
// Unix timestamp in milliseconds
typedef i64 Time
struct Amount
{
1: required i32 integral = 0;
2: required i64 fraction = 0;
}
struct CumulativeAmount
{
1: required i64 integral = 0;
2: required i64 fraction = 0;
}
typedef map<Currency, Amount> Balance;
typedef map<Currency, CumulativeAmount> Total;
// Smart contract info
struct SmartContract
{
1: required Address address;
2: Address deployer
3: string sourceCode;
4: binary byteCode;
// Wrong name, here resides hash of smart contract's bytecode
5: string hashState;
}
struct SmartContractInvocation
{
1: string sourceCode;
2: binary byteCode;
// Not processed, input only
3: string hashState;
// Empty on deploy, method name of a smart contract class on execute
4: string method;
// Empty on deploy, method params stringified Java-side with conversion to string on execute
5: list<string> params;
// If true, do not emit any transactions to blockchain (execute smart contract and forget state change if any)
6: bool forgetNewState;
}
//
// Transactions
//
struct TransactionId
{
1: PoolHash poolHash
// Position inside of block
2: i32 index
}
struct Transaction
{
// Inner transaction ID for protection against replay attack
1: i64 id
// Giver if no smart contract invokation is present, otherwise deployer.
// Generally, public key against of which signature is validated
2: Address source
// Smart contract address if one's invokation is present, otherwise acceptor's address
3: Address target
// Transfer amount for payment transaction
4: Amount amount
// Wallet's view on it's balance
5: Amount balance
6: Currency currency
// Signature is formed against node's custom binary serialization format,
// see other docs for description
7: binary signature
8: optional SmartContractInvocation smartContract
// Max fee acceptable for donor to be subtracted
9: Amount fee
}
// Structure for tranactions that have been emplaced to the blockchain
struct SealedTransaction {
1: TransactionId id
2: Transaction trxn
}
//
// Pools
//
typedef binary PoolHash
// Sequential index of block, starting with zero
typedef i64 PoolNumber
struct Pool
{
1: PoolHash hash
// Previous block hash
2: PoolHash prevHash
// Timestamp from writer (?)
3: Time time
// Amount of transactions in this block
4: i32 transactionsCount
5: PoolNumber poolNumber
}
//
// Stats
//
typedef i32 Count
struct PeriodStats
{
// Amount of milliseconds over which following aggregated results are reported
1: Time periodDuration
// Amount of pools
2: Count poolsCount
// Amount of transactionss
3: Count transactionsCount
// Cumulative volume of each currency transferred
4: Total balancePerCurrency
// Amount of smart contracts transactions
5: Count smartContractsCount
}
// Periods are 24h, 1 month, 1 year, and cover-all period
typedef list<PeriodStats> StatsPerPeriod
//
// API responses
//
struct APIResponse
{
// 0 for success, 1 for failure, 2 for not being implemented (currently unused)
1: i8 code
// Explanation
2: string message
}
struct BalanceGetResult
{
1: APIResponse status
2: Amount amount
}
struct TransactionGetResult
{
1: APIResponse status
2: bool found
3: SealedTransaction transaction
}
struct TransactionsGetResult
{
1: APIResponse status
2: bool result
3: list<SealedTransaction> transactions
}
struct TransactionFlowResult
{
1: APIResponse status
2: optional variant.Variant smart_contract_result
}
struct PoolListGetResult
{
1: APIResponse status
2: bool result
3: list<Pool> pools
}
struct PoolInfoGetResult
{
1: APIResponse status
2: bool isFound
3: Pool pool
}
struct PoolTransactionsGetResult
{
1: APIResponse status
2: list<SealedTransaction> transactions
}
struct StatsGetResult
{
1: APIResponse status
2: StatsPerPeriod stats
}
struct SmartContractGetResult
{
1: APIResponse status
2: SmartContract smartContract
}
struct SmartContractAddressesListGetResult
{
1: APIResponse status
2: list<Address> addressesList
}
struct SmartContractsListGetResult
{
1: APIResponse status
2: list<SmartContract> smartContractsList
}
service API
{
BalanceGetResult BalanceGet(1:Address address, 2:Currency currency = 1)
TransactionGetResult TransactionGet(1:TransactionId transactionId)
// Get transactions where `address` is either sender or receiver
TransactionsGetResult TransactionsGet(1:Address address, 2:i64 offset, 3:i64 limit)
// Not for monitor. Transmit transaction to network for approval
TransactionFlowResult TransactionFlow(1:Transaction transaction)
// For tetris for now.
PoolHash GetLastHash()
// Was intended for use by web monitor. Never tested, get blocks starting from `hash` up to `limit` instances
PoolListGetResult PoolListGetStable(1:PoolHash hash, 2:i64 limit)
// For web monitor, used now. Get metainfo about pools skipping `offset` up to `limit` in amount
PoolListGetResult PoolListGet(1:i64 offset, 2:i64 limit)
// For web monitor. Get metainfo about block by hash
PoolInfoGetResult PoolInfoGet(1:PoolHash hash, 2:i64 index)
// For web monitor. Get transactions from exactly `hash` pool, skipping `offset` and retrieiving at most `limit`
PoolTransactionsGetResult PoolTransactionsGet(1:PoolHash hash, 2:i64 offset, 3:i64 limit)
// For web monitor.
StatsGetResult StatsGet()
SmartContractGetResult SmartContractGet(1:Address address)
SmartContractsListGetResult SmartContractsListGet(1:Address deployer)
SmartContractAddressesListGetResult SmartContractAddressesListGet(1:Address deployer)
// Blocks till `obsolete` is not the last block in chain.
PoolHash WaitForBlock(1:PoolHash obsolete)
// Blocks till there are transactions arrived to `smart_address`
// not yet seen in current node's process lifetime.
TransactionId WaitForSmartTransaction(1:Address smart_address)
SmartContractsListGetResult SmartContractsAllListGet(1:i64 offset, 2:i64 limit)
}
namespace cpp variant
namespace java com.credits.thrift.generated
union Variant {
1: bool v_bool;
2: i8 v_i8;
3: i16 v_i16;
4: i32 v_i32;
5: i64 v_i64;
6: double v_double;
7: string v_string;
8: list<Variant> v_list;
9: set<Variant> v_set;
10: map<Variant, Variant> v_map;
}