Skip to content

Commit ad8b245

Browse files
committed
Support ignoring test/spec files inside trigger dirs (fixes #1593)
1 parent 51a054f commit ad8b245

File tree

16 files changed

+845
-64
lines changed

16 files changed

+845
-64
lines changed

docs/config/config-file.mdx

+27
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,33 @@ The config file handles a lot of things, like:
4747
imports used inside build config with be tree-shaken out.
4848
</Note>
4949

50+
## Dirs
51+
52+
You can specify the directories where your tasks are located using the `dirs` option:
53+
54+
```ts trigger.config.ts
55+
import { defineConfig } from "@trigger.dev/sdk/v3";
56+
57+
export default defineConfig({
58+
project: "<project ref>",
59+
dirs: ["./trigger"],
60+
});
61+
```
62+
63+
If you omit the `dirs` option, we will automatically detect directories that are named `trigger` in your project, but we recommend specifying the directories explicitly. The `dirs` option is an array of strings, so you can specify multiple directories if you have tasks in multiple locations.
64+
65+
We will search for TypeScript and JavaScript files in the specified directories and include them in the build process. We automatically exclude files that have `.test` or `.spec` in the name, but you can customize this by specifying glob patterns in the `ignorePatterns` option:
66+
67+
```ts trigger.config.ts
68+
import { defineConfig } from "@trigger.dev/sdk/v3";
69+
70+
export default defineConfig({
71+
project: "<project ref>",
72+
dirs: ["./trigger"],
73+
ignorePatterns: ["**/*.my-test.ts"],
74+
});
75+
```
76+
5077
## Lifecycle functions
5178

5279
You can add lifecycle functions to get notified when any task starts, succeeds, or fails using `onStart`, `onSuccess` and `onFailure`:

docs/mint.json

+50-14
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,10 @@
11
{
22
"$schema": "https://mintlify.com/schema.json",
33
"name": "Trigger.dev",
4-
"openapi": ["/openapi.yml", "/v3-openapi.yaml"],
4+
"openapi": [
5+
"/openapi.yml",
6+
"/v3-openapi.yaml"
7+
],
58
"api": {
69
"playground": {
710
"mode": "simple"
@@ -128,7 +131,6 @@
128131
"quick-start",
129132
"video-walkthrough",
130133
"how-it-works",
131-
"upgrading-beta",
132134
"limits"
133135
]
134136
},
@@ -137,20 +139,30 @@
137139
"pages": [
138140
{
139141
"group": "Tasks",
140-
"pages": ["tasks/overview", "tasks/schemaTask", "tasks/scheduled"]
142+
"pages": [
143+
"tasks/overview",
144+
"tasks/schemaTask",
145+
"tasks/scheduled"
146+
]
141147
},
142148
"triggering",
143149
"runs",
144150
"apikeys",
145151
{
146152
"group": "Configuration",
147-
"pages": ["config/config-file", "config/extensions/overview"]
153+
"pages": [
154+
"config/config-file",
155+
"config/extensions/overview"
156+
]
148157
}
149158
]
150159
},
151160
{
152161
"group": "Development",
153-
"pages": ["cli-dev", "run-tests"]
162+
"pages": [
163+
"cli-dev",
164+
"run-tests"
165+
]
154166
},
155167
{
156168
"group": "Deployment",
@@ -160,7 +172,9 @@
160172
"github-actions",
161173
{
162174
"group": "Deployment integrations",
163-
"pages": ["vercel-integration"]
175+
"pages": [
176+
"vercel-integration"
177+
]
164178
}
165179
]
166180
},
@@ -172,7 +186,13 @@
172186
"errors-retrying",
173187
{
174188
"group": "Wait",
175-
"pages": ["wait", "wait-for", "wait-until", "wait-for-event", "wait-for-request"]
189+
"pages": [
190+
"wait",
191+
"wait-for",
192+
"wait-until",
193+
"wait-for-event",
194+
"wait-for-request"
195+
]
176196
},
177197
"queue-concurrency",
178198
"versioning",
@@ -217,7 +237,10 @@
217237
"management/overview",
218238
{
219239
"group": "Tasks API",
220-
"pages": ["management/tasks/trigger", "management/tasks/batch-trigger"]
240+
"pages": [
241+
"management/tasks/trigger",
242+
"management/tasks/batch-trigger"
243+
]
221244
},
222245
{
223246
"group": "Runs API",
@@ -256,7 +279,9 @@
256279
},
257280
{
258281
"group": "Projects API",
259-
"pages": ["management/projects/runs"]
282+
"pages": [
283+
"management/projects/runs"
284+
]
260285
}
261286
]
262287
},
@@ -294,6 +319,7 @@
294319
"pages": [
295320
"troubleshooting",
296321
"upgrading-packages",
322+
"upgrading-beta",
297323
"troubleshooting-alerts",
298324
"troubleshooting-uptime-status",
299325
"troubleshooting-github-issues",
@@ -302,11 +328,17 @@
302328
},
303329
{
304330
"group": "Help",
305-
"pages": ["community", "help-slack", "help-email"]
331+
"pages": [
332+
"community",
333+
"help-slack",
334+
"help-email"
335+
]
306336
},
307337
{
308338
"group": "",
309-
"pages": ["guides/introduction"]
339+
"pages": [
340+
"guides/introduction"
341+
]
310342
},
311343
{
312344
"group": "Frameworks",
@@ -380,16 +412,20 @@
380412
},
381413
{
382414
"group": "Dashboard",
383-
"pages": ["guides/dashboard/creating-a-project"]
415+
"pages": [
416+
"guides/dashboard/creating-a-project"
417+
]
384418
},
385419
{
386420
"group": "Migrations",
387-
"pages": ["guides/use-cases/upgrading-from-v2"]
421+
"pages": [
422+
"guides/use-cases/upgrading-from-v2"
423+
]
388424
}
389425
],
390426
"footerSocials": {
391427
"twitter": "https://twitter.com/triggerdotdev",
392428
"github": "https://github.com/triggerdotdev",
393429
"linkedin": "https://www.linkedin.com/company/triggerdotdev"
394430
}
395-
}
431+
}

