About the Nano Ledger S

The Nano ledger S is an offline hardware wallet used for storing cryptocurrencies. It supports most of the different cryptocurrencies like bitcoin, ethereum and other altcoins. Lately, I’ve been working on a web application for a blockchain based startup. The cool part about the app was that the user would need to sign specific blockchain transactions with their private wallets. A requirement was that the web application needed to support the Nano ledger S for authentication and message signing.

Through this post, I will give a simple example of how you can set up your application to be able to communicate with a Nano ledger S, get the basic information from the wallet and sign transactions.

Communication

There are different ways how you can set up the communication between the Nano ledger S and your application. Some of them are:

  1. Node - with node-hid (via USB)
  2. Browser - using (U2F API)
  3. Bluetooth And several others.

All supported communication options can be found on Ledger GitHub repo.

In our project, we chose the U2F communication as we were going to use the Nano ledger S directly in our web application.

U2F requirements

U2F itself has its own requirements which are:

Solution

Let’s start by installing and adding the necessary libs to our project.

// terminal
yarn add @ledgerhq/hw-transport-u2f
yarn add @ledgerhq/hw-app-eth
// app.js
import Transport from "@ledgerhq/hw-transport-u2f";
import AppEth from "@ledgerhq/hw-app-eth";

As mentioned before we decided to communicate with our Ledger nano S through the browser which is why we are using the @ledgerhq/hw-transport-u2f library for transport. As our blockchain is ethereum based we also needed the @ledgerhq/hw-app-eth library to be able to communicate with our wallet.

Getting the address

To get the address we need to open a transport connection to the device and connect to our Ethereum app and pass the wallet BIP32 path. Once connected we can call the getAddress function on our wallet.

Getting the BIP32 path

You can get your wallet path with the Ledger Live app.

  1. Go to your account in the app
  2. Click on edit account
  3. Toggle Advanced Logs
  4. In the JSON object, you will find your account path under the field freshAddressPath.

Getting the account

const getAccount = async () => {
  const path = "44'/60'/0'/0/0";
  const transport = await Transport.create();
  const ethApp = new AppEth(transport);
  const result = await ethApp.getAddress(path);
  return result;
};

Signing a transaction

To sign a specific transaction we can use the signTransaction function provided by the @ledgerhq/hw-app-eth app. The function expects again the BIP32 path of the account and the raw transaction to be signed

Signing a raw transaction

export const signTransaction = (rawTx) => {
  const path = "44'/60'/0'/0/0";
  const transport = await Transport.create();
  const ethApp = new AppEth(transport);
  return ethApp.signTranscation(path, rawTx);
}; // returns a Promise

// example
signTransaction("e8018504e3b292008252089428ee52a8f3d6e5d15f8b131996950d7f296c7952872bd72a2487400080").then(result => {
  // do something with result
})
.catch(ex => {
  console.log(ex);
});

Complete solution

// ledger-api.js
import Transport from "@ledgerhq/hw-transport-u2f";
import AppEth from "@ledgerhq/hw-app-eth";

export const getAccount = async () => {
  const path = "44'/60'/0'/0/0";
  const transport = await Transport.create();
  const ethApp = new AppEth(transport);
  const result = await ethApp.getAddress(path);
  return result;
};

export const signTransaction = (rawTx) => {
  const path = "44'/60'/0'/0/0";
  const transport = await Transport.create();
  const ethApp = new AppEth(transport);
  return ethApp.signTranscation(path, rawTx);
};