1.0.3-alpha.0 • Published 2 years ago

near-willem-workspaces v1.0.3-alpha.0

Weekly downloads
-
License
(MIT AND Apache-2...
Repository
github
Last release
2 years ago

near-willem-workspaces for TypeScript/JavaScript

Controlled, concurrent workspaces in local NEAR Sandbox blockchains or on NEAR TestNet. Fun, deterministic testing and powerful scripting for NEAR.

Quick Start with AVA

near-willem-workspaces-ava is a thin wrapper around near-willem-workspaces-js designed to get you up and running as quickly as possible, with minimal configuration and power-boosts like TypeScript. You can install it with one command. You will need NodeJS installed. Then:

npx near-willem-workspaces-init

This command will:

  • Add a near-willem-workspaces directory to the folder where you ran the command. This directory contains all the configuration needed to get you started with near-willem-workspaces-ava, and a __tests__ subfolder with a well-commented example test file.
  • Create test.sh and test.bat scripts in the folder where you ran the command. These can be used to quickly run the tests in near-willem-workspaces. Feel free to integrate test-running into your project in a way that makes more sense for you, and then remove these scripts.
  • Install NPM dependencies using npm install. Most of the output you see when running the command comes from this step. You can skip this: npx near-willem-workspaces-init --no-install.

If you want to install near-willem-workspaces-ava manually, see its README.

How It Works

Let's look at some code that focuses on near-willem-workspaces itself, without any AVA or other testing logic.

  1. Initializing a Workspace. This will be used as the starting point for more workspaces soon.

    const workspaces = Workspace.init(async ({root}) => {
      const alice = await root.createAccount('alice');
      const contract = await root.createAndDeploy(
        'contract-account-name',
        'path/to/compiled.wasm'
      );
      return {alice, contract};
    });

    Let's step through this.

    1. Workspace.init initializes a new NEAR Sandbox node/instance. This is essentially a mini-NEAR blockchain created just for this test. Each of these Sandbox instances gets its own data directory and port, so that tests can run in parallel.
    2. This blockchain also has a root user. Mainnet has *.near, testnet has *.testnet, and these tests have *.${root.accountId}. This account name is not currently sandbox but might be in the future. Since it doesn't matter, you can think of it as being called sandbox while you're still figuring things out.
    3. root.createAccount creates a new subaccount of root with the given name, for example alice.sandbox.
    4. root.createAndDeploy creates a new subaccount with the given name, contract-account-name.sandbox, then deploys the specified Wasm file to it.
    5. path/to/compiled.wasm will resolve relative to your project root. That is, the nearest directory with a package.json file, or your current working directory if no package.json is found. To construct a path relative to your test file, you can use path.join(__dirname, '../etc/etc.wasm') (more info).
    6. After Workspace.create finishes running the function passed into it, it gracefully shuts down the Sandbox instance it ran in the background. However, it keeps the data directory around. That's what stores the state of the two accounts that were created (alice and contract-account-name with its deployed contract).
    7. workspace contains a reference to this data directory, so that multiple tests can use it as a starting point.
    8. The object returned, {alice, contract}, will be passed along to subsequent tests.
  2. Writing tests.

    near-willem-workspaces is designed for concurrency (which is why it's a great fit for AVA, which runs tests concurrently by default). Here's a simple way to get concurrent runs using plain JS (for a working example, see near-examples/rust-status-message):

    import {strict as assert} from 'assert';
    
    await Promise.all([
      workspace.fork(async ({alice, contract}) => {
        await alice.call(
          contract,
          'some_update_function',
          {some_string_argument: 'cool', some_number_argument: 42}
        );
        const result = await contract.view(
          'some_view_function',
          {account_id: alice}
        );
        assert.equal(result, 'whatever');
      }),
      workspace.fork(async ({alice, contract}) => {
        const result = await contract.view(
          'some_view_function',
          {account_id: alice}
        );
        // Note that we expect the value returned from `some_view_function` to be
        // a default here, because this `fork` runs *at the same time* as the
        // previous, in a separate local blockchain
        assert.equal(result, 'some default');
      });
    ]);

    Let's step through this.

    1. Like the earlier call to Workspace.init, each call to workspace.fork sets up its own Sandbox instance. Each will copy the data directory set up earlier as the starting point for its tests. Each will use a unique port so that tests can be safely run in parallel.
    2. call syntax mirrors near-cli and either returns the successful return value of the given function or throws the encountered error. If you want to inspect a full transaction and/or avoid the throw behavior, you can use call_raw instead.
    3. While call is invoked on the account doing the call (alice.call(contract, …)), view is invoked on the account being viewed (contract.view(…)). This is because the caller of a view is irrelevant and ignored.
    4. Gotcha: the full account names may or may not match the strings passed to createAccount and createAndDeploy, which is why you must write alice.call(contract, …) rather than alice.call('contract-account-name', …). But! The Account class overrides toJSON so that you can pass {account_id: alice} in arguments rather than {account_id: alice.accountId}. If you need the generated account ID in some other circumstance, remember to use alice.accountId.

See the __tests__ directory in this project for more examples.

"Spooning" Contracts from Testnet and Mainnet

Spooning a blockchain is copying the data from one network into a different network. near-willem-workspaces makes it easy to copy data from Mainnet or Testnet contracts into your local Sandbox environment:

await workspace.fork(async ({root}) => {
  const refFinance = await root.createAccountFrom({
    mainnetContract: 'v2.ref-finance.near',
    blockId: 50_000_000,
    withData: true,
  });
});

This would copy the Wasm bytes and contract state from v2.ref-finance.near to your local blockchain as it existed at block 50_000_000. This makes use of Sandbox's special patch state feature to keep the contract name the same, even though the top level account might not exist locally (note that this means it only works in Sandbox testing mode). You can then interact with the contract in a deterministic way the same way you interact with all other accounts created with near-willem-workspaces.

Gotcha: withData will only work out-of-the-box if the contract's data is 50kB or less. This is due to the default configuration of RPC servers; see the "Heads Up" note here. Some teams at NEAR are hard at work giving you an easy way to run your own RPC server, at which point you can point tests at your custom RPC endpoint and get around the 50kB limit.

See an example of spooning contracts at tests/05.spoon-contract-to-sandbox.ava.ts.

Running on Testnet

near-willem-workspaces is set up so that you can write tests once and run them against a local Sandbox node (the default behavior) or against NEAR TestNet. Some reasons this might be helpful:

  • Gives higher confidence that your contracts work as expected
  • You can test against deployed testnet contracts
  • If something seems off in Sandbox mode, you can compare it to testnet
  • Until we have a full-featured dev environment that includes Explorer, Wallet, etc, you can write full end-to-end tests using a tool like Cypress

You can run in testnet mode in three ways.

  1. When creating your Workspace, pass a config object as the first argument:

    const workspaces = Workspace.init(
      {network: 'testnet'},
      async ({root}) => { … }
    )
  2. Set the NEAR_WORKSPACES_NETWORK environment variable when running your tests:

    NEAR_WORKSPACES_NETWORK=testnet node test.js

    If you set this environment variable and pass {network: 'testnet'} to Workspace.init, the config object takes precedence.

  3. If using near-willem-workspaces-ava, you can use a custom config file. Other test runners allow similar config files; adjust the following instructions for your situation.

    Create a file in the same directory as your package.json called ava.testnet.config.cjs with the following contents:

    module.exports = {
      ...require('near-willem-workspaces-ava/ava.testnet.config.cjs'),
      ...require('./ava.config.cjs'),
    };

    The near-willem-workspaces-ava/ava.testnet.config.cjs import sets the NEAR_WORKSPACES_NETWORK environment variable for you. A benefit of this approach is that you can then easily ignore files that should only run in Sandbox mode. See this project's testnet config for an example.

    Now you'll also want to add a test:testnet script to your package.json's scripts section:

     "scripts": {
       "test": "near-willem-workspaces-ava",
    +  "test:testnet": "near-willem-workspaces-ava --config ./ava.testnet.config.cjs"
     }

Stepping through a testnet example

Let's revisit a shortened version of the example from How It Works above, describing what will happen in Testnet.

  1. Create a Workspace.

    const workspaces = Workspace.init(async ({root}) => {
      await root.createAccount('alice');
      await root.createAndDeploy(
        'contract-account-name',
        'path/to/compiled.wasm'
      );
    });

    Workspace.init does not interact with Testnet at all yet. Instead, the function runs at the beginning of each subsequent call to workspace.fork. This matches the semantics of the sandbox that all subsequent calls to fork have the same starting point, however, testnet requires that each forkd workspace has its own root account. In fact Workspace.init creates a unique testnet account and each test is a unique sub-account.

    If you want to run a single script on Testnet, you can use Workspace.open:

    Workspace.open(async ({root}) => {
      // Anything here will run right away, rather than needing a subsequent `workspace.fork`
    })
  2. Write tests.

    await Promise.all([
      workspace.fork(async ({alice, contract}) => {
        await alice.call(
          contract,
          'some_update_function',
          {some_string_argument: 'cool', some_number_argument: 42}
        );
        const result = await contract.view(
          'some_view_function',
          {account_id: alice}
        );
        assert.equal(result, 'whatever');
      }),
      workspace.fork(async ({alice, contract}) => {
        const result = await contract.view(
          'some_view_function',
          {account_id: alice}
        );
        assert.equal(result, 'some default');
      });
    ]);

    Each call to workspace.fork will:

    • Get or create its own sub-account on testnet account, e.g. t.rdsq0289478. If creating the account the keys will be stored at $PWD/.near-credentials/workspaces/testnet/t.rdsq0289478.json.
    • Run the initFn passed to Workspace.init
    • Create sub-accounts for each createAccount and createAndDeploy, such as alice.t.rdsq0289478
    • If the test account runs out of funds to create accounts it will request a transfer from the root account.
    • After the test is finished each account created is deleted and the funds sent back to the test account.

Note: Since the testnet accounts are cached, if account creation rate limits are reached simply wait a little while and try again.

Skipping Sandbox-specific tests

If some of your runs take advantage of Sandbox-specific features, you can skip these on testnet in a few ways:

  1. runSandbox: Instead of workspace.fork, you can use workspace.forkSandbox:

    await Promise.all([
      workspace.fork(async ({…}) => {
        // runs on any network, sandbox or testnet
      }),
      workspace.runSandbox(async ({…}) => {
        // only runs on sandbox network
      });
    ]);
  2. Workspace.networkIsSandbox: You can also skip entire sections of your files by checking Workspace.networkIsSandbox (Workspace.networkIsTestnet and Workspace.getNetworkFromEnv are also available).

    let workspaces = Workspace.init(async ({root}) => ({ // note the implicit return
      contract: await root.createAndDeploy(
        'contract-account-name',
        'path/to/compiled.wasm'
      )
    }));
    workspace.fork('thing that makes sense on any network', async ({…}) => {
      // logic using basic contract & account interactions
    });
    if (Workspace.networkIsSandbox) {
      workspace.fork('thing that only makes sense with sandbox', async ({…}) => {
        // logic using patch-state, fast-forwarding, etc
      });
    }
  3. Use a separate testnet config file, as described under the "Running on Testnet" heading above.

Patch State on the Fly

In Sandbox-mode tests, you can add or modify any contract state, contract code, account or access key with patchState.

You cannot perform arbitrary mutation on contract state with transactions since transactions can only include contract calls that mutate state in a contract-programmed way. For example, with an NFT contract, you can perform some operation with NFTs you have ownership of, but you cannot manipulate NFTs that are owned by other accounts since the smart contract is coded with checks to reject that. This is the expected behavior of the NFT contract. However, you may want to change another person's NFT for a test setup. This is called "arbitrary mutation on contract state" and can be done with patchState. Alternatively you can stop the node, dump state at genesis, edit genesis, and restart the node. The later approach is more complicated to do and also cannot be performed without restarting the node.

It is true that you can alter contract code, accounts, and access keys using normal transactions via the DeployContract, CreateAccount, and AddKey actions. But this limits you to altering your own account or sub-account. patchState allows you to perform these operations on any account.

To see an example of how to do this, see the patch-state test.

near-willem-workspaces will support expanded patchState-based functionality in the future:

Pro Tips

  • NEAR_WORKSPACES_DEBUG=true – run tests with this environment variable set to get copious debug output and a full log file for each Sandbox instance.

  • Workspace.init config – you can pass a config object as the first argument to Workspace.init. This lets you do things like:

    • skip initialization if specified data directory already exists

      Workspace.init(
        { init: false, homeDir: './test-data/alice-owns-an-nft' },
        async ({root}) => { … }
      )
    • always recreate such data directory instead (the default behavior)

    • specify which port to run on

    • and more!