Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Unit test] role based key fee payer #59

Open
wants to merge 7 commits into
base: dev
Choose a base branch
from
32 changes: 16 additions & 16 deletions web3js-ext/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

141 changes: 141 additions & 0 deletions web3js-ext/test/role_based_key_feepayer.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,141 @@
import { Web3, TxType, AccountKeyType, getPublicKeyFromPrivate, toPeb } from "@kaiachain/web3js-ext";
import { assert } from "chai";

const provider = new Web3.providers.HttpProvider("https://public-en-kairos.node.kaia.io");
const web3 = new Web3(provider);

type Account = {
address: string;
privateKey: string;
signTransaction: (tx: any) => Promise<any>;
}

// Feedback1. Generate Temporary Key.
function generateTemporaryAccount(): Account {
return web3.eth.accounts.create();
}

describe("Role-based Key Tests", function () {
this.timeout(10000);

let roleTransactionAccount: Account;
let roleAccountUpdate: Account;
let roleFeePayerAccount: Account;

// Before all tests, set up Role-based Key
before(async function () {
console.log("\n--- Generating Temporary Accounts ---");
roleTransactionAccount = generateTemporaryAccount();
roleAccountUpdate = generateTemporaryAccount();
roleFeePayerAccount = generateTemporaryAccount();
Copy link
Author

@minminkikiki minminkikiki Nov 16, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[Feedback 1] fixed this test to generate temporary key and update the account key to role based.
cc. @kjeom


const pub1 = getPublicKeyFromPrivate(roleTransactionAccount.privateKey);
const pub2 = getPublicKeyFromPrivate(roleAccountUpdate.privateKey);
const pub3 = getPublicKeyFromPrivate(roleFeePayerAccount.privateKey);
kjeom marked this conversation as resolved.
Show resolved Hide resolved

console.log("Generated Public keys:", { pub1, pub2, pub3 });
kjeom marked this conversation as resolved.
Show resolved Hide resolved

const updateTx = {
type: TxType.AccountUpdate,
from: roleAccountUpdate.address,
gasLimit: 100000,
key: {
type: AccountKeyType.RoleBased,
keys: [
{ type: AccountKeyType.Public, key: pub1 },
{ type: AccountKeyType.Public, key: pub2 },
{ type: AccountKeyType.Public, key: pub3 }
]
}
};

const signedUpdateTx = await roleAccountUpdate.signTransaction(updateTx);
kjeom marked this conversation as resolved.
Show resolved Hide resolved
console.log("Account Updated:", signedUpdateTx);
assert.isNotNull(signedUpdateTx.transactionHash, "Account update transaction should succeed");
});

// Test Case 1: Sending a normal transaction with RoleTransaction key
it("1. Sending a normal transaction with RoleTransaction key", async function () {
const valueTx = {
type: TxType.ValueTransfer,
from: generateTemporaryAccount().address,
to: generateTemporaryAccount().address,
value: toPeb("0.01"),
gasLimit: 100000
};

const signedTx = await roleTransactionAccount.signTransaction(valueTx);
console.log("RoleTransaction signedTx:", signedTx.transactionHash);
assert.isNotNull(signedTx.transactionHash, "RoleTransaction transaction should succeed");
});

// Test Case 2: Attempting to sign a regular transaction with RoleFeePayer key (should fail)
it("2. Attempting to sign a regular transaction with RoleFeePayer key (failure test)", async function () {
const valueTx = {
type: TxType.ValueTransfer,
from: generateTemporaryAccount().address,
to: generateTemporaryAccount().address,
value: toPeb("0.01"),
gasLimit: 100000
};

try {
const signedTx = await roleFeePayerAccount.signTransaction(valueTx);
kjeom marked this conversation as resolved.
Show resolved Hide resolved
console.log("Unexpected Success - Signed Transaction:", signedTx);
assert.fail("RoleFeePayer key should not sign a ValueTransfer transaction.");
} catch (error: any) {
console.log("Expected Error (RoleFeePayer):", error.message);
assert.isTrue(true, "Error occurred as expected due to role mismatch");
}
});

// Test Case 3: Fee Delegated transaction signed by RoleFeePayer key (should succeed)
it("3. Fee Delegated transaction signed by RoleTransaction and RoleFeePayer keys", async function () {
const feeDelegatedTx = {
type: TxType.FeeDelegatedValueTransfer,
from: generateTemporaryAccount().address,
to: generateTemporaryAccount().address,
value: toPeb("0.01"),
gasLimit: 100000
};

// Feedback2. Remove the dependencies with network connection. just check the signed result.
// Feedback3. signed by a user (from address,roleTransactionAccount) first
const signedTxBySender = await roleTransactionAccount.signTransaction(feeDelegatedTx);
assert.isNotNull(signedTxBySender.rawTransaction, "Transaction should be signed by RoleTransaction");
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[Feedback 2] don't send the tx to the testnet, just check the signed result.
cc. @kjeom


console.log("Sender signed transaction:", signedTxBySender.rawTransaction);

// Feedback3. FeePayer(FeePayerAccount) sign the tx after the user's sign.
try {
const signedTxByFeePayer = await roleFeePayerAccount.signTransaction({
senderRawTransaction: signedTxBySender.rawTransaction
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[Feedback 3] It should be signed by a user (from address) first. FeePayer normally sign the tx after the user's sign.

Actually, I have two questions about this Feedback3.

  1. when you said "It should be signed by a user (from address) first. FeePayer normally sign the tx after the user's sign.", Is it correct to sign with roleTransactionAccount(user who signs the transaction account) first and then append roleFeePayerAccount to the rawTransaction of that signature?

  2. If there is one unexpected issue in this test, it is “Returned error: insufficient funds for transfer”. I created temporary addresses via [Feedback 1], but now that accounts don't have enough klay(kaia). Is there a good way to charge klay(kaia) to the temporary accounts...? (I can't think of a good way... T^T...)

cc. @kjeom

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@minminkikiki "A user" means another account that is used by user, not feePayer.
roleTransactionKey and roleFeePayerKey are owned by feePayer. That means all keys of each role belong to another key owner who is not the user.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@minminkikiki If you got the new error regarding the sign of feepayer, please add the description of the error in the PR description.

});
assert.isNotNull(signedTxByFeePayer.rawTransaction, "FeePayer should sign the transaction");
console.log("Signed Fee Delegated Transaction:", signedTxByFeePayer.rawTransaction);
} catch (error: any) {
console.log("Expected Error (FeePayer signing):", error.message);
assert.fail("FeePayer failed to sign the already signed transaction.");
}
});

// Test Case 4: Attempting to sign Fee Delegated transaction with RoleTransaction key (should fail)
it("4. Attempting to sign Fee Delegated transaction with RoleTransaction key (should fail)", async function () {
const feeDelegatedTx = {
type: TxType.FeeDelegatedValueTransfer,
from: generateTemporaryAccount().address,
to: generateTemporaryAccount().address,
value: toPeb("0.01"),
gasLimit: 100000
};

try {
const signedTx = await roleTransactionAccount.signTransaction(feeDelegatedTx);
console.log("Unexpected Success - Transaction Hash:", signedTx.transactionHash);
assert.fail("RoleTransaction key should not sign Fee Delegated transactions");
} catch (error: any) {
console.log("Expected Error (RoleTransaction as FeePayer):", error.message);
assert.isTrue(true, "Error occurred as expected");
}
});
});