Skip to content

Commit

Permalink
Stop run if /start fails (#138)
Browse files Browse the repository at this point in the history
  • Loading branch information
Nicole White authored Apr 18, 2024
1 parent 669654a commit 47d9b1c
Show file tree
Hide file tree
Showing 2 changed files with 96 additions and 30 deletions.
77 changes: 51 additions & 26 deletions src/testing/run.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,10 @@ const evaluatorSemaphoreRegistry: Record<
> = {}; // testId -> evaluatorId -> Semaphore

const client = {
post: async (path: string, data: unknown): Promise<void> => {
post: async (args: {
path: string;
body: unknown;
}): Promise<Pick<Response, 'ok'>> => {
const serverAddress = readEnv(
AutoblocksEnvVar.AUTOBLOCKS_CLI_SERVER_ADDRESS,
);
Expand All @@ -28,15 +31,18 @@ $ npx autoblocks testing exec -- <your test command>
}

try {
await fetch(serverAddress + path, {
return await fetch(serverAddress + args.path, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify(data),
body: JSON.stringify(args.body),
});
} catch (err) {
} catch {
// Ignore, any errors for these requests are displayed by the CLI server
return {
ok: false,
};
}
},
};
Expand All @@ -60,14 +66,17 @@ async function sendError(args: {
errorStack: '',
};

await client.post('/errors', {
testExternalId: args.testId,
testCaseHash: args.testCaseHash,
evaluatorExternalId: args.evaluatorId,
error: {
name: errorName,
message: errorMessage,
stacktrace: errorStack,
await client.post({
path: '/errors',
body: {
testExternalId: args.testId,
testCaseHash: args.testCaseHash,
evaluatorExternalId: args.evaluatorId,
error: {
name: errorName,
message: errorMessage,
stacktrace: errorStack,
},
},
});
}
Expand Down Expand Up @@ -97,13 +106,16 @@ async function runEvaluatorUnsafe<TestCaseType, OutputType>(args: {
});
});

await client.post('/evals', {
testExternalId: args.testId,
testCaseHash: args.testCaseHash,
evaluatorExternalId: args.evaluator.id,
score: evaluation.score,
threshold: evaluation.threshold,
metadata: evaluation.metadata,
await client.post({
path: '/evals',
body: {
testExternalId: args.testId,
testCaseHash: args.testCaseHash,
evaluatorExternalId: args.evaluator.id,
score: evaluation.score,
threshold: evaluation.threshold,
metadata: evaluation.metadata,
},
});
}

Expand Down Expand Up @@ -164,11 +176,14 @@ async function runTestCaseUnsafe<TestCaseType, OutputType>(args: {
// with the result.
await flush();

await client.post('/results', {
testExternalId: args.testId,
testCaseHash: args.testCaseHash,
testCaseBody: args.testCase,
testCaseOutput: isPrimitive(output) ? output : JSON.stringify(output),
await client.post({
path: '/results',
body: {
testExternalId: args.testId,
testCaseHash: args.testCaseHash,
testCaseBody: args.testCase,
testCaseOutput: isPrimitive(output) ? output : JSON.stringify(output),
},
});

return output;
Expand Down Expand Up @@ -289,7 +304,17 @@ export async function runTestSuite<
]),
);

await client.post('/start', { testExternalId: args.id });
const startResp = await client.post({
path: '/start',
body: { testExternalId: args.id },
});
if (!startResp.ok) {
// Don't allow the run to continue if /start failed, since all subsequent
// requests will fail if the CLI was not able to start the run.
// Also note we don't need to sendError here, since the CLI will
// have reported the HTTP error itself.
return;
}

try {
await Promise.allSettled(
Expand All @@ -312,5 +337,5 @@ export async function runTestSuite<
});
}

await client.post('/end', { testExternalId: args.id });
await client.post({ path: '/end', body: { testExternalId: args.id } });
}
49 changes: 45 additions & 4 deletions test/testing/run.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,10 +24,10 @@ describe('Testing SDK', () => {
let mockFetch: jest.SpyInstance;

beforeEach(() => {
mockFetch = jest
.spyOn(global, 'fetch')
// eslint-disable-next-line @typescript-eslint/no-explicit-any
.mockResolvedValue({ json: () => Promise.resolve() } as any);
mockFetch = jest.spyOn(global, 'fetch').mockResolvedValue({
json: () => Promise.resolve(),
ok: true,
} as Response);
});

// eslint-disable-next-line @typescript-eslint/no-explicit-any
Expand Down Expand Up @@ -1012,3 +1012,44 @@ describe('Testing SDK', () => {
});
});
});

/**
* This is a separate describe() because the main one mocks
* all fetch requests to be successful. This one will test
* unsuccessful fetch requests within each test.
*/
describe('Testing SDK with HTTP Errors', () => {
beforeAll(() => {
process.env.AUTOBLOCKS_CLI_SERVER_ADDRESS = MOCK_CLI_SERVER_ADDRESS;
});

afterAll(() => {
delete process.env.AUTOBLOCKS_CLI_SERVER_ADDRESS;
});

it('stops if /start fails', async () => {
const mockFetch = jest
.spyOn(global, 'fetch')
.mockResolvedValueOnce({ ok: false } as Response);

await runTestSuite<MyTestCase, string>({
id: 'my-test-id',
testCases: [{ x: 1, y: 2 }],
testCaseHash: ['x', 'y'],
evaluators: [],
fn: ({ testCase }: { testCase: MyTestCase }) =>
`${testCase.x} + ${testCase.y} = ${testCase.x + testCase.y}`,
});

expect(mockFetch).toHaveBeenCalledTimes(1);
expect(mockFetch).toHaveBeenCalledWith(`${MOCK_CLI_SERVER_ADDRESS}/start`, {
method: 'POST',
body: JSON.stringify({
testExternalId: 'my-test-id',
}),
headers: {
'Content-Type': 'application/json',
},
});
});
});

0 comments on commit 47d9b1c

Please sign in to comment.