import isUrl from 'is-url'; import { sha256 } from 'js-sha256'; import { generatePostBodyBroadcast, BroadcastMode } from '@tharsis/provider'; import { Chain, Sender, Fee, createMessageSend, MessageSendParams } from '@tharsis/transactions' import { createTxMsgCancelBond, createTxMsgCreateBond, createTxMsgRefillBond, createTxMsgWithdrawBond, MessageMsgCancelBond, MessageMsgCreateBond, MessageMsgRefillBond, MessageMsgWithdrawBond } from "./messages/bond"; import { RegistryClient } from "./registry-client"; import { Account } from "./account"; import { createTransaction } from "./txbuilder"; import { createTxMsgDeleteName, createTxMsgReserveAuthority, createTxMsgSetAuthorityBond, createTxMsgSetName, createTxMsgSetRecord, MessageMsgDeleteName, MessageMsgReserveAuthority, MessageMsgSetAuthorityBond, MessageMsgSetName, MessageMsgSetRecord } from './messages/nameservice'; import { Payload, Record } from './types'; const DEFAULT_WRITE_ERROR = 'Unable to write to chiba-clonk.'; export const DEFAULT_CHAIN_ID = 'chibaclonk_9000-1'; // Parse Tx response from cosmos-sdk. export const parseTxResponse = (result: any) => { const { txhash: hash, height, ...txResponse } = result; txResponse.data = txResponse.data && Buffer.from(txResponse.data, 'base64').toString('utf8'); txResponse.events.forEach((event:any) => { event.attributes = event.attributes.map(({ key, value }: { key: string, value: string }) => ({ key: Buffer.from(key, 'base64').toString('utf8'), value: Buffer.from(value, 'base64').toString('utf8') })); }); return { hash, height, ...txResponse }; }; export const isKeyValid = (key: string) => key && key.match(/^[0-9a-fA-F]{64}$/); export class Registry { _endpoint: string _chain: Chain _client: RegistryClient static processWriteError(error: Error) { /** Example: { message: '{"code":18,"data":null,"log":"invalid request: Name already reserved.: failed to execute message; message index: 0","info":"","gasWanted":"200000","gasUsed":"86717","events":[],"codespace":"sdk"}', path: [ 'submit' ] }g */ console.error(error) const message = JSON.parse(error.message); return message.log || DEFAULT_WRITE_ERROR; } constructor(restUrl: string, gqlUrl: string, cosmosChainId = DEFAULT_CHAIN_ID) { if (!isUrl(restUrl)) { throw new Error('Path to a REST endpoint should be provided.'); } if (!isUrl(gqlUrl)) { throw new Error('Path to a GQL endpoint should be provided.'); } this._endpoint = restUrl; this._client = new RegistryClient(restUrl, gqlUrl); this._chain = { chainId: 9000, cosmosChainId } } /** * Get account by addresses. */ async getAccount(address: string) { return this._client.getAccount(address); } /** * Get records by attributes. */ async queryRecords(attributes: {[key: string]: any}, all = false, refs = false) { return this._client.queryRecords(attributes, all, refs); } /** * Resolve names to records. */ async resolveNames(names: string[], refs = false) { return this._client.resolveNames(names, refs); } /** * Publish record. * @param transactionPrivateKey - private key in HEX to sign transaction. */ async setRecord( params: { privateKey: string, record: any, bondId: string }, senderAddress: string, transactionPrivateKey: string, fee: Fee ) { let result; try { result = await this._submitRecordTx(params, senderAddress, transactionPrivateKey, fee); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Send coins. */ async sendCoins(params: MessageSendParams, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createMessageSend(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Computes the next bondId for the given account private key. */ async getNextBondId(address: string) { let result; try { const { account } = await this.getAccount(address); const accountObj = account.base_account; const nextSeq = parseInt(accountObj.sequence, 10) + 1; result = sha256(`${accountObj.address}:${accountObj.account_number}:${nextSeq}`); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return result; } /** * Get bonds by ids. */ async getBondsByIds(ids: string[]) { return this._client.getBondsByIds(ids); } /** * Query bonds by attributes. */ async queryBonds(attributes = {}) { return this._client.queryBonds(attributes); } /** * Create bond. */ async createBond(params: MessageMsgCreateBond, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgCreateBond(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Refill bond. */ async refillBond(params: MessageMsgRefillBond, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgRefillBond(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Withdraw (from) bond. */ async withdrawBond(params: MessageMsgWithdrawBond, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgWithdrawBond(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Cancel bond. */ async cancelBond(params: MessageMsgCancelBond, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgCancelBond(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Reserve authority. */ async reserveAuthority(params: MessageMsgReserveAuthority, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgReserveAuthority(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Set authority bond. * @param {string} name * @param {string} bondId * @param {string} privateKey * @param {object} fee */ async setAuthorityBond(params: MessageMsgSetAuthorityBond, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgSetAuthorityBond(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Lookup authorities by names. */ async lookupAuthorities(names: string[], auction = false) { return this._client.lookupAuthorities(names, auction); } /** * Set name (WRN) to record ID (CID). * @param {string} wrn * @param {string} id * @param {string} privateKey * @param {object} fee */ async setName(params: MessageMsgSetName, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgSetName(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Lookup naming information. */ async lookupNames(names: string[], history = false) { return this._client.lookupNames(names, history); } /** * Delete name (WRN) mapping. */ async deleteName(params: MessageMsgDeleteName, senderAddress: string, privateKey: string, fee: Fee) { let result; try { const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgDeleteName(this._chain, sender, fee, '', params) result = await this._submitTx(msg, privateKey, sender); } catch (err: any) { const error = err[0] || err; throw new Error(Registry.processWriteError(error)); } return parseTxResponse(result); } /** * Submit record transaction. * @param privateKey - private key in HEX to sign message. * @param txPrivateKey - private key in HEX to sign transaction. */ async _submitRecordTx( { privateKey, record, bondId }: { privateKey: string, record: any, bondId: string }, senderAddress: string, txPrivateKey: string, fee: Fee ) { if (!isKeyValid(privateKey)) { throw new Error('Registry privateKey should be a hex string.'); } if (!isKeyValid(bondId)) { throw new Error(`Invalid bondId: ${bondId}.`); } // Sign record. const recordSignerAccount = new Account(Buffer.from(privateKey, 'hex')); await recordSignerAccount.init(); const registryRecord = new Record(record); const payload = new Payload(registryRecord); await recordSignerAccount.signPayload(payload); // Send record payload Tx. return this._submitRecordPayloadTx({ payload, bondId }, senderAddress, txPrivateKey, fee); } async _submitRecordPayloadTx(params: MessageMsgSetRecord, senderAddress: string, privateKey: string, fee: Fee) { if (!isKeyValid(privateKey)) { throw new Error('Registry privateKey should be a hex string.'); } if (!isKeyValid(params.bondId)) { throw new Error(`Invalid bondId: ${params.bondId}.`); } const { account: { base_account: accountInfo } } = await this.getAccount(senderAddress); const sender = { accountAddress: accountInfo.address, sequence: accountInfo.sequence, accountNumber: accountInfo.account_number, pubkey: accountInfo.pub_key.key, } const msg = createTxMsgSetRecord(this._chain, sender, fee, '', params) return this._submitTx(msg, privateKey, sender); } /** * Submit a generic Tx to the chain. */ async _submitTx(message: any, privateKey: string, sender: Sender) { // Check private key. if (!isKeyValid(privateKey)) { throw new Error('Registry privateKey should be a hex string.'); } // Check that the account exists on-chain. const account = new Account(Buffer.from(privateKey, 'hex')); // Generate signed Tx. const transaction = createTransaction(message, account, sender, this._chain); const tx = generatePostBodyBroadcast(transaction, BroadcastMode.Block) // Submit Tx to chain. const { tx_response: response } = await this._client.submit(tx); return response; } } export { Account }