diff --git a/.changeset/kind-students-kneel.md b/.changeset/kind-students-kneel.md new file mode 100644 index 00000000000..dfbaf4f9f50 --- /dev/null +++ b/.changeset/kind-students-kneel.md @@ -0,0 +1,6 @@ +--- +"@fuel-ts/account": patch +"fuels": patch +--- + +fix: `fuels dev` cleanup not killing node diff --git a/.github/actions/test-setup/action.yaml b/.github/actions/test-setup/action.yaml index 8d7c3d6cf0c..744b2a7ad1b 100644 --- a/.github/actions/test-setup/action.yaml +++ b/.github/actions/test-setup/action.yaml @@ -33,6 +33,12 @@ runs: version: ${{ inputs.pnpm-version }} run_install: true + - name: Setup forc and fuel-core paths + shell: bash + run: | + echo "$PWD/internal/forc/forc-binaries" >> $GITHUB_PATH + echo "$PWD/internal/fuel-core/fuel-core-binaries" >> $GITHUB_PATH + - name: Setup Bun if: ${{ inputs.should-install-bun == 'true' }} uses: oven-sh/setup-bun@v1 diff --git a/packages/account/src/test-utils/launchNode.ts b/packages/account/src/test-utils/launchNode.ts index 2e24f9ce4ee..b9a8ce38252 100644 --- a/packages/account/src/test-utils/launchNode.ts +++ b/packages/account/src/test-utils/launchNode.ts @@ -261,7 +261,6 @@ export const launchNode = async ({ } childState.isDead = true; - removeSideffects(); if (child.pid !== undefined) { try { process.kill(-child.pid); diff --git a/packages/fuels/src/cli.ts b/packages/fuels/src/cli.ts index 0299c304750..9a716cbfb7d 100644 --- a/packages/fuels/src/cli.ts +++ b/packages/fuels/src/cli.ts @@ -65,6 +65,10 @@ export const configureCli = () => { .option('--forc-path ', 'Path to the `forc` binary') .option('--fuel-core-path ', 'Path to the `fuel-core` binary') .option('--auto-start-fuel-core', 'Auto-starts a `fuel-core` node during `dev` command') + .option( + '--fuel-core-port ', + 'Port to use when starting a local `fuel-core` node for dev mode' + ) .action(withProgram(command, Commands.init, init)); (command = program.command(Commands.dev)) diff --git a/packages/fuels/src/cli/commands/dev/autoStartFuelCore.ts b/packages/fuels/src/cli/commands/dev/autoStartFuelCore.ts index 3853682c678..66d653f64b2 100644 --- a/packages/fuels/src/cli/commands/dev/autoStartFuelCore.ts +++ b/packages/fuels/src/cli/commands/dev/autoStartFuelCore.ts @@ -25,9 +25,7 @@ export const autoStartFuelCore = async (config: FuelsConfig) => { const port = config.fuelCorePort ?? (await getPortPromise({ port: 4000 })); - const providerUrl = `http://${accessIp}:${port}/v1/graphql`; - - const { cleanup, snapshotDir } = await launchNode({ + const { cleanup, snapshotDir, url } = await launchNode({ args: [ ['--snapshot', config.snapshotDir], ['--db-type', 'in-memory'], @@ -44,7 +42,7 @@ export const autoStartFuelCore = async (config: FuelsConfig) => { bindIp, accessIp, port, - providerUrl, + providerUrl: url, snapshotDir, killChildProcess: cleanup, }; diff --git a/packages/fuels/src/cli/commands/init/index.ts b/packages/fuels/src/cli/commands/init/index.ts index 0f537c5431a..cf15d11af49 100644 --- a/packages/fuels/src/cli/commands/init/index.ts +++ b/packages/fuels/src/cli/commands/init/index.ts @@ -10,7 +10,7 @@ import { log } from '../../utils/logger'; export function init(program: Command) { const options = program.opts(); - const { path, autoStartFuelCore, forcPath, fuelCorePath } = options; + const { path, autoStartFuelCore, forcPath, fuelCorePath, fuelCorePort } = options; let workspace: string | undefined; let absoluteWorkspace: string | undefined; @@ -61,6 +61,7 @@ export function init(program: Command) { forcPath, fuelCorePath, autoStartFuelCore, + fuelCorePort, }); writeFileSync(fuelsConfigPath, renderedConfig); diff --git a/packages/fuels/src/cli/templates/fuels.config.hbs b/packages/fuels/src/cli/templates/fuels.config.hbs index dc351121c38..33ceaa33b18 100644 --- a/packages/fuels/src/cli/templates/fuels.config.hbs +++ b/packages/fuels/src/cli/templates/fuels.config.hbs @@ -36,6 +36,9 @@ export default createConfig({ {{#if (isDefined autoStartFuelCore)}} autoStartFuelCore: {{autoStartFuelCore}}, {{/if}} + {{#if (isDefined fuelCorePort)}} + fuelCorePort: {{fuelCorePort}}, + {{/if}} }); /** diff --git a/packages/fuels/src/cli/templates/fuels.config.ts b/packages/fuels/src/cli/templates/fuels.config.ts index 71930e975a9..cda74e22e1a 100644 --- a/packages/fuels/src/cli/templates/fuels.config.ts +++ b/packages/fuels/src/cli/templates/fuels.config.ts @@ -16,6 +16,7 @@ export function renderFuelsConfigTemplate(props: { forcPath?: string; fuelCorePath?: string; autoStartFuelCore?: boolean; + fuelCorePort?: string; }) { const renderTemplate = Handlebars.compile(fuelsConfigTemplate, { strict: true, diff --git a/packages/fuels/test/features/dev-2.test.ts b/packages/fuels/test/features/dev-2.test.ts new file mode 100644 index 00000000000..01924679182 --- /dev/null +++ b/packages/fuels/test/features/dev-2.test.ts @@ -0,0 +1,81 @@ +import { execSync, execFileSync, spawn } from 'child_process'; +import { mkdirSync, rmSync } from 'fs'; +import { tmpdir } from 'os'; +import path from 'path'; + +import { deferPromise, randomUUID } from '../../src'; +import { findChildProcessPid, waitProcessEnd } from '../utils/processUtils'; + +function runInit() { + const fuelsPath = path.join(process.cwd(), 'packages/fuels'); + + const init = path.join(tmpdir(), '.fuels', 'tests', randomUUID()); + + mkdirSync(init, { recursive: true }); + + execFileSync('pnpm', ['init'], { cwd: init }); + execFileSync('pnpm', ['link', fuelsPath], { cwd: init }); + + const contractDir = path.join(init, 'contract'); + const outputDir = path.join(init, 'output'); + mkdirSync(contractDir); + mkdirSync(outputDir); + + execSync(`${process.env.FORC_PATH} init`, { cwd: contractDir }); + execSync(`pnpm fuels init -o ${outputDir} -c ${contractDir} --fuel-core-port 0`, { cwd: init }); + + return { + init, + [Symbol.dispose]: () => { + rmSync(init, { recursive: true }); + }, + }; +} + +/** + * @group node + */ +describe('dev', () => { + it( + 'cleans up resources on graceful shutdown', + async () => { + using paths = runInit(); + + const devProcess = spawn('pnpm fuels dev', { + shell: 'bash', + detached: true, + cwd: paths.init, + }); + + const devCompleted = deferPromise(); + + devProcess.stdout.on('data', (chunk) => { + const text = chunk.toString(); + if (text.indexOf('Dev completed successfully!') !== -1) { + devCompleted.resolve(undefined); + } + }); + + await devCompleted.promise; + + const devExited = deferPromise(); + devProcess.on('exit', () => { + devExited.resolve(undefined); + }); + + const devPid = devProcess.pid as number; + + const fuelCorePid = findChildProcessPid(devPid, 'fuel-core') as number; + + // we kill the pnpm fuels dev process group + // and we want to verify that the fuel-core process is also killed + process.kill(-devPid, 'SIGINT'); + + await devExited.promise; + + // if it finishes before timeout, it means the process was killed successfully + await waitProcessEnd(fuelCorePid); + }, + { timeout: 15000 } + ); +}); diff --git a/packages/fuels/test/utils/processUtils.ts b/packages/fuels/test/utils/processUtils.ts new file mode 100644 index 00000000000..1ebc049a1bb --- /dev/null +++ b/packages/fuels/test/utils/processUtils.ts @@ -0,0 +1,52 @@ +import { sleep } from '@fuel-ts/utils'; +import { execSync } from 'child_process'; + +export function findChildProcessPid( + parentPid: number, + childProcessName: string +): number | undefined { + const childProcesses = execSync(`ps --ppid ${parentPid} -o pid,cmd --no-headers || true`) + .toString() + .split('\n') + .map((s) => s.trim()) + .filter((s) => s !== ''); + + for (const cp of childProcesses) { + const [pid, name] = cp.split(' '); + if (name.indexOf(childProcessName) !== -1) { + return +pid; + } + const childPid = findChildProcessPid(+pid, childProcessName); + if (childPid) { + return childPid; + } + } + + return undefined; +} + +function isProcessRunning(pid: number) { + try { + // Check if the process exists + process.kill(pid, 0); + return true; // If no error, the process is running + } catch (e) { + const error = e as Error & { code: string }; + // Error codes: + // ESRCH: No such process + // EPERM: Permission denied (you don't have permissions to check) + if (error.code === 'ESRCH') { + return false; // No such process + } + if (error.code === 'EPERM') { + return true; // Process exists, but we don't have permission to send a signal + } + throw error; // Some other unexpected error + } +} + +export async function waitProcessEnd(pid: number) { + while (isProcessRunning(pid)) { + await sleep(100); + } +} diff --git a/vitest.global-setup.ts b/vitest.global-setup.ts index 48593227974..76715c2a703 100644 --- a/vitest.global-setup.ts +++ b/vitest.global-setup.ts @@ -1,3 +1,4 @@ export default function setup() { process.env.FUEL_CORE_PATH = 'fuels-core'; + process.env.FORC_PATH = 'fuels-forc'; }