Skip to content

Commit

Permalink
feat: tx_validator checks fn selectors
Browse files Browse the repository at this point in the history
  • Loading branch information
alexghr committed Apr 10, 2024
1 parent 80c498d commit 684b5d0
Show file tree
Hide file tree
Showing 27 changed files with 952 additions and 674 deletions.
25 changes: 20 additions & 5 deletions yarn-project/circuit-types/src/interfaces/configs.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,19 @@
import { type AztecAddress, type EthAddress, type Fr } from '@aztec/circuits.js';
import { type AztecAddress, type EthAddress, type Fr, type FunctionSelector } from '@aztec/circuits.js';

/** A function that the sequencer allows to run in either setup or teardown phase */
export type AllowedFunction =
| {
/** The contract address this selector is valid for */
address: AztecAddress;
/** The function selector */
selector: FunctionSelector;
}
| {
/** The contract class this selector is valid for */
classId: Fr;
/** The function selector */
selector: FunctionSelector;
};

/**
* The sequencer configuration.
Expand All @@ -19,9 +34,9 @@ export interface SequencerConfig {
/** The path to the ACVM binary */
acvmBinaryPath?: string;

/** The list of permitted fee payment contract classes */
allowedFeePaymentContractClasses?: Fr[];
/** The list of functions calls allowed to run in setup */
allowedFunctionsInSetup?: AllowedFunction[];

