123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257 |
- import { z } from "zod";
- import { DataParser, RJSVM_Builder, RJSVM_Implementations, RJSVM_Interface, RJSVM, RJSVM_Config } from "../src/main";
-
- //Test requirements
- import { assert, expect } from 'chai';
- import { describe, it } from "mocha";
- import { makeTestnetWallet } from "./tools";
- import { Wallet } from "xrpio";
- import { Datawriter } from "../src/RJSVM/datawriter/datawriter";
- import { InsufficientFeeError, RestrictedAccessError } from "../src/util/errors";
- var should = require('chai').should();
- var chai = require('chai');
- var chaiAsPromised = require('chai-as-promised');
- chai.use(chaiAsPromised);
-
-
- const xrpNode = "wss://s.altnet.rippletest.net:51233"
-
- let ownerWallet: Wallet //the RJSVM owner
- let userWallet: Wallet //a RJSVM user
- let listeningWallet: Wallet //wallet the RJSVM listens to
- let drainWallet: Wallet //random wallet to send stuff to, could be any user-controlled secondary wallet
- let rjsvm: RJSVM
-
- let user_datawriter: Datawriter
- let owner_datawriter: Datawriter
-
-
- const setup = async () => {
- // [ownerWallet, listeningWallet, userWallet, drainWallet] = await Promise.all([makeTestnetWallet(),makeTestnetWallet(),makeTestnetWallet(),makeTestnetWallet()])
- [ownerWallet, listeningWallet, userWallet, drainWallet] = [
- {
- secret: 'sEd7tPDHFk3w7ABSaCPYi12iwucMQLt',
- address: 'rDML7QhJeVxZTwsjPtm4iyLUJTCzzuppMo'
- },
- {
- secret: 'sEdSgbyouvLM26vq2F2XthfKGWndP72',
- address: 'rPvwbb4JHncG9rKVrYATJw6HRixR69wgjg'
- },
- {
- secret: 'sEd7c1Y8JwWyA2owbchMWnuzAhB3ypY',
- address: 'rwuuWdyxwf3o78Ce2DAZrYCQ12SSawTp6M'
- },
- {
- secret: 'sEdScantwGA5gYe4atKPbG5bsriURpf',
- address: 'rDiwSioTRyV29rNk2FskxqFtMaRsmWvS7C'
- }
- ]
- owner_datawriter = new Datawriter({
- receiveAddress: drainWallet.address,
- sendWallet: ownerWallet,
- xrpNode: xrpNode,
- contractAddress: listeningWallet.address
- })
-
- user_datawriter = new Datawriter({
- receiveAddress: drainWallet.address,
- sendWallet: userWallet,
- xrpNode: xrpNode,
- contractAddress: listeningWallet.address
- })
-
- //await owner_dw.callEndpoint('submit', { title: "1", body: "2", from: "3" }, 100)
- //await owner_dw.callEndpoint('submit', { title: "1", body: "2", from: "3" }, 100)
- //await user_dw.callEndpoint('restricted', { title: "user", body: "user", from: "user" })
- //await owner_dw.callEndpoint('restricted', { title: "owner", body: "owner", from: "owner" })
-
- // #########################
- // Define parameter types
- // #########################
-
- const shoutSchema = z.object({
- title: z.string(),
- body: z.string(),
- from: z.string(),
- id: z.optional(z.string())
- })
- type Shout = z.infer<typeof shoutSchema>
-
- type State = {
- shouts: Shout[]
- }
-
- // #########################
- // Define endpoints
- // #########################
-
- type RJSVM_Endpoints = {
- submit: (data: Shout) => void
- restricted: (data: Shout) => void
- }
-
- // #########################
- // Define init state
- // #########################
-
- abstract class RJSVM_Base
- extends RJSVM<State, RJSVM_Endpoints>
- implements RJSVM_Interface<RJSVM_Base> {
- owner = ownerWallet.address
-
- state: State = {
- shouts: []
- }
- }
-
- // #########################
- // Implement logic
- // #########################
-
- const RJSVM_Contract: RJSVM_Implementations<RJSVM_Base> = {
- submit: {
- implementation: function (env, shout) {
- this.state.shouts.unshift(shout)
- },
- visibility: 'public',
- fee: 10,
- parameterSchema: shoutSchema
- },
-
- restricted: {
- implementation: function (env, shout) {
- this.state.shouts.unshift(shout)
- },
- visibility: 'owner',
- parameterSchema: shoutSchema
- },
- }
-
- // #########################
- // Build and connect
- // #########################
-
- const Rjsvm = RJSVM_Builder.from(RJSVM_Base, RJSVM_Contract);
-
- const conf: RJSVM_Config = {
- listeningAddress: listeningWallet.address,
- rippleNode: xrpNode
- }
-
- rjsvm = new Rjsvm(conf)
- await rjsvm.connect()
- }
-
- describe('RJSVM basic functions', () => {
-
- before(async function () {
- this.timeout(100000)
- await setup()
- })
-
- after(async () => {
- await rjsvm.disconnect()
- })
-
- it('Env contains the payload carrying transaction', function (done) {
- this.timeout(45000)
-
- console.log("Making wallet...")
- makeTestnetWallet().then(async testWallet => {
- console.log(testWallet)
- //create a mock RJSVM
- abstract class RJSVM_Base
- extends RJSVM<undefined, { testEndpoint: () => void }>
- implements RJSVM_Interface<RJSVM_Base> {
- owner = ownerWallet.address
- state = undefined
- }
-
- //Implementation of the 'test' endpoint
- const RJSVM_Contract: RJSVM_Implementations<RJSVM_Base> = {
- testEndpoint: {
- implementation: function (env) {
- expect(env.Account).to.be.equal(userWallet.address)
- expect(env.Amount).to.be.equal('1')
- expect(env.Destination).to.be.equal(testWallet.address)
- expect(Number(env.Fee)).to.be.greaterThanOrEqual(10)
- expect(env.LastLedgerSequence).to.be.a('number')
- expect(env.Memos).to.be.an('Array')
- expect(env.Sequence).to.be.a('number')
- expect(env.SigningPubKey).to.be.a('string')
- expect(env.TransactionType).to.be.equal('Payment')
- expect(env.TxnSignature).to.be.a('string')
- expect(env.date).to.be.a('number')
- expect(env.hash).to.be.a('string')
- expect(env.inLedger).to.be.a('number')
- expect(env.ledger_index).to.be.a('number')
- runnable.disconnect()
- done()
- },
- visibility: 'public',
- parameterSchema: z.any()
- },
- }
-
- const Rjsvm = RJSVM_Builder.from(RJSVM_Base, RJSVM_Contract)
- const runnable = new Rjsvm({ listeningAddress: testWallet.address, rippleNode: xrpNode })
- await runnable.connect()
-
- const dw = new Datawriter({
- contractAddress: testWallet.address,
- sendWallet: userWallet,
- receiveAddress: drainWallet.address,
- xrpNode: xrpNode
- })
- dw.callEndpoint('testEndpoint', "")
-
- })
- })
-
- it('Called endpoint triggers in RJSVM', function (done) {
- this.timeout(45000)
-
- const data = { title: "1", body: "2", from: "3" };
- rjsvm.once('submit', (payload) => {
- expect(payload).to.be.an('object')
- expect(payload).to.deep.equal(data)
- done()
- })
- owner_datawriter.callEndpoint('submit', data, 10)
- })
-
- it('Calling an endpoint with insufficient fee fails', function (done) {
- this.timeout(45000)
-
- const data = { title: "f", body: "f", from: "f" };
- rjsvm.once('error', (err) => {
- expect(err).to.be.instanceOf(InsufficientFeeError)
- done()
- })
- owner_datawriter.callEndpoint('submit', data, 1)
- })
-
- it('Restricted endpoint can be called by owner', function (done) {
- this.timeout(45000)
-
- const data = { title: "11", body: "22", from: "33" };
- rjsvm.once('restricted', (payload) => {
- expect(payload).to.be.an('object')
- expect(payload).to.deep.equal(data)
- done()
- })
- owner_datawriter.callEndpoint('restricted', data)
- })
-
-
- it('Restricted endpoint cannot be called by non-owner', function (done) {
- this.timeout(45000)
-
- const data = { title: "e", body: "e", from: "e" };
- rjsvm.once('error', (err) => {
- expect(err).to.be.instanceOf(RestrictedAccessError)
- done()
- })
- user_datawriter.callEndpoint('restricted', data)
- })
- })
|