From 1387b10d34adb33779b8f9afb5e9765cf34a0189 Mon Sep 17 00:00:00 2001 From: Joe Caulfield Date: Tue, 5 Nov 2024 16:47:31 +0400 Subject: [PATCH] add firedancer conformance to ci --- .github/workflows/main.yml | 25 +++++++++++ .gitignore | 1 + Cargo.toml | 2 + scripts/ci/fd-conformance.mjs | 83 +++++++++++++++++++++++++++++++++++ scripts/utils.mjs | 21 +++++++++ 5 files changed, 132 insertions(+) create mode 100644 scripts/ci/fd-conformance.mjs diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 1046afa..791eb79 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -140,6 +140,31 @@ jobs: echo "CU usage has changed. Please run `cargo bench` and commit the new results."; exit 1; fi + + fd_conformance: + name: Builtin-BPF Conformance Test + runs-on: ubuntu-latest + needs: build_programs + steps: + - name: Git Checkout + uses: actions/checkout@v4 + + - name: Setup Environment + uses: ./.github/actions/setup + with: + cargo-cache-key: cargo-program-conformance + cargo-cache-fallback-key: cargo-programs + solana: true + + - name: Restore Program Builds + uses: actions/cache/restore@v4 + with: + path: ./**/*.so + key: ${{ runner.os }}-builds-${{ github.sha }} + + - name: Builtin-BPF Conformance Test + shell: bash + run: pnpm zx ./scripts/ci/fd-conformance.mjs generate_clients: name: Check Client Generation diff --git a/.gitignore b/.gitignore index 70487d2..74673ed 100644 --- a/.gitignore +++ b/.gitignore @@ -7,3 +7,4 @@ node_modules test-ledger dist +solana-conformance diff --git a/Cargo.toml b/Cargo.toml index cdc22df..85af759 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,8 @@ [workspace] resolver = "2" members = ["clients/rust", "program"] +# Required for CI +exclude = ["solana-conformance/impl/solfuzz-agave"] [workspace.metadata.cli] solana = "2.0.1" diff --git a/scripts/ci/fd-conformance.mjs b/scripts/ci/fd-conformance.mjs new file mode 100644 index 0000000..23a5ee7 --- /dev/null +++ b/scripts/ci/fd-conformance.mjs @@ -0,0 +1,83 @@ +#!/usr/bin/env zx + +// Firedancer conformance testing of the Core BPF Config program against its +// original builtin implementation. +// +// Note: This script can only be run on Ubuntu. + +import 'zx/globals'; +import { getProgramId, getProgramSharedObjectPath, workingDirectory } from '../utils.mjs'; + +// Clone the conformance harness. +const harnessPath = path.join(workingDirectory, 'solana-conformance'); +await $`git clone https://github.com/firedancer-io/solana-conformance.git`; + +// Clone the test vectors. +const testVectorsPath = path.join(harnessPath, 'impl', 'test-vectors'); +await $`git clone https://github.com/firedancer-io/test-vectors.git ${testVectorsPath}`; + +// Add the Mollusk-generated fixtures to the test inputs. +const firedancerFixturesPath = path.join(testVectorsPath, 'instr', 'fixtures', 'config'); +const molluskFixturesPath = path.join(workingDirectory, 'program', 'fuzz', 'blob'); +await $`cp -a ${molluskFixturesPath}/. ${firedancerFixturesPath}/`; + +// Clone the SolFuzz-Agave harness. +const solFuzzAgavePath = path.join(harnessPath, 'impl', 'solfuzz-agave'); +await $`git clone -b agave-v2.1.3 http://github.com/firedancer-io/solfuzz-agave.git ${solFuzzAgavePath}`; + +// Fetch protobuf files. +await $`make -j -C ${solFuzzAgavePath} fetch_proto` + +// Move into the conformance harness. +cd(harnessPath); + +// Build the environment. +await $`bash install_ubuntu_lite.sh`; + +const solFuzzAgaveManifestPath = path.join(solFuzzAgavePath, 'Cargo.toml'); +const solFuzzAgaveTargetPath = path.join( + solFuzzAgavePath, + 'target', + 'x86_64-unknown-linux-gnu', + 'release', + 'libsolfuzz_agave.so', +); + +const testTargetsDir = path.join(harnessPath, 'impl', 'lib'); +await $`mkdir -p ${testTargetsDir}`; + +// Build the Agave target with the builtin version. +const testTargetPathBuiltin = path.join(testTargetsDir, 'builtin.so'); +await $`cargo build --manifest-path ${solFuzzAgaveManifestPath} \ + --lib --release --target x86_64-unknown-linux-gnu`; +await $`mv ${solFuzzAgaveTargetPath} ${testTargetPathBuiltin}`; + +// Build the Agave target with the BPF version. +const testTargetPathCoreBpf = path.join(testTargetsDir, 'core_bpf.so'); +await $`CORE_BPF_PROGRAM_ID=${getProgramId('program')} \ + CORE_BPF_TARGET=${getProgramSharedObjectPath('program')} \ + FORCE_RECOMPILE=true \ + cargo build --manifest-path ${solFuzzAgaveManifestPath} \ + --lib --release --target x86_64-unknown-linux-gnu \ + --features core-bpf-conformance`; +await $`mv ${solFuzzAgaveTargetPath} ${testTargetPathCoreBpf}`; + +// Remove any test results if they exist. +await $`rm -rf test_results`; + +// Run the tests. +const fixturesPath = path.join(testVectorsPath, 'instr', 'fixtures', 'address-lookup-table'); +await $`source test_suite_env/bin/activate && \ + solana-test-suite run-tests \ + -i ${fixturesPath} -s ${testTargetPathBuiltin} -t ${testTargetPathCoreBpf} \ + --consensus-mode --save-failures`; + +// Assert conformance. +// There should be no fixtures in the `failed_protobufs` directory. +if (fs.existsSync('test_results/failed_protobufs')) { + if (fs.readdirSync('test_results/failed_protobufs').length > 0) { + throw new Error(`Error: mismatches detected.`); + } +} + +console.log('All tests passed.'); diff --git a/scripts/utils.mjs b/scripts/utils.mjs index 44ee564..77bba9a 100644 --- a/scripts/utils.mjs +++ b/scripts/utils.mjs @@ -70,6 +70,27 @@ export function getAllProgramFolders() { ); } +export function getProgramId(folder) { + return getCargoMetadata(folder)?.solana?.['program-id']; +} + +export function getProgramName(folder) { + return getCargo(folder).package?.name; +} + +export function getProgramSharedObjectName(folder) { + return `${getProgramName(folder).replace(/-/g, '_')}.so`; +} + +export function getProgramSharedObjectPath(folder) { + return path.join( + workingDirectory, + 'target', + 'deploy', + getProgramSharedObjectName(folder), + ); +} + export function getCargo(folder) { return parseToml( fs.readFileSync(