/** The list of permitted fee payment contract instances. Takes precedence over contract classes */
allowedFeePaymentContractInstances?: AztecAddress[];
/** The list of functions calls allowed to run teardown */
allowedFunctionsInTeardown?: AllowedFunction[];
}
10 changes: 5 additions & 5 deletions yarn-project/circuit-types/src/mocks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -57,13 +57,13 @@ export const mockTx = (

data.forPublic.endNonRevertibleData.newNullifiers[0] = firstNullifier;

data.forPublic.endNonRevertibleData.publicCallStack = makeTuple(MAX_PUBLIC_CALL_STACK_LENGTH_PER_TX, i =>
i < numberOfNonRevertiblePublicCallRequests ? publicCallRequests[i].toCallRequest() : CallRequest.empty(),
data.forPublic.end.publicCallStack = makeTuple(MAX_PUBLIC_CALL_STACK_LENGTH_PER_TX, i =>
i < numberOfRevertiblePublicCallRequests ? publicCallRequests[i].toCallRequest() : CallRequest.empty(),
);

data.forPublic.end.publicCallStack = makeTuple(MAX_PUBLIC_CALL_STACK_LENGTH_PER_TX, i =>
i < numberOfRevertiblePublicCallRequests
? publicCallRequests[i + numberOfNonRevertiblePublicCallRequests].toCallRequest()
data.forPublic.endNonRevertibleData.publicCallStack = makeTuple(MAX_PUBLIC_CALL_STACK_LENGTH_PER_TX, i =>
i < numberOfNonRevertiblePublicCallRequests
? publicCallRequests[i + numberOfRevertiblePublicCallRequests].toCallRequest()
: CallRequest.empty(),
);
} else {
Expand Down
24 changes: 22 additions & 2 deletions yarn-project/end-to-end/src/benchmarks/bench_tx_size_fees.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,14 @@ import {
type AztecAddress,
type EthAddress,
type FeePaymentMethod,
FunctionSelector,
NativeFeePaymentMethod,
PrivateFeePaymentMethod,
PublicFeePaymentMethod,
TxStatus,
getContractClassFromArtifact,
} from '@aztec/aztec.js';
import { FPCContract, GasTokenContract, TokenContract } from '@aztec/noir-contracts.js';
import { FPCContract, GasTokenContract, SchnorrAccountContractArtifact, TokenContract } from '@aztec/noir-contracts.js';
import { getCanonicalGasTokenAddress } from '@aztec/protocol-contracts/gas-token';

import { jest } from '@jest/globals';
Expand Down Expand Up @@ -39,7 +40,26 @@ describe('benchmarks/tx_size_fees', () => {

await aztecNode.setConfig({
feeRecipient: sequencerAddress,
allowedFeePaymentContractClasses: [getContractClassFromArtifact(FPCContract.artifact).id],
allowedFunctionsInSetup: [
{
classId: getContractClassFromArtifact(SchnorrAccountContractArtifact).id,
selector: FunctionSelector.fromSignature('approve_public_authwit(Field'),
},
{
classId: getContractClassFromArtifact(FPCContract.artifact).id,
selector: FunctionSelector.fromSignature('prepare_fee((Field),Field,(Field),Field)'),
},
],
allowedFunctionsInTeardown: [
{
classId: getContractClassFromArtifact(FPCContract.artifact).id,
selector: FunctionSelector.fromSignature('pay_fee((Field),Field,(Field))'),
},
{
classId: getContractClassFromArtifact(GasTokenContract.artifact).id,
selector: FunctionSelector.fromSignature('pay_fee(Field)'),
},
],
});

await publicDeployAccounts(aliceWallet, wallets);
Expand Down
23 changes: 22 additions & 1 deletion yarn-project/end-to-end/src/e2e_dapp_subscription.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import {
type DeployL1Contracts,
type FeePaymentMethod,
Fr,
FunctionSelector,
type PXE,
PrivateFeePaymentMethod,
PublicFeePaymentMethod,
Expand All @@ -19,6 +20,7 @@ import {
CounterContract,
FPCContract,
GasTokenContract,
SchnorrAccountContractArtifact,
} from '@aztec/noir-contracts.js';
import { getCanonicalGasTokenAddress } from '@aztec/protocol-contracts/gas-token';

Expand Down Expand Up @@ -71,7 +73,26 @@ describe('e2e_dapp_subscription', () => {
await publicDeployAccounts(wallets[0], wallets);

await aztecNode.setConfig({
allowedFeePaymentContractClasses: [getContractClassFromArtifact(FPCContract.artifact).id],
allowedFunctionsInSetup: [
{
classId: getContractClassFromArtifact(SchnorrAccountContractArtifact).id,
selector: FunctionSelector.fromSignature('approve_public_authwit(Field'),
},
{
classId: getContractClassFromArtifact(FPCContract.artifact).id,
selector: FunctionSelector.fromSignature('prepare_fee((Field),Field,(Field),Field)'),
},
],
allowedFunctionsInTeardown: [
{
classId: getContractClassFromArtifact(FPCContract.artifact).id,
selector: FunctionSelector.fromSignature('pay_fee((Field),Field,(Field))'),
},
{
classId: getContractClassFromArtifact(GasTokenContract.artifact).id,
selector: FunctionSelector.fromSignature('pay_fee(Field)'),
},
],
});

// this should be a SignerlessWallet but that can't call public functions directly
Expand Down
30 changes: 29 additions & 1 deletion yarn-project/end-to-end/src/e2e_fees.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ import {
FPCContract,
GasTokenContract,
SchnorrAccountContract,
SchnorrAccountContractArtifact,
} from '@aztec/noir-contracts.js';

import { jest } from '@jest/globals';
Expand Down Expand Up @@ -58,7 +59,34 @@ describe('e2e_fees', () => {
wallets = _wallets;

await aztecNode.setConfig({
allowedFeePaymentContractClasses: [getContractClassFromArtifact(FPCContract.artifact).id],
allowedFunctionsInSetup: [
{
classId: getContractClassFromArtifact(SchnorrAccountContractArtifact).id,
selector: FunctionSelector.fromSignature('approve_public_authwit(Field)'),
},
{
classId: getContractClassFromArtifact(BananaCoin.artifact).id,
selector: FunctionSelector.fromSignature('_increase_public_balance((Field),Field)'),
},
{
classId: getContractClassFromArtifact(FPCContract.artifact).id,
selector: FunctionSelector.fromSignature('prepare_fee((Field),Field,(Field),Field)'),
},
],
allowedFunctionsInTeardown: [
{
classId: getContractClassFromArtifact(FPCContract.artifact).id,
selector: FunctionSelector.fromSignature('pay_fee((Field),Field,(Field))'),
},
{
classId: getContractClassFromArtifact(GasTokenContract.artifact).id,
selector: FunctionSelector.fromSignature('pay_fee(Field)'),
},
{
classId: getContractClassFromArtifact(FPCContract.artifact).id,
selector: FunctionSelector.fromSignature('pay_fee_with_shielded_rebate(Field,(Field),Field)'),
},
],
});

logFunctionSignatures(BananaCoin.artifact, logger);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import { getGlobalVariableBuilder } from '../global_variable_builder/index.js';
import { getL1Publisher } from '../publisher/index.js';
import { Sequencer, type SequencerConfig } from '../sequencer/index.js';
import { PublicProcessorFactory } from '../sequencer/public_processor.js';
import { TxValidatorFactory } from '../sequencer/tx_validator_factory.js';
import { TxValidatorFactory } from '../tx_validator/tx_validator_factory.js';

/**
* Encapsulates the full sequencer and publisher.
Expand Down
33 changes: 26 additions & 7 deletions yarn-project/sequencer-client/src/config.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { AztecAddress, Fr } from '@aztec/circuits.js';
import { type AllowedFunction } from '@aztec/circuit-types';
import { AztecAddress, Fr, FunctionSelector } from '@aztec/circuits.js';
import { type L1ContractAddresses, NULL_KEY } from '@aztec/ethereum';
import { EthAddress } from '@aztec/foundation/eth-address';

Expand Down Expand Up @@ -40,8 +41,8 @@ export function getConfigEnvVars(): SequencerClientConfig {
SEQ_TX_POLLING_INTERVAL_MS,
SEQ_MAX_TX_PER_BLOCK,
SEQ_MIN_TX_PER_BLOCK,
SEQ_FPC_CLASSES,
SEQ_FPC_INSTANCES,
SEQ_ALLOWED_SETUP_FN,
SEQ_ALLOWED_TEARDOWN_FN,
AVAILABILITY_ORACLE_CONTRACT_ADDRESS,
ROLLUP_CONTRACT_ADDRESS,
REGISTRY_CONTRACT_ADDRESS,
Expand Down Expand Up @@ -90,9 +91,27 @@ export function getConfigEnvVars(): SequencerClientConfig {
feeRecipient: FEE_RECIPIENT ? AztecAddress.fromString(FEE_RECIPIENT) : undefined,
acvmWorkingDirectory: ACVM_WORKING_DIRECTORY ? ACVM_WORKING_DIRECTORY : undefined,
acvmBinaryPath: ACVM_BINARY_PATH ? ACVM_BINARY_PATH : undefined,
allowedFeePaymentContractClasses: SEQ_FPC_CLASSES ? SEQ_FPC_CLASSES.split(',').map(Fr.fromString) : [],
allowedFeePaymentContractInstances: SEQ_FPC_INSTANCES
? SEQ_FPC_INSTANCES.split(',').map(AztecAddress.fromString)
: [],
allowedFunctionsInSetup: parseSequencerAllowList(SEQ_ALLOWED_SETUP_FN ?? ''),
allowedFunctionsInTeardown: parseSequencerAllowList(SEQ_ALLOWED_TEARDOWN_FN ?? ''),
};
}

function parseSequencerAllowList(value: string): AllowedFunction[] {
const entries: AllowedFunction[] = [];

if (!value) {
return entries;
}

for (const val of value.split(',')) {
const [identifierString, selectorString] = val.split(':');
const selector = FunctionSelector.fromString(selectorString);
const identifier = identifierString.startsWith('0x')
? AztecAddress.fromString(identifierString)
: Fr.fromString(identifierString);

entries.push({ address: identifier, selector });
}

return entries;
}
Original file line number Diff line number Diff line change
Expand Up @@ -127,11 +127,18 @@ export abstract class AbstractPhaseManager {
publicInputs: PrivateKernelTailCircuitPublicInputs,
enqueuedPublicFunctionCalls: PublicCallRequest[],
): Record<PublicKernelPhase, PublicCallRequest[]> {
const data = publicInputs.forPublic;
if (!data) {
return {
[PublicKernelPhase.SETUP]: [],
[PublicKernelPhase.APP_LOGIC]: [],
[PublicKernelPhase.TEARDOWN]: [],
[PublicKernelPhase.TAIL]: [],
};
}
const publicCallsStack = enqueuedPublicFunctionCalls.slice().reverse();
const nonRevertibleCallStack = publicInputs.forPublic!.endNonRevertibleData.publicCallStack.filter(
i => !i.isEmpty(),
);
const revertibleCallStack = publicInputs.forPublic!.end.publicCallStack.filter(i => !i.isEmpty());
const nonRevertibleCallStack = data.endNonRevertibleData.publicCallStack.filter(i => !i.isEmpty());
const revertibleCallStack = data.end.publicCallStack.filter(i => !i.isEmpty());

const callRequestsStack = publicCallsStack
.map(call => call.toCallRequest())
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -51,8 +51,8 @@ import { type MockProxy, mock } from 'jest-mock-extended';
import { type PublicKernelCircuitSimulator } from '../simulator/index.js';
import { type ContractsDataSourcePublicDB, type WorldStatePublicDB } from '../simulator/public_executor.js';
import { RealPublicKernelCircuitSimulator } from '../simulator/public_kernel.js';
import { type TxValidator } from '../tx_validator/tx_validator.js';
import { PublicProcessor } from './public_processor.js';
import { type TxValidator } from './tx_validator.js';

describe('public_processor', () => {
let db: MockProxy<MerkleTreeOperations>;
Expand Down Expand Up @@ -276,7 +276,7 @@ describe('public_processor', () => {
throw new Error(`Unexpected execution request: ${execution}`);
});

const txValidator: MockProxy<TxValidator> = mock<TxValidator>();
const txValidator: MockProxy<TxValidator<ProcessedTx>> = mock();
txValidator.validateTxs.mockRejectedValue([[], [tx]]);

const [processed, failed] = await processor.process([tx], 1, prover, txValidator);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,9 +21,9 @@ import { type MerkleTreeOperations } from '@aztec/world-state';
import { type PublicKernelCircuitSimulator } from '../simulator/index.js';
import { ContractsDataSourcePublicDB, WorldStateDB, WorldStatePublicDB } from '../simulator/public_executor.js';
import { RealPublicKernelCircuitSimulator } from '../simulator/public_kernel.js';
import { type TxValidator } from '../tx_validator/tx_validator.js';
import { type AbstractPhaseManager, PublicKernelPhase } from './abstract_phase_manager.js';
import { PhaseManagerFactory } from './phase_manager_factory.js';
import { type TxValidator } from './tx_validator.js';

/**
* Creates new instances of PublicProcessor given the provided merkle tree db and contract data source.
Expand Down Expand Up @@ -90,7 +90,7 @@ export class PublicProcessor {
txs: Tx[],
maxTransactions = txs.length,
blockProver?: BlockProver,
txValidator?: TxValidator,
txValidator?: TxValidator<ProcessedTx>,
): Promise<[ProcessedTx[], FailedTx[], ProcessReturnValues[]]> {
// The processor modifies the tx objects in place, so we need to clone them.
txs = txs.map(tx => Tx.clone(tx));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,9 +29,9 @@ import { type MockProxy, mock, mockFn } from 'jest-mock-extended';

import { type GlobalVariableBuilder } from '../global_variable_builder/global_builder.js';
import { type L1Publisher } from '../index.js';
import { TxValidatorFactory } from '../tx_validator/tx_validator_factory.js';
import { type PublicProcessor, type PublicProcessorFactory } from './public_processor.js';
import { Sequencer } from './sequencer.js';
import { TxValidatorFactory } from './tx_validator_factory.js';

describe('sequencer', () => {
let publisher: MockProxy<L1Publisher>;
Expand Down Expand Up @@ -109,9 +109,6 @@ describe('sequencer', () => {
l1ToL2MessageSource,
publicProcessorFactory,
new TxValidatorFactory(merkleTreeOps, contractSource, EthAddress.random()),
{
allowedFeePaymentContractClasses: [fpcClassId],
},
);
});

Expand Down
Loading

0 comments on commit 684b5d0

Please sign in to comment.