2

I am trying to implement hyperledger sawtooth transaction through javascript SDK following this https://sawtooth.hyperledger.org/docs/core/releases/1.0/_autogen/sdk_submit_tutorial_js.html#encoding-your-payload.

/*
*Create the transaction header
*/
const createTransactionHeader = function createTransactionHeader(payloadBytes) {

    return  protobuf.TransactionHeader.encode({
        familyName: 'intkey',
        familyVersion: '1.0',
        inputs: [],
        outputs: [],
        signerPublicKey: '02cb65a26f7af4286d5f8118400262f7790e20018f2d01e1a9ffc25de1aafabdda',

        batcherPublicKey: '02cb65a26f7af4286d5f8118400262f7790e20018f2d01e1a9ffc25de1aafabdda',
        dependencies: [],
        payloadSha512: createHash('sha512').update(payloadBytes).digest('hex')
    }).finish();


}
/*
* Create the transactions
*/
const createTransaction = function createTransaction(transactionHeaderBytes, payloadBytes) {

    const signature = signer.sign(transactionHeaderBytes)

    return transaction = protobuf.Transaction.create({
        header: transactionHeaderBytes,
        headerSignature: Buffer.from(signature, 'utf8').toString('hex'),
        payload: payloadBytes
    });
}

While submitting the transaction I am getting the following error from REST API

{
  "error": {
    "code": 30,
    "message": "The submitted BatchList was rejected by the validator. It was poorly formed, or has an invalid signature.",
    "title": "Submitted Batches Invalid"
  }
}

Found the following issue similar to my problem

Sawtooth Invalid Batch or Signature

But its implemented in java the solution not work for my case

kboul
  • 13,836
  • 5
  • 42
  • 53
shamon shamsudeen
  • 5,466
  • 17
  • 64
  • 129

1 Answers1

0

This should work, try this:

const cbor = require('cbor');
const {createContext, CryptoFactory} = require('sawtooth-sdk/signing');
const {createHash} = require('crypto');
const {protobuf} = require('sawtooth-sdk');
const request = require('request');
const crypto = require('crypto');

const context = createContext('secp256k1');
const privateKey = context.newRandomPrivateKey();
const signer = CryptoFactory(context).newSigner(privateKey);


// Here's how you can generate the input output address
const FAMILY_NAMESPACE = crypto.createHash('sha512').update('intkey').digest('hex').toLowerCase().substr(0, 6);
const address = FAMILY_NAMESPACE + crypto.createHash('sha512').update('foo').digest('hex').toLowerCase().substr(0, 64);

const payload = {
  Verb: 'set',
  Name: 'foo',
  Value: 42
};

const payloadBytes = cbor.encode(payload);

const transactionHeaderBytes = protobuf.TransactionHeader.encode({
  familyName: 'intkey',
  familyVersion: '1.0',
  inputs: [address],
  outputs: [address],
  signerPublicKey: signer.getPublicKey().asHex(),
  batcherPublicKey: signer.getPublicKey().asHex(),
  dependencies: [],
  payloadSha512: createHash('sha512').update(payloadBytes).digest('hex')
}).finish();

const transactionHeaderSignature = signer.sign(transactionHeaderBytes);

const transaction = protobuf.Transaction.create({
  header: transactionHeaderBytes,
  headerSignature: transactionHeaderSignature,
  payload: payloadBytes
});

const transactions = [transaction]

const batchHeaderBytes = protobuf.BatchHeader.encode({
  signerPublicKey: signer.getPublicKey().asHex(),
  transactionIds: transactions.map((txn) => txn.headerSignature),
}).finish();

const batchHeaderSignature = signer.sign(batchHeaderBytes)

const batch = protobuf.Batch.create({
  header: batchHeaderBytes,
  headerSignature: batchHeaderSignature,
  transactions: transactions
};

const batchListBytes = protobuf.BatchList.encode({
  batches: [batch]
}).finish();

request.post({
  url: 'http://rest.api.domain/batches',
  body: batchListBytes,
  headers: {'Content-Type': 'application/octet-stream'}
}, (err, response) => {
  if(err) {
    return console.log(err);
  }

  console.log(response.body);
});
Rohit Khatri
  • 1,980
  • 3
  • 25
  • 45
  • How can i get input and output – shamon shamsudeen Jul 02 '18 at 06:35
  • Do you have any comments for this https://stackoverflow.com/questions/51131415/hyperledger-sawtooth-transaction-processor-in-javacript – shamon shamsudeen Jul 02 '18 at 07:54
  • @iambatman I have updated the answer with `input output` address generating code. – Rohit Khatri Jul 02 '18 at 08:10
  • @iambatman basically, It's a resource address which will keep a state of some asset and will be changed through transactions, and It's prefixed with the `family namespace` to keep it unique. – Rohit Khatri Jul 02 '18 at 08:12
  • When i create agent signup for the first time, i am storing private key in my db for further transactions. But if i try to submit transaction with the same private key getting error. The error is ´Private key should be a buffer´. How can i sign transaction with the same private key. – U rock Jan 28 '19 at 14:45