@lumenflow/cli 5.0.0 → 5.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/docs-generate-pack-reference.js +24 -5
- package/dist/docs-generate-pack-reference.js.map +1 -1
- package/dist/release.js +18 -0
- package/dist/release.js.map +1 -1
- package/dist/wu-prune.js +63 -5
- package/dist/wu-prune.js.map +1 -1
- package/package.json +11 -11
- package/packs/agent-runtime/manifest.ts +9 -3
- package/packs/agent-runtime/manifest.yaml +18 -0
- package/packs/agent-runtime/package.json +1 -1
- package/packs/sidekick/package.json +1 -1
- package/packs/software-delivery/manifest.ts +61 -10
- package/packs/software-delivery/manifest.yaml +168 -234
- package/packs/software-delivery/package.json +1 -1
- package/packs/software-delivery/src/config/delivery-review-contract.ts +249 -13
- package/packs/software-delivery/src/config/env-accessors.ts +59 -12
- package/packs/software-delivery/src/config/normalize-config-keys.ts +3 -13
- package/packs/agent-runtime/.turbo/turbo-build.log +0 -4
- package/packs/agent-runtime/.turbo/turbo-test.log +0 -25
- package/packs/agent-runtime/.turbo/turbo-typecheck.log +0 -4
- package/packs/sidekick/.lumenflow/state/conductor/outbox/sidekick-events.jsonl +0 -213
- package/packs/sidekick/.turbo/turbo-build.log +0 -4
- package/packs/sidekick/.turbo/turbo-test.log +0 -25
- package/packs/sidekick/.turbo/turbo-typecheck.log +0 -4
- package/packs/software-delivery/.turbo/turbo-build.log +0 -4
- package/packs/software-delivery/.turbo/turbo-test.log +0 -63
- package/packs/software-delivery/.turbo/turbo-typecheck.log +0 -4
|
@@ -1,20 +1,256 @@
|
|
|
1
1
|
// Copyright (c) 2026 Hellmai Ltd
|
|
2
2
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
3
3
|
|
|
4
|
+
import { z } from 'zod';
|
|
5
|
+
import { asRecord } from '@lumenflow/kernel/primitives/object-guards';
|
|
6
|
+
import { LUMENFLOW_CLIENT_IDS } from '../constants/client-ids.js';
|
|
7
|
+
import { WU_TYPES } from '../constants/wu-statuses.js';
|
|
8
|
+
|
|
4
9
|
/**
|
|
5
|
-
* Pack-
|
|
6
|
-
*
|
|
7
|
-
* Part of the staged migration pattern: consumers migrate OFF the
|
|
8
|
-
* `@lumenflow/core/delivery-review-contract` subpath and onto
|
|
9
|
-
* `@lumenflow/packs-software-delivery/config/delivery-review-contract` FIRST,
|
|
10
|
-
* while the canonical implementation still lives in `@lumenflow/core`. A
|
|
11
|
-
* follow-up WU will move the canonical file into the pack and update this
|
|
12
|
-
* wrapper to become the canonical definition.
|
|
10
|
+
* Pack-local delivery review contract surface.
|
|
13
11
|
*
|
|
14
|
-
*
|
|
15
|
-
*
|
|
16
|
-
* a shim chain — WU-2712 hit TS2305 on type-only imports through chained
|
|
17
|
-
* shims.
|
|
12
|
+
* Keeping this module in the pack avoids a package cycle with @lumenflow/core
|
|
13
|
+
* while preserving the same public API for pack consumers.
|
|
18
14
|
*/
|
|
19
15
|
|
|
20
|
-
export
|
|
16
|
+
export const DELIVERY_REVIEW_COMMAND_CONTEXTS = {
|
|
17
|
+
GATES: 'gates',
|
|
18
|
+
WU_PREP: 'wu:prep',
|
|
19
|
+
} as const;
|
|
20
|
+
|
|
21
|
+
export type DeliveryReviewCommandContext =
|
|
22
|
+
(typeof DELIVERY_REVIEW_COMMAND_CONTEXTS)[keyof typeof DELIVERY_REVIEW_COMMAND_CONTEXTS];
|
|
23
|
+
|
|
24
|
+
export const DEFAULT_DELIVERY_REVIEW_SKIP_TYPES = [
|
|
25
|
+
WU_TYPES.DOCUMENTATION,
|
|
26
|
+
WU_TYPES.PROCESS,
|
|
27
|
+
] as const;
|
|
28
|
+
|
|
29
|
+
export const DELIVERY_REVIEW_ARTIFACT_DIR = '.lumenflow/artifacts/delivery-review' as const;
|
|
30
|
+
|
|
31
|
+
export const DeliveryReviewGateConfigSchema = z
|
|
32
|
+
.object({
|
|
33
|
+
enabled: z.boolean().default(false),
|
|
34
|
+
skip_types: z.array(z.string().min(1)).default([...DEFAULT_DELIVERY_REVIEW_SKIP_TYPES]),
|
|
35
|
+
})
|
|
36
|
+
.strict();
|
|
37
|
+
|
|
38
|
+
export type DeliveryReviewGateConfig = z.infer<typeof DeliveryReviewGateConfigSchema>;
|
|
39
|
+
|
|
40
|
+
export const DeliveryReviewClientFeatureSchema = z
|
|
41
|
+
.object({
|
|
42
|
+
enabled: z.boolean().default(false),
|
|
43
|
+
auto_run: z.boolean().default(false),
|
|
44
|
+
})
|
|
45
|
+
.strict();
|
|
46
|
+
|
|
47
|
+
export type DeliveryReviewClientFeature = z.infer<typeof DeliveryReviewClientFeatureSchema>;
|
|
48
|
+
|
|
49
|
+
export const DeliveryReviewVerdictSchema = z.enum(['PASS', 'FAIL', 'PARTIAL']);
|
|
50
|
+
export type DeliveryReviewVerdict = z.infer<typeof DeliveryReviewVerdictSchema>;
|
|
51
|
+
|
|
52
|
+
export const DeliveryReviewFindingSeveritySchema = z.enum(['critical', 'high', 'medium', 'low']);
|
|
53
|
+
export type DeliveryReviewFindingSeverity = z.infer<typeof DeliveryReviewFindingSeveritySchema>;
|
|
54
|
+
|
|
55
|
+
export const DeliveryReviewFindingSchema = z
|
|
56
|
+
.object({
|
|
57
|
+
severity: DeliveryReviewFindingSeveritySchema,
|
|
58
|
+
title: z.string().min(1),
|
|
59
|
+
detail: z.string().min(1),
|
|
60
|
+
acceptanceCriteriaRefs: z.array(z.string().min(1)).optional(),
|
|
61
|
+
fileRefs: z.array(z.string().min(1)).optional(),
|
|
62
|
+
})
|
|
63
|
+
.strict();
|
|
64
|
+
|
|
65
|
+
export type DeliveryReviewFinding = z.infer<typeof DeliveryReviewFindingSchema>;
|
|
66
|
+
|
|
67
|
+
export const DeliveryReviewAcceptanceCriterionSchema = z
|
|
68
|
+
.object({
|
|
69
|
+
criterion: z.string().min(1),
|
|
70
|
+
status: z.enum(['satisfied', 'unclear', 'not_satisfied']),
|
|
71
|
+
evidence: z.array(z.string().min(1)).optional(),
|
|
72
|
+
})
|
|
73
|
+
.strict();
|
|
74
|
+
|
|
75
|
+
export type DeliveryReviewAcceptanceCriterion = z.infer<
|
|
76
|
+
typeof DeliveryReviewAcceptanceCriterionSchema
|
|
77
|
+
>;
|
|
78
|
+
|
|
79
|
+
export const DeliveryReviewResultSchema = z
|
|
80
|
+
.object({
|
|
81
|
+
wuId: z.string().min(1),
|
|
82
|
+
verdict: DeliveryReviewVerdictSchema,
|
|
83
|
+
summary: z.string().min(1),
|
|
84
|
+
findings: z.array(DeliveryReviewFindingSchema),
|
|
85
|
+
acceptanceCriteria: z.array(DeliveryReviewAcceptanceCriterionSchema),
|
|
86
|
+
metadata: z
|
|
87
|
+
.object({
|
|
88
|
+
runtimeClient: z.string().min(1).optional(),
|
|
89
|
+
startedAt: z.string().datetime(),
|
|
90
|
+
completedAt: z.string().datetime(),
|
|
91
|
+
})
|
|
92
|
+
.strict(),
|
|
93
|
+
})
|
|
94
|
+
.strict();
|
|
95
|
+
|
|
96
|
+
export type DeliveryReviewResult = z.infer<typeof DeliveryReviewResultSchema>;
|
|
97
|
+
|
|
98
|
+
export interface DeliveryReviewRuntimeConfigInput {
|
|
99
|
+
gates?: {
|
|
100
|
+
delivery_review?: unknown;
|
|
101
|
+
};
|
|
102
|
+
agents?: {
|
|
103
|
+
defaultClient?: unknown;
|
|
104
|
+
clients?: Record<string, unknown>;
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
export interface ResolvedDeliveryReviewRuntimeConfig {
|
|
109
|
+
clientName: string;
|
|
110
|
+
gate: DeliveryReviewGateConfig;
|
|
111
|
+
feature: DeliveryReviewClientFeature;
|
|
112
|
+
enabled: boolean;
|
|
113
|
+
autoRun: boolean;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
function resolveClientNameFromOptions(options: {
|
|
117
|
+
config: DeliveryReviewRuntimeConfigInput;
|
|
118
|
+
clientName?: string;
|
|
119
|
+
env?: Readonly<Record<string, string | undefined>>;
|
|
120
|
+
}): string {
|
|
121
|
+
const explicitClient = options.clientName?.trim();
|
|
122
|
+
if (explicitClient) {
|
|
123
|
+
return explicitClient;
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
const envClient = options.env?.LUMENFLOW_CLIENT?.trim();
|
|
127
|
+
if (envClient) {
|
|
128
|
+
return envClient;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
const defaultClient =
|
|
132
|
+
typeof options.config.agents?.defaultClient === 'string'
|
|
133
|
+
? options.config.agents.defaultClient.trim()
|
|
134
|
+
: '';
|
|
135
|
+
if (defaultClient) {
|
|
136
|
+
return defaultClient;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
return LUMENFLOW_CLIENT_IDS.CLAUDE_CODE;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
function resolveClientRecord(
|
|
143
|
+
clients: Record<string, unknown> | undefined,
|
|
144
|
+
clientName: string,
|
|
145
|
+
): Record<string, unknown> | null {
|
|
146
|
+
if (!clients) {
|
|
147
|
+
return null;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
const exactMatch = asRecord(clients[clientName]);
|
|
151
|
+
if (exactMatch) {
|
|
152
|
+
return exactMatch;
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
const matchKey = Object.keys(clients).find(
|
|
156
|
+
(key) => key.toLowerCase() === clientName.toLowerCase(),
|
|
157
|
+
);
|
|
158
|
+
return matchKey ? asRecord(clients[matchKey]) : null;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
export function resolveDeliveryReviewRuntimeState(options: {
|
|
162
|
+
config: DeliveryReviewRuntimeConfigInput;
|
|
163
|
+
clientName?: string;
|
|
164
|
+
env?: Readonly<Record<string, string | undefined>>;
|
|
165
|
+
}): ResolvedDeliveryReviewRuntimeConfig {
|
|
166
|
+
const clientName = resolveClientNameFromOptions(options);
|
|
167
|
+
const gate = DeliveryReviewGateConfigSchema.parse(options.config.gates?.delivery_review ?? {});
|
|
168
|
+
const clientConfig = resolveClientRecord(options.config.agents?.clients, clientName);
|
|
169
|
+
const features = asRecord(clientConfig?.features);
|
|
170
|
+
const feature = DeliveryReviewClientFeatureSchema.parse(features?.delivery_review ?? {});
|
|
171
|
+
|
|
172
|
+
return {
|
|
173
|
+
clientName,
|
|
174
|
+
gate,
|
|
175
|
+
feature,
|
|
176
|
+
enabled: gate.enabled && feature.enabled,
|
|
177
|
+
autoRun: gate.enabled && feature.enabled && feature.auto_run,
|
|
178
|
+
};
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
export function shouldRunDeliveryReviewForCommand(options: {
|
|
182
|
+
runtime: ResolvedDeliveryReviewRuntimeConfig;
|
|
183
|
+
commandContext: DeliveryReviewCommandContext;
|
|
184
|
+
}): boolean {
|
|
185
|
+
if (!options.runtime.enabled) {
|
|
186
|
+
return false;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
if (options.commandContext === DELIVERY_REVIEW_COMMAND_CONTEXTS.WU_PREP) {
|
|
190
|
+
return options.runtime.autoRun;
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
return true;
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
export function isDeliveryReviewSkippedForType(options: {
|
|
197
|
+
gate: DeliveryReviewGateConfig;
|
|
198
|
+
wuType?: unknown;
|
|
199
|
+
}): boolean {
|
|
200
|
+
if (typeof options.wuType !== 'string' || options.wuType.trim().length === 0) {
|
|
201
|
+
return false;
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
return options.gate.skip_types.includes(options.wuType);
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
export function flattenDeliveryReviewAcceptanceCriteria(acceptance: unknown): string[] {
|
|
208
|
+
if (Array.isArray(acceptance)) {
|
|
209
|
+
return acceptance
|
|
210
|
+
.filter((entry): entry is string => typeof entry === 'string')
|
|
211
|
+
.map((entry) => entry.trim())
|
|
212
|
+
.filter(Boolean);
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
const acceptanceRecord = asRecord(acceptance);
|
|
216
|
+
if (!acceptanceRecord) {
|
|
217
|
+
return [];
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
return Object.values(acceptanceRecord)
|
|
221
|
+
.flatMap((entry) =>
|
|
222
|
+
Array.isArray(entry)
|
|
223
|
+
? entry.filter((value): value is string => typeof value === 'string')
|
|
224
|
+
: [],
|
|
225
|
+
)
|
|
226
|
+
.map((entry) => entry.trim())
|
|
227
|
+
.filter(Boolean);
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
export function calculateDeliveryReviewVerdict(input: {
|
|
231
|
+
findings: DeliveryReviewFinding[];
|
|
232
|
+
acceptanceCriteria: DeliveryReviewAcceptanceCriterion[];
|
|
233
|
+
}): DeliveryReviewVerdict {
|
|
234
|
+
const hasBlockingFinding = input.findings.some(
|
|
235
|
+
(finding) => finding.severity === 'critical' || finding.severity === 'high',
|
|
236
|
+
);
|
|
237
|
+
if (hasBlockingFinding) {
|
|
238
|
+
return 'FAIL';
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
const hasUnsatisfiedCriterion = input.acceptanceCriteria.some(
|
|
242
|
+
(criterion) => criterion.status === 'not_satisfied',
|
|
243
|
+
);
|
|
244
|
+
if (hasUnsatisfiedCriterion) {
|
|
245
|
+
return 'FAIL';
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
const hasUnclearCriterion = input.acceptanceCriteria.some(
|
|
249
|
+
(criterion) => criterion.status === 'unclear',
|
|
250
|
+
);
|
|
251
|
+
if (hasUnclearCriterion || input.findings.length > 0) {
|
|
252
|
+
return 'PARTIAL';
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
return 'PASS';
|
|
256
|
+
}
|
|
@@ -1,19 +1,66 @@
|
|
|
1
1
|
// Copyright (c) 2026 Hellmai Ltd
|
|
2
2
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
3
3
|
|
|
4
|
+
import { createError, ErrorCodes } from '@lumenflow/kernel/primitives/error-handler';
|
|
5
|
+
|
|
4
6
|
/**
|
|
5
|
-
* Pack-
|
|
6
|
-
*
|
|
7
|
-
* Part of the staged migration pattern: consumers migrate OFF the
|
|
8
|
-
* `@lumenflow/core/env-accessors` subpath and onto
|
|
9
|
-
* `@lumenflow/packs-software-delivery/config/env-accessors` FIRST, while the
|
|
10
|
-
* canonical implementation still lives in `@lumenflow/core`. A follow-up WU
|
|
11
|
-
* will move the canonical file into the pack and update this wrapper to
|
|
12
|
-
* become the canonical definition.
|
|
7
|
+
* Pack-local environment access helpers.
|
|
13
8
|
*
|
|
14
|
-
*
|
|
15
|
-
*
|
|
16
|
-
* type-only imports through chained shims.
|
|
9
|
+
* These stay local to the pack so clean workspace builds do not need to route
|
|
10
|
+
* back through @lumenflow/core, which already depends on this pack.
|
|
17
11
|
*/
|
|
18
12
|
|
|
19
|
-
export
|
|
13
|
+
export interface GetEnvOptions {
|
|
14
|
+
env?: NodeJS.ProcessEnv;
|
|
15
|
+
trim?: boolean;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export interface RequireEnvOptions extends GetEnvOptions {
|
|
19
|
+
context?: string;
|
|
20
|
+
remediation?: string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const DEFAULT_REMEDIATION_SUFFIX = 'Set the environment variable and retry.';
|
|
24
|
+
const CONTEXT_SEPARATOR = ': ';
|
|
25
|
+
|
|
26
|
+
function getEnvSource(options: GetEnvOptions): NodeJS.ProcessEnv {
|
|
27
|
+
return options.env ?? process.env;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export function getEnv(envVar: string, options: GetEnvOptions = {}): string | undefined {
|
|
31
|
+
const source = getEnvSource(options);
|
|
32
|
+
const rawValue = source[envVar];
|
|
33
|
+
if (rawValue === undefined) {
|
|
34
|
+
return undefined;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
const trimValue = options.trim ?? true;
|
|
38
|
+
const normalizedValue = trimValue ? rawValue.trim() : rawValue;
|
|
39
|
+
if (normalizedValue.length === 0) {
|
|
40
|
+
return undefined;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
return normalizedValue;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
function formatContextPrefix(context?: string): string {
|
|
47
|
+
if (!context) {
|
|
48
|
+
return '';
|
|
49
|
+
}
|
|
50
|
+
return `${context}${CONTEXT_SEPARATOR}`;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export function requireEnv(envVar: string, options: RequireEnvOptions = {}): string {
|
|
54
|
+
const value = getEnv(envVar, options);
|
|
55
|
+
if (value !== undefined) {
|
|
56
|
+
return value;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
const remediation = options.remediation ?? DEFAULT_REMEDIATION_SUFFIX;
|
|
60
|
+
const contextPrefix = formatContextPrefix(options.context);
|
|
61
|
+
throw createError(
|
|
62
|
+
ErrorCodes.CONFIG_ERROR,
|
|
63
|
+
`${contextPrefix}Missing required environment variable ${envVar}. ${remediation}`,
|
|
64
|
+
{ envVar, context: options.context },
|
|
65
|
+
);
|
|
66
|
+
}
|
|
@@ -2,18 +2,8 @@
|
|
|
2
2
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
3
3
|
|
|
4
4
|
/**
|
|
5
|
-
*
|
|
6
|
-
*
|
|
7
|
-
* Part of the staged migration pattern: consumers migrate OFF the
|
|
8
|
-
* `@lumenflow/core/normalize-config-keys` subpath and onto
|
|
9
|
-
* `@lumenflow/packs-software-delivery/config/normalize-config-keys` FIRST,
|
|
10
|
-
* while the canonical implementation still lives in `@lumenflow/core`. A
|
|
11
|
-
* follow-up WU will move the canonical file into the pack and update this
|
|
12
|
-
* wrapper to become the canonical definition.
|
|
13
|
-
*
|
|
14
|
-
* This file must re-export DIRECTLY from
|
|
15
|
-
* `@lumenflow/core/normalize-config-keys` (single level). Do NOT introduce a
|
|
16
|
-
* shim chain — WU-2712 hit TS2305 on type-only imports through chained shims.
|
|
5
|
+
* This utility already lives in kernel primitives, so the pack can depend on
|
|
6
|
+
* it directly without routing through @lumenflow/core.
|
|
17
7
|
*/
|
|
18
8
|
|
|
19
|
-
export * from '@lumenflow/
|
|
9
|
+
export * from '@lumenflow/kernel/primitives/normalize-config-keys';
|
|
@@ -1,25 +0,0 @@
|
|
|
1
|
-
|
|
2
|
-
> @lumenflow/packs-agent-runtime@4.24.0 test /home/tom/source/hellmai/lumenflow-dev/worktrees/framework-agent-wu-2733/packages/@lumenflow/packs/agent-runtime
|
|
3
|
-
> vitest run
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
[1m[46m RUN [49m[22m [36mv4.0.18 [39m[90m/home/tom/source/hellmai/lumenflow-dev/worktrees/framework-agent-wu-2733/packages/@lumenflow/packs/agent-runtime[39m
|
|
7
|
-
|
|
8
|
-
[32m✓[39m __tests__/turn-lifecycle-emissions.test.ts [2m([22m[2m64 tests[22m[2m)[22m[32m 31[2mms[22m[39m
|
|
9
|
-
[32m✓[39m __tests__/capability-factory.test.ts [2m([22m[2m2 tests[22m[2m)[22m[32m 3[2mms[22m[39m
|
|
10
|
-
[32m✓[39m __tests__/session-schema.test.ts [2m([22m[2m2 tests[22m[2m)[22m[32m 13[2mms[22m[39m
|
|
11
|
-
[32m✓[39m __tests__/auto-session-integration.test.ts [2m([22m[2m3 tests[22m[2m)[22m[32m 18[2mms[22m[39m
|
|
12
|
-
[32m✓[39m __tests__/remote-controls.test.ts [2m([22m[2m22 tests[22m[2m)[22m[33m 303[2mms[22m[39m
|
|
13
|
-
[32m✓[39m __tests__/manifest.test.ts [2m([22m[2m3 tests[22m[2m)[22m[32m 17[2mms[22m[39m
|
|
14
|
-
[32m✓[39m __tests__/workflow-state.test.ts [2m([22m[2m1 test[22m[2m)[22m[32m 53[2mms[22m[39m
|
|
15
|
-
[32m✓[39m __tests__/policy-factory.test.ts [2m([22m[2m3 tests[22m[2m)[22m[32m 31[2mms[22m[39m
|
|
16
|
-
[32m✓[39m __tests__/orchestration.test.ts [2m([22m[2m4 tests[22m[2m)[22m[32m 58[2mms[22m[39m
|
|
17
|
-
[32m✓[39m __tests__/workflow-dag.test.ts [2m([22m[2m2 tests[22m[2m)[22m[32m 127[2mms[22m[39m
|
|
18
|
-
[32m✓[39m __tests__/remote-controls-mock.test.ts [2m([22m[2m53 tests[22m[2m)[22m[32m 82[2mms[22m[39m
|
|
19
|
-
[32m✓[39m __tests__/resume.test.ts [2m([22m[2m2 tests[22m[2m)[22m[32m 228[2mms[22m[39m
|
|
20
|
-
|
|
21
|
-
[2m Test Files [22m [1m[32m12 passed[39m[22m[90m (12)[39m
|
|
22
|
-
[2m Tests [22m [1m[32m161 passed[39m[22m[90m (161)[39m
|
|
23
|
-
[2m Start at [22m 00:22:52
|
|
24
|
-
[2m Duration [22m 2.72s[2m (transform 10.34s, setup 0ms, import 17.40s, tests 964ms, environment 3ms)[22m
|
|
25
|
-
|