packages/cli-v3/package.json

+2
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,7 @@
9191
"@trigger.dev/core": "workspace:3.3.9",
9292
"c12": "^1.11.1",
9393
"chalk": "^5.2.0",
94+
"chokidar": "^3.6.0",
9495
"cli-table3": "^0.6.3",
9596
"commander": "^9.4.1",
9697
"defu": "^6.1.4",
@@ -119,6 +120,7 @@
119120
"terminal-link": "^3.0.0",
120121
"tiny-invariant": "^1.2.0",
121122
"tinyexec": "^0.3.1",
123+
"tinyglobby": "^0.2.2",
122124
"ws": "^8.18.0",
123125
"xdg-app-paths": "^8.3.0",
124126
"zod": "3.23.8",

packages/cli-v3/src/build/bundle.ts

+83-45
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ import {
1717
telemetryEntryPoint,
1818
} from "./packageModules.js";
1919
import { buildPlugins } from "./plugins.js";
20+
import { createEntryPointManager } from "./entryPoints.js";
2021

2122
export interface BundleOptions {
2223
target: BuildTarget;
@@ -45,12 +46,30 @@ export type BundleResult = {
4546
export async function bundleWorker(options: BundleOptions): Promise<BundleResult> {
4647
const { resolvedConfig } = options;
4748

48-
// We need to add the package entry points here somehow
49-
// Then we need to get them out of the build result into the build manifest
50-
// taskhero/dist/esm/workers/dev.js
51-
// taskhero/dist/esm/telemetry/loader.js
52-
const entryPoints = await getEntryPoints(options.target, resolvedConfig);
53-
const $buildPlugins = await buildPlugins(options.target, resolvedConfig);
49+
let currentContext: esbuild.BuildContext | undefined;
50+
51+
const entryPointManager = await createEntryPointManager(
52+
resolvedConfig.dirs,
53+
resolvedConfig,
54+
options.target,
55+
typeof options.watch === "boolean" ? options.watch : false,
56+
async (newEntryPoints) => {
57+
if (currentContext) {
58+
// Rebuild with new entry points
59+
await currentContext.cancel();
60+
await currentContext.dispose();
61+
const buildOptions = await createBuildOptions({
62+
...options,
63+
entryPoints: newEntryPoints,
64+
});
65+
66+
logger.debug("Rebuilding worker with options", buildOptions);
67+
68+
currentContext = await esbuild.context(buildOptions);
69+
await currentContext.watch();
70+
}
71+
}
72+
);
5473

5574
let initialBuildResult: (result: esbuild.BuildResult) => void;
5675
const initialBuildResultPromise = new Promise<esbuild.BuildResult>(
@@ -63,12 +82,63 @@ export async function bundleWorker(options: BundleOptions): Promise<BundleResult
6382
},
6483
};
6584

85+
const buildOptions = await createBuildOptions({
86+
...options,
87+
entryPoints: entryPointManager.entryPoints,
88+
buildResultPlugin,
89+
});
90+
91+
let result: esbuild.BuildResult<typeof buildOptions>;
92+
let stop: BundleResult["stop"];
93+
94+
logger.debug("Building worker with options", buildOptions);
95+
96+
if (options.watch) {
97+
currentContext = await esbuild.context(buildOptions);
98+
await currentContext.watch();
99+
result = await initialBuildResultPromise;
100+
if (result.errors.length > 0) {
101+
throw new Error("Failed to build");
102+
}
103+
104+
stop = async function () {
105+
await entryPointManager.stop();
106+
await currentContext?.dispose();
107+
};
108+
} else {
109+
result = await esbuild.build(buildOptions);
110+
111+
stop = async function () {
112+
await entryPointManager.stop();
113+
};
114+
}
115+
116+
const bundleResult = await getBundleResultFromBuild(
117+
options.target,
118+
options.cwd,
119+
options.resolvedConfig,
120+
result
121+
);
122+
123+
if (!bundleResult) {
124+
throw new Error("Failed to get bundle result");
125+
}
126+
127+
return { ...bundleResult, stop };
128+
}
129+
130+
// Helper function to create build options
131+
async function createBuildOptions(
132+
options: BundleOptions & { entryPoints: string[]; buildResultPlugin?: esbuild.Plugin }
133+
): Promise<esbuild.BuildOptions & { metafile: true }> {
66134
const customConditions = options.resolvedConfig.build?.conditions ?? [];
67135

68136
const conditions = [...customConditions, "trigger.dev", "module", "node"];
69137

70-
const buildOptions: esbuild.BuildOptions & { metafile: true } = {
71-
entryPoints,
138+
const $buildPlugins = await buildPlugins(options.target, options.resolvedConfig);
139+
140+
return {
141+
entryPoints: options.entryPoints,
72142
outdir: options.destination,
73143
absWorkingDir: options.cwd,
74144
bundle: true,
@@ -93,50 +163,18 @@ export async function bundleWorker(options: BundleOptions): Promise<BundleResult
93163
inject: [...shims], // TODO: copy this into the working dir to work with Yarn PnP
94164
jsx: options.jsxAutomatic ? "automatic" : undefined,
95165
jsxDev: options.jsxAutomatic && options.target === "dev" ? true : undefined,
96-
plugins: [...$buildPlugins, ...(options.plugins ?? []), buildResultPlugin],
166+
plugins: [
167+
...$buildPlugins,
168+
...(options.plugins ?? []),
169+
...(options.buildResultPlugin ? [options.buildResultPlugin] : []),
170+
],
97171
...(options.jsxFactory && { jsxFactory: options.jsxFactory }),
98172
...(options.jsxFragment && { jsxFragment: options.jsxFragment }),
99173
logLevel: "silent",
100174
logOverride: {
101175
"empty-glob": "silent",
102176
},
103177
};
104-
105-
let result: esbuild.BuildResult<typeof buildOptions>;
106-
let stop: BundleResult["stop"];
107-
108-
logger.debug("Building worker with options", buildOptions);
109-
110-
if (options.watch) {
111-
const ctx = await esbuild.context(buildOptions);
112-
await ctx.watch();
113-
result = await initialBuildResultPromise;
114-
if (result.errors.length > 0) {
115-
throw new Error("Failed to build");
116-
}
117-
118-
stop = async function () {
119-
await ctx.dispose();
120-
};
121-
} else {
122-
result = await esbuild.build(buildOptions);
123-
// Even when we're not watching, we still want some way of cleaning up the
124-
// temporary directory when we don't need it anymore
125-
stop = async function () {};
126-
}
127-
128-
const bundleResult = await getBundleResultFromBuild(
129-
options.target,
130-
options.cwd,
131-
options.resolvedConfig,
132-
result
133-
);
134-
135-
if (!bundleResult) {
136-
throw new Error("Failed to get bundle result");
137-
}
138-
139-
return { ...bundleResult, stop };
140178
}
141179

142180
export async function getBundleResultFromBuild(

0 commit comments

Comments
 (0)