Package Management via NPM

TronBox comes standard with npm integration, and is aware of the node_modules directory in your project if it exists. This means you can use and distribute contracts, DApps, and libraries in Solidity via npm, making your code available to others and others' code available to you.

Package layout

Projects created with TronBox have a specific layout by default which enables them to be used as packages. This layout isn't required, but if used as a common convention–or "de-facto standard"–then distributing contracts and DApps through NPM will become much easier.
The most important directories in a TronBox package are the following:

  • /contracts
  • /build (which includes /build/contracts created by TronBox)

The first directory is your contracts directory, which includes your raw Solidity contracts. Including raw contracts in your package will allow others to import those contracts within their own solidity code. The second directory is the build directory, and more specifically /build/contracts, which holds build artifacts in the form of .json files. Including your .json build artifacts in your package will allow others to seamlessly interact with your contracts from JavaScript, which can be used in DApps, scripts and migrations.

Using a package

When using a package within your own project, it is important to note that there are two places where you might be interested in using other's contract code: within your contracts and within your Javascript code (migrations and tests). The following provides an example of each case, and discusses techniques for making the most of other's contracts and build artifacts.

Installing

For this example, we're going to use the Example TronBox Library, which provides a simple name registry that is deployed to the test network. In order to use it as a dependency, we must first install it within our project through npm:

$ cd my_project

$ npm install example-tronbox-library

Note that the last command above downloads the package and places it in my_project/node_modules directory, which is important for the examples below.See the npm documentation for help using npm to install packages.

Use a package within your contracts

To use a package's contracts within your contracts, this can be as simple as Solidity's import statement. When your import path isn't explicitly relative or absolute , this signifies to TronBox that you're looking for a file from a specifically named package. Consider this example using the Example TronBox Library mentioned above:

import "example-tronbox-library/contracts/SimpleNameRegistry.sol";

Since the path didn't start with ./, TronBox knows to look in your project's node_modules directory for the example-tronbox-library folder. From there, it resolves the path to provide you with the contract you requested.

Package's deployed addresses

Sometimes you want your contracts to interact with the package's previously deployed contracts. Since the deployed addresses exist within the package's .json files, you must perform an extra step to get those addresses into your contracts. To do so, make your contract accept the address of the dependency contract and then use migrations. The following is an example contract that exists within your project, as well as an example migration:

Contract: MyContract.sol

pragma solidity ^0.4.13;
 
import "example-tronbox-library/contracts/SimpleNameRegistry.sol";
 
contract MyContract {
  SimpleNameRegistry registry;
  address public owner;
 
  function MyContract {
    owner = msg.sender;
  }
 
  // Simple example that uses the deployed registry from the package.
  function getModule(bytes32 name) returns (address) {
    return registry.names(name);
  }
 
  // Set the registry if you're the owner.
  function setRegistry(address addr) {
    require(msg.sender == owner);
 
    registry = SimpleNameRegistry(addr);
  }
}

Migration: 3_hook_up_example_library.js

// Note that artifacts.require takes care of creating an abstraction for us.
var SimpleNameRegistry = artifacts.require('example-tronbox-library/SimpleNameRegistry');
 
module.exports = function (deployer) {
  // Deploy our contract, then set the address of the registry.
  deployer
    .deploy(MyContract)
    .then(function () {
      return MyContract.deployed();
    })
    .then(function (deployed) {
      return deployed.setRegistry(SimpleNameRegistry.address);
    });
};