DokoJS is a powerful and lightweight library designed for seamless interaction with the Aleo blockchain and its diverse ecosystem. Drawing inspiration from the zk-gaming-toolkit, Dokojs fully harnesses existing tools while offering a user-friendly interface for developers keen on building atop the Aleo blockchain.
Before beginning, make sure you have the following set up:
1. Rust: Installation Guide
2. Leo language: Installation Guide
Install dokojs globally using npm:
npm install -g @doko-js/cli@latest
In case pnpm is not set up, follow the pnpm installation guide
# Download the source file
git clone https://github.com/venture23-aleo/doko-js
cd doko-js
# Install the dependencies
pnpm install
# Build the project
npm run build
# Install dokojs
npm run install:cli
To use DokoJS, in your terminal, run:
dokojs
The expected output is as following:
____ _ _ ____
| _ \ ___ | | _____ | / ___|
| | | |/ _ \| |/ / _ \ _ | \___ \
| |_| | (_) | | (_) | |_| |___) |
|____/ \___/|_|\_\___/ \___/|____/
Usage: dokojs [options] [command]
DokoJS CLI
Options:
-V, --version output the version number
-h, --help display help for command
Commands:
init [options] <project-name> Initialize your DokoJS project
add <program-name> Add a new component or resource
compile Compile your DokoJS project
unflatten Create leo build for programs
autogen Generate ts types for contracts - use only after the build has been generated
run [options] <file> Run file
deploy [options] <program-name> Deploy program
execute <file-path> Execute script
help [command] display help for command
Initialize a new project by giving the name of the project.
dokojs init <PROJECT_NAME>
Let's create a new project called token
.
This will internally install the dependencies for the project.
After project initialization, DokoJS generates the following directory structure:
├── contract
│ └── base-contract.ts
├── node_modules/
├── programs
│ ├── sample_program.leo
│ └── token.leo
├── scripts
│ └── deploy.ts
├── test
│ ├── sample_program.test.ts
│ └── token.test.ts
├── .env
├── .gitignore
├── aleo-config.js
├── babel.config.json
├── jest.config.json
├── package-lock.json
├── package.json
├── README.md
├── node_modules/
└── tsconfig.json
After initializing a projects, it will create following directories:
- contract: This directory contains a single file -
base-contract.ts
, a class containing common method and config for aleo programs in js. - programs: This directory is made to hold all the leo programs.
token.leo
file is created during the initialization along withsample_program.leo
- scripts: This directory is made to hold specific tasks and user scripts.
- test: This directory is made to hold all the tests.
aleo-config.js
-> This is a configuration file consisting of private key, method to execute on chain or dry run and different networks.
Let's explore aleo-config.js
.
import dotenv from 'dotenv';
dotenv.config();
export default {
accounts: [process.env.ALEO_PRIVATE_KEY],
mode: 'execute',
mainnet: {},
networks: {
testnet: {
endpoint: 'http://localhost:3030',
accounts: [process.env.ALEO_PRIVATE_KEY_TESTNET3,
process.env.ALEO_DEVNET_PRIVATE_KEY2]
priorityFee: 0.01
},
mainnet: {
endpoint: 'https://api.explorer.aleo.org/v1',
accounts: [process.env.ALEO_PRIVATE_KEY_MAINNET],
priorityFee: 0.001
}
},
defaultNetwork: 'testnet'
};
We have two modes of execution supported:
execute
: In this mode, proof is generated and broadcasted on chain. Internally, it callsleo developer execute
command.evaluate
: In this mode, no proof is generated and broadcasted on chain. Internally, it callsleo run
command.
aleo-config
acts as a default configuation for the entire project. It can be overwritten on per program basis as well.
To add a new program create a new file inside the programs/
directory.
To modify the existing file, simply modify the existing file or run command
dokojs add [PROGRAM_NAME]
To compile the project, run:
dokojs compile
This will create the artifacts
folder. The artifacts folder has the two main directories:
- leo - This directory contains the Leo packages. For each program in
programs
directory, a corresponding Leo package is created. The leo code fromprograms
is copied to thesrc/main.leo
file and is then compiled. If the compilation for all the programs are successful, the generated.aleo
files are parsed to generate Leo and JS types which are inside the js directory. - js - This directory contains the both Leo types, JS types,
js2leo
(Leo type to Js type conversion),leo2js
(JS type to Leo type conversion). For each of the program, it also creates .ts file that contains all the transitions and mappings of the program.
├── js
│ ├── js2leo
│ │ ├── index.ts
│ │ └── token.ts
│ ├── leo2js
│ │ ├── index.ts
│ │ └── token.ts
│ └── types
│ ├── index.ts
│ └── token.ts
│ ├── sample_program.ts
│ ├── token.ts
└── leo
├── sample_program
│ ├── README.md
│ ├── build
│ ├── inputs
│ ├── leo.lock
│ ├── outputs
│ ├── program.json
│ └── src
└── token
├── README.md
├── build
├── inputs
├── leo.lock
├── outputs
├── program.json
└── src
Since,
sample_program
did not have any types, there is no need for js2leo, leo2js or type file.
After the successful compilation, tests can be written based on the files generated after compilation.
A sample test file is created for both the sample_program
and token
program. Run all the tests with:
npm run test
If you want to test a particular test file:
npm run test -- sample_program.test.ts
Pro Tip: You don't need to enter the full test file name. You can use part of the name of the file and the tests that matches the entered name will run. Example:
npm run test -- sample
This documentation provides a comprehensive guide to installing Dokojs, starting a project, adding programs, compiling, running tests, and deploying programs. Happy coding with Dokojs!
Create a test file (e.g., token.test.ts) inside the test directory. An example test file is provided below:
import { ExecutionMode } from '@doko-js/core';
import { TokenContract } from '../artifacts/js/token';
import { decrypttoken } from '../artifacts/js/leo2js/token';
import { PrivateKey } from '@provablehq/sdk';
const TIMEOUT = 200_000;
const amount = BigInt(2);
// Available modes are evaluate | execute (Check README.md for further description)
const mode = ExecutionMode.SnarkExecute;
// Contract class initialization
const contract = new TokenContract({ mode });
// This maps the accounts defined inside networks in aleo-config.js and return array of address of respective private keys
const [admin] = contract.getAccounts();
const recipient = process.env.ALEO_DEVNET_PRIVATE_KEY3;
describe('deploy test', () => {
test('deploy', async () => {
if ((mode as ExecutionMode) == ExecutionMode.SnarkExecute) {
const tx = await contract.deploy();
await tx.wait();
}
}, 10000000);
test('mint public', async () => {
const actualAmount = BigInt(100000);
const tx = await contract.mint_public(admin, actualAmount);
await tx.wait();
const expected = await contract.account(admin);
expect(expected).toBe(actualAmount);
}, 10000000);
test('mint private', async () => {
const actualAmount = BigInt(100000);
const tx = await contract.mint_private(
'aleo1rhgdu77hgyqd3xjj8ucu3jj9r2krwz6mnzyd80gncr5fxcwlh5rsvzp9px',
actualAmount
);
const [record1] = await tx.wait();
// @NOTE Only decrypt in SnarkExecute use JSON.parse in LeoRun
const decryptedRecord = decrypttoken(
record1,
process.env.ALEO_PRIVATE_KEY_TESTNET3
);
expect(decryptedRecord.amount).toBe(actualAmount);
}, 10000000);
test(
'private transfer to user',
async () => {
const account = contract.config.privateKey;
const amount1 = BigInt(1000000000);
const amount2 = BigInt(100000000);
const mintTx = await contract.mint_private(admin, amount1);
const [result] = await mintTx.wait();
const decryptedRecord = decrypttoken(result, account);
const receiptAddress = PrivateKey.from_string(recipient)
.to_address()
.to_string();
const tx = await contract.transfer_private(
decryptedRecord,
receiptAddress,
amount
);
const [record1, record2] = await tx.wait();
const decryptedRecord2 = decrypttoken(record1, account);
expect(decryptedRecord2.amount).toBe(amount1 - amount2);
},
TIMEOUT
);
});
Execute the test file:
npm test -- --runInBand token.test.ts