@salesforce/plugin-agent 1.15.1-dev.3 → 1.15.1
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/README.md +26 -31
- package/lib/commands/agent/create.js +2 -2
- package/lib/commands/agent/create.js.map +1 -1
- package/lib/commands/agent/generate/test-spec.d.ts +0 -5
- package/lib/commands/agent/generate/test-spec.js +78 -196
- package/lib/commands/agent/generate/test-spec.js.map +1 -1
- package/lib/commands/agent/test/create.d.ts +2 -3
- package/lib/commands/agent/test/create.js +21 -80
- package/lib/commands/agent/test/create.js.map +1 -1
- package/lib/flags.d.ts +0 -1
- package/lib/flags.js +0 -31
- package/lib/flags.js.map +1 -1
- package/lib/handleTestResults.js +21 -2
- package/lib/handleTestResults.js.map +1 -1
- package/lib/read-dir.d.ts +1 -0
- package/lib/read-dir.js +16 -0
- package/lib/read-dir.js.map +1 -0
- package/messages/agent.generate.test-spec.md +0 -16
- package/messages/agent.test.create.md +8 -18
- package/npm-shrinkwrap.json +30 -522
- package/oclif.lock +11 -115
- package/oclif.manifest.json +10 -35
- package/package.json +5 -7
- package/lib/yes-no-cancel.d.ts +0 -10
- package/lib/yes-no-cancel.js +0 -60
- package/lib/yes-no-cancel.js.map +0 -1
package/README.md
CHANGED
|
@@ -132,7 +132,7 @@ EXAMPLES
|
|
|
132
132
|
$ sf agent create --agent-name "Resort Manager" --spec specs/resortManagerAgent.yaml --preview
|
|
133
133
|
```
|
|
134
134
|
|
|
135
|
-
_See code: [src/commands/agent/create.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
135
|
+
_See code: [src/commands/agent/create.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/create.ts)_
|
|
136
136
|
|
|
137
137
|
## `sf agent generate agent-spec`
|
|
138
138
|
|
|
@@ -236,7 +236,7 @@ EXAMPLES
|
|
|
236
236
|
$ sf agent generate agent-spec --tone formal --agent-user resortmanager@myorg.com
|
|
237
237
|
```
|
|
238
238
|
|
|
239
|
-
_See code: [src/commands/agent/generate/agent-spec.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
239
|
+
_See code: [src/commands/agent/generate/agent-spec.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/generate/agent-spec.ts)_
|
|
240
240
|
|
|
241
241
|
## `sf agent generate template`
|
|
242
242
|
|
|
@@ -270,7 +270,7 @@ EXAMPLES
|
|
|
270
270
|
force-app/main/default/bots/My_Awesome_Agent/My_Awesome_Agent.bot-meta.xml --agent-version 1
|
|
271
271
|
```
|
|
272
272
|
|
|
273
|
-
_See code: [src/commands/agent/generate/template.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
273
|
+
_See code: [src/commands/agent/generate/template.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/generate/template.ts)_
|
|
274
274
|
|
|
275
275
|
## `sf agent generate test-spec`
|
|
276
276
|
|
|
@@ -278,12 +278,7 @@ Interactively generate a specification file for a AI evaluation test.
|
|
|
278
278
|
|
|
279
279
|
```
|
|
280
280
|
USAGE
|
|
281
|
-
$ sf agent generate test-spec [--flags-dir <value>]
|
|
282
|
-
|
|
283
|
-
FLAGS
|
|
284
|
-
-d, --from-definition=<value> The API name of the AIEvaluationDefinition that you want to convert to a spec file.
|
|
285
|
-
-f, --output-file=<value> The name of the generated spec file. Defaults to "specs/<AGENT_API_NAME>-testSpec.yaml"
|
|
286
|
-
--force-overwrite Don't prompt for confirmation when overwriting an existing test spec file.
|
|
281
|
+
$ sf agent generate test-spec [--flags-dir <value>]
|
|
287
282
|
|
|
288
283
|
GLOBAL FLAGS
|
|
289
284
|
--flags-dir=<value> Import flag values from a directory.
|
|
@@ -298,7 +293,7 @@ EXAMPLES
|
|
|
298
293
|
$ sf agent generate test-spec
|
|
299
294
|
```
|
|
300
295
|
|
|
301
|
-
_See code: [src/commands/agent/generate/test-spec.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
296
|
+
_See code: [src/commands/agent/generate/test-spec.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/generate/test-spec.ts)_
|
|
302
297
|
|
|
303
298
|
## `sf agent preview`
|
|
304
299
|
|
|
@@ -333,43 +328,43 @@ FLAG DESCRIPTIONS
|
|
|
333
328
|
the API name of the agent? (TBD based on agents library)
|
|
334
329
|
```
|
|
335
330
|
|
|
336
|
-
_See code: [src/commands/agent/preview.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
331
|
+
_See code: [src/commands/agent/preview.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/preview.ts)_
|
|
337
332
|
|
|
338
333
|
## `sf agent test create`
|
|
339
334
|
|
|
340
|
-
|
|
335
|
+
Summary of a command.
|
|
341
336
|
|
|
342
337
|
```
|
|
343
338
|
USAGE
|
|
344
|
-
$ sf agent test create -o <value> [--json] [--flags-dir <value>] [--
|
|
345
|
-
[--api-version <value>] [--preview] [--force-overwrite]
|
|
339
|
+
$ sf agent test create -s <value> -o <value> [--json] [--flags-dir <value>] [--api-version <value>] [--preview] [-p]
|
|
346
340
|
|
|
347
341
|
FLAGS
|
|
348
|
-
-o, --target-org=<value>
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
--
|
|
353
|
-
--
|
|
354
|
-
--test-api-name=<value> The API name of the AiEvaluationDefinition.
|
|
342
|
+
-o, --target-org=<value> (required) Username or alias of the target org. Not required if the `target-org`
|
|
343
|
+
configuration variable is already set.
|
|
344
|
+
-p, --no-prompt Don't prompt for confirmation when overwriting an existing test.
|
|
345
|
+
-s, --spec=<value> (required) Description of a flag.
|
|
346
|
+
--api-version=<value> Override the api version used for api requests made by this command
|
|
347
|
+
--preview Preview the test metadata without deploying to your org.
|
|
355
348
|
|
|
356
349
|
GLOBAL FLAGS
|
|
357
350
|
--flags-dir=<value> Import flag values from a directory.
|
|
358
351
|
--json Format output as json.
|
|
359
352
|
|
|
360
353
|
DESCRIPTION
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
This command will convert a test spec file into an AiEvaluationDefinition and deploy it to your org. The spec file
|
|
364
|
-
must be in yaml format.
|
|
354
|
+
Summary of a command.
|
|
365
355
|
|
|
366
|
-
|
|
356
|
+
More information about a command. Don't repeat the summary.
|
|
367
357
|
|
|
368
358
|
EXAMPLES
|
|
369
359
|
$ sf agent test create
|
|
360
|
+
|
|
361
|
+
FLAG DESCRIPTIONS
|
|
362
|
+
-s, --spec=<value> Description of a flag.
|
|
363
|
+
|
|
364
|
+
More information about a flag. Don't repeat the summary.
|
|
370
365
|
```
|
|
371
366
|
|
|
372
|
-
_See code: [src/commands/agent/test/create.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
367
|
+
_See code: [src/commands/agent/test/create.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/test/create.ts)_
|
|
373
368
|
|
|
374
369
|
## `sf agent test list`
|
|
375
370
|
|
|
@@ -398,7 +393,7 @@ EXAMPLES
|
|
|
398
393
|
$ sf agent test list
|
|
399
394
|
```
|
|
400
395
|
|
|
401
|
-
_See code: [src/commands/agent/test/list.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
396
|
+
_See code: [src/commands/agent/test/list.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/test/list.ts)_
|
|
402
397
|
|
|
403
398
|
## `sf agent test results`
|
|
404
399
|
|
|
@@ -454,7 +449,7 @@ FLAG DESCRIPTIONS
|
|
|
454
449
|
test results aren't written.
|
|
455
450
|
```
|
|
456
451
|
|
|
457
|
-
_See code: [src/commands/agent/test/results.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
452
|
+
_See code: [src/commands/agent/test/results.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/test/results.ts)_
|
|
458
453
|
|
|
459
454
|
## `sf agent test resume`
|
|
460
455
|
|
|
@@ -517,7 +512,7 @@ FLAG DESCRIPTIONS
|
|
|
517
512
|
test results aren't written.
|
|
518
513
|
```
|
|
519
514
|
|
|
520
|
-
_See code: [src/commands/agent/test/resume.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
515
|
+
_See code: [src/commands/agent/test/resume.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/test/resume.ts)_
|
|
521
516
|
|
|
522
517
|
## `sf agent test run`
|
|
523
518
|
|
|
@@ -580,6 +575,6 @@ FLAG DESCRIPTIONS
|
|
|
580
575
|
test results aren't written.
|
|
581
576
|
```
|
|
582
577
|
|
|
583
|
-
_See code: [src/commands/agent/test/run.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1
|
|
578
|
+
_See code: [src/commands/agent/test/run.ts](https://github.com/salesforcecli/plugin-agent/blob/1.15.1/src/commands/agent/test/run.ts)_
|
|
584
579
|
|
|
585
580
|
<!-- commandsstop -->
|
|
@@ -13,7 +13,7 @@ import { MultiStageOutput } from '@oclif/multi-stage-output';
|
|
|
13
13
|
import { input as inquirerInput } from '@inquirer/prompts';
|
|
14
14
|
import { colorize } from '@oclif/core/ux';
|
|
15
15
|
import { Agent, AgentCreateLifecycleStagesV2, generateAgentApiName, } from '@salesforce/agents';
|
|
16
|
-
import { makeFlags, promptForFlag, getAgentUserId, validateAgentType, validateTone,
|
|
16
|
+
import { makeFlags, promptForFlag, getAgentUserId, validateAgentType, validateTone, } from '../../flags.js';
|
|
17
17
|
import { theme } from '../../inquirer-theme.js';
|
|
18
18
|
Messages.importMessagesDirectoryFromMetaUrl(import.meta.url);
|
|
19
19
|
const messages = Messages.loadMessages('@salesforce/plugin-agent', 'agent.create');
|
|
@@ -90,7 +90,7 @@ export default class AgentCreate extends SfCommand {
|
|
|
90
90
|
}
|
|
91
91
|
}
|
|
92
92
|
// If we don't have an agent spec yet, prompt.
|
|
93
|
-
const specPath = flags.spec ?? (await
|
|
93
|
+
const specPath = flags.spec ?? (await promptForFlag(FLAGGABLE_PROMPTS['spec']));
|
|
94
94
|
// Read the agent spec and validate
|
|
95
95
|
const inputSpec = YAML.parse(readFileSync(resolve(specPath), 'utf8'));
|
|
96
96
|
validateSpec(inputSpec);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"create.js","sourceRoot":"","sources":["../../../src/commands/agent/create.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AACH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAClE,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,6BAA6B,CAAC;AAC/D,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AACvD,OAAO,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC7D,OAAO,EAAE,KAAK,IAAI,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAC3D,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAC1C,OAAO,EACL,KAAK,EAGL,4BAA4B,EAE5B,oBAAoB,GACrB,MAAM,oBAAoB,CAAC;AAC5B,OAAO,EAEL,SAAS,EACT,aAAa,EACb,cAAc,EACd,iBAAiB,EACjB,YAAY,
|
|
1
|
+
{"version":3,"file":"create.js","sourceRoot":"","sources":["../../../src/commands/agent/create.ts"],"names":[],"mappings":"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"}
|
|
@@ -5,10 +5,5 @@ export default class AgentGenerateTestSpec extends SfCommand<void> {
|
|
|
5
5
|
static readonly examples: string[];
|
|
6
6
|
static readonly enableJsonFlag = false;
|
|
7
7
|
static readonly state = "beta";
|
|
8
|
-
static readonly flags: {
|
|
9
|
-
'from-definition': import("@oclif/core/interfaces").OptionFlag<string | undefined, import("@oclif/core/interfaces").CustomOptions>;
|
|
10
|
-
'force-overwrite': import("@oclif/core/interfaces").BooleanFlag<boolean>;
|
|
11
|
-
'output-file': import("@oclif/core/interfaces").OptionFlag<string | undefined, import("@oclif/core/interfaces").CustomOptions>;
|
|
12
|
-
};
|
|
13
8
|
run(): Promise<void>;
|
|
14
9
|
}
|
|
@@ -1,75 +1,45 @@
|
|
|
1
1
|
/*
|
|
2
|
-
* Copyright (c)
|
|
2
|
+
* Copyright (c) 2024, salesforce.com, inc.
|
|
3
3
|
* All rights reserved.
|
|
4
4
|
* Licensed under the BSD 3-Clause license.
|
|
5
5
|
* For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause
|
|
6
6
|
*/
|
|
7
|
+
import { join } from 'node:path';
|
|
7
8
|
import { readFile } from 'node:fs/promises';
|
|
8
|
-
import {
|
|
9
|
-
import {
|
|
10
|
-
import {
|
|
11
|
-
import { Messages, SfError, SfProject } from '@salesforce/core';
|
|
12
|
-
import { writeTestSpec, generateTestSpecFromAiEvalDefinition } from '@salesforce/agents';
|
|
9
|
+
import { SfCommand } from '@salesforce/sf-plugins-core';
|
|
10
|
+
import { Messages, SfError } from '@salesforce/core';
|
|
11
|
+
import { generateTestSpec } from '@salesforce/agents';
|
|
13
12
|
import { select, input, confirm, checkbox } from '@inquirer/prompts';
|
|
14
13
|
import { XMLParser } from 'fast-xml-parser';
|
|
15
|
-
import { ComponentSetBuilder } from '@salesforce/source-deploy-retrieve';
|
|
16
|
-
import { warn } from '@oclif/core/errors';
|
|
17
14
|
import { theme } from '../../../inquirer-theme.js';
|
|
18
|
-
import
|
|
15
|
+
import { readDir } from '../../../read-dir.js';
|
|
19
16
|
Messages.importMessagesDirectoryFromMetaUrl(import.meta.url);
|
|
20
17
|
const messages = Messages.loadMessages('@salesforce/plugin-agent', 'agent.generate.test-spec');
|
|
21
18
|
function castArray(value) {
|
|
22
|
-
if (!value)
|
|
23
|
-
return [];
|
|
24
19
|
return Array.isArray(value) ? value : [value];
|
|
25
20
|
}
|
|
26
|
-
|
|
27
|
-
* Prompts the user for test case information through interactive prompts.
|
|
28
|
-
*
|
|
29
|
-
* @param genAiPlugins - Record mapping topic names to GenAiPlugin XML file paths (used to find the related actions)
|
|
30
|
-
* @param genAiFunctions - Array of GenAiFunction names from the GenAiPlanner
|
|
31
|
-
* @returns Promise resolving to a TestCase object containing:
|
|
32
|
-
* - utterance: The user input string
|
|
33
|
-
* - expectedTopic: The expected topic for classification
|
|
34
|
-
* - expectedActions: Array of expected action names
|
|
35
|
-
* - expectedOutcome: Expected outcome string
|
|
36
|
-
*
|
|
37
|
-
* @remarks
|
|
38
|
-
* This function guides users through creating a test case by:
|
|
39
|
-
* 1. Prompting for an utterance
|
|
40
|
-
* 2. Selecting an expected topic (from GenAiPlugins specified in the Bot's GenAiPlanner)
|
|
41
|
-
* 3. Choosing expected actions (from GenAiFunctions in the GenAiPlanner or GenAiPlugin)
|
|
42
|
-
* 4. Defining an expected outcome
|
|
43
|
-
*/
|
|
44
|
-
async function promptForTestCase(genAiPlugins, genAiFunctions) {
|
|
21
|
+
async function promptForTestCase(genAiPlugins) {
|
|
45
22
|
const utterance = await input({
|
|
46
23
|
message: 'Utterance',
|
|
47
24
|
validate: (d) => d.length > 0 || 'utterance cannot be empty',
|
|
48
25
|
theme,
|
|
49
26
|
});
|
|
50
|
-
const
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
theme,
|
|
54
|
-
});
|
|
55
|
-
// GenAiFunctions (aka actions) can be defined in the GenAiPlugin or GenAiPlanner
|
|
56
|
-
// the actions from the planner are passed in as an argument to this function
|
|
57
|
-
// the actions from the plugin are read from the GenAiPlugin file
|
|
58
|
-
let actions = [];
|
|
59
|
-
if (genAiPlugins[expectedTopic]) {
|
|
60
|
-
const genAiPluginXml = await readFile(genAiPlugins[expectedTopic], 'utf-8');
|
|
61
|
-
const parser = new XMLParser();
|
|
62
|
-
const parsed = parser.parse(genAiPluginXml);
|
|
63
|
-
actions = castArray(parsed.GenAiPlugin.genAiFunctions ?? []).map((f) => f.functionName);
|
|
64
|
-
}
|
|
65
|
-
const expectedActions = await checkbox({
|
|
27
|
+
const customKey = '<OTHER>';
|
|
28
|
+
const topics = Object.keys(genAiPlugins);
|
|
29
|
+
const askForOtherActions = async () => (await input({
|
|
66
30
|
message: 'Expected action(s)',
|
|
67
|
-
|
|
31
|
+
validate: (d) => {
|
|
32
|
+
if (!d.length) {
|
|
33
|
+
return 'expected value cannot be empty';
|
|
34
|
+
}
|
|
35
|
+
return true;
|
|
36
|
+
},
|
|
68
37
|
theme,
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
38
|
+
}))
|
|
39
|
+
.split(',')
|
|
40
|
+
.map((a) => a.trim());
|
|
41
|
+
const askForBotRating = async () => input({
|
|
42
|
+
message: 'Expected response',
|
|
73
43
|
validate: (d) => {
|
|
74
44
|
if (!d.length) {
|
|
75
45
|
return 'expected value cannot be empty';
|
|
@@ -78,155 +48,62 @@ async function promptForTestCase(genAiPlugins, genAiFunctions) {
|
|
|
78
48
|
},
|
|
79
49
|
theme,
|
|
80
50
|
});
|
|
51
|
+
const expectedTopic = await select({
|
|
52
|
+
message: 'Expected topic',
|
|
53
|
+
choices: [...topics, customKey],
|
|
54
|
+
theme,
|
|
55
|
+
});
|
|
56
|
+
if (expectedTopic === customKey) {
|
|
57
|
+
return {
|
|
58
|
+
utterance,
|
|
59
|
+
expectedTopic: await input({
|
|
60
|
+
message: 'Expected topic',
|
|
61
|
+
validate: (d) => {
|
|
62
|
+
if (!d.length) {
|
|
63
|
+
return 'expected value cannot be empty';
|
|
64
|
+
}
|
|
65
|
+
return true;
|
|
66
|
+
},
|
|
67
|
+
theme,
|
|
68
|
+
}),
|
|
69
|
+
// If the user selects OTHER for the topic, then we don't have a genAiPlugin to get actions from so we ask for them for custom input
|
|
70
|
+
expectedActions: await askForOtherActions(),
|
|
71
|
+
expectedOutcome: await askForBotRating(),
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
const genAiPluginXml = await readFile(genAiPlugins[expectedTopic], 'utf-8');
|
|
75
|
+
const parser = new XMLParser();
|
|
76
|
+
const parsed = parser.parse(genAiPluginXml);
|
|
77
|
+
const actions = castArray(parsed.GenAiPlugin.genAiFunctions).map((f) => f.functionName);
|
|
78
|
+
let expectedActions = await checkbox({
|
|
79
|
+
message: 'Expected action(s)',
|
|
80
|
+
choices: [...actions, customKey],
|
|
81
|
+
theme,
|
|
82
|
+
required: true,
|
|
83
|
+
});
|
|
84
|
+
if (expectedActions.includes(customKey)) {
|
|
85
|
+
const additional = await askForOtherActions();
|
|
86
|
+
expectedActions = [...expectedActions.filter((a) => a !== customKey), ...additional];
|
|
87
|
+
}
|
|
88
|
+
const expectedOutcome = await askForBotRating();
|
|
81
89
|
return {
|
|
82
90
|
utterance,
|
|
83
|
-
expectedTopic,
|
|
84
91
|
expectedActions,
|
|
85
92
|
expectedOutcome,
|
|
93
|
+
expectedTopic,
|
|
86
94
|
};
|
|
87
95
|
}
|
|
88
|
-
function getMetadataFilePaths(cs, type) {
|
|
89
|
-
return [...cs.filter((component) => component.type.name === type && component.fullName !== '*')].reduce((acc, component) => ({
|
|
90
|
-
...acc,
|
|
91
|
-
[component.fullName]: cs.getComponentFilenamesByNameAndType({
|
|
92
|
-
fullName: component.fullName,
|
|
93
|
-
type,
|
|
94
|
-
})[0],
|
|
95
|
-
}), {});
|
|
96
|
-
}
|
|
97
|
-
/**
|
|
98
|
-
* Retrieves GenAIPlugins and GenAiFunctions from a Bot's GenAiPlanner
|
|
99
|
-
*
|
|
100
|
-
* We have to get the bot version and planner for the selected bot so that we can get
|
|
101
|
-
* the actions (GenAiFunctions) and topics (GenAiPlugins) that can be selected for the
|
|
102
|
-
* test cases.
|
|
103
|
-
*
|
|
104
|
-
* The BotVersion tells us which GenAiPlanner to use, and the GenAiPlanner
|
|
105
|
-
* tells us which GenAiPlugins and GenAiFunctions are available. More GenAiFunctions
|
|
106
|
-
* might be available in the GenAiPlugin, so we read those later when the user
|
|
107
|
-
* has selected a GenAiPlugin/topic - inside of `promptForTestCase`.
|
|
108
|
-
*
|
|
109
|
-
* @param subjectName - The name of the Bot to analyze
|
|
110
|
-
* @param cs - ComponentSet containing Bot, GenAiPlanner, and GenAiPlugin components
|
|
111
|
-
*
|
|
112
|
-
* @returns Object containing:
|
|
113
|
-
* - genAiPlugins: Record of plugin names to their file paths
|
|
114
|
-
* - genAiFunctions: Array of function names
|
|
115
|
-
*/
|
|
116
|
-
async function getPluginsAndFunctions(subjectName, cs) {
|
|
117
|
-
const botVersions = getMetadataFilePaths(cs, 'Bot');
|
|
118
|
-
const genAiPlanners = getMetadataFilePaths(cs, 'GenAiPlanner');
|
|
119
|
-
const parser = new XMLParser();
|
|
120
|
-
const botVersionXml = await readFile(botVersions[subjectName], 'utf-8');
|
|
121
|
-
const parsedBotVersion = parser.parse(botVersionXml);
|
|
122
|
-
const plannerXml = await readFile(genAiPlanners[parsedBotVersion.BotVersion.conversationDefinitionPlanners.genAiPlannerName ?? subjectName], 'utf-8');
|
|
123
|
-
const parsedPlanner = parser.parse(plannerXml);
|
|
124
|
-
const genAiFunctions = castArray(parsedPlanner.GenAiPlanner.genAiFunctions).map(({ genAiFunctionName }) => genAiFunctionName);
|
|
125
|
-
const genAiPlugins = castArray(parsedPlanner.GenAiPlanner.genAiPlugins).reduce((acc, { genAiPluginName }) => ({
|
|
126
|
-
...acc,
|
|
127
|
-
[genAiPluginName]: cs.getComponentFilenamesByNameAndType({
|
|
128
|
-
fullName: genAiPluginName,
|
|
129
|
-
type: 'GenAiPlugin',
|
|
130
|
-
})[0],
|
|
131
|
-
}), {});
|
|
132
|
-
return { genAiPlugins, genAiFunctions };
|
|
133
|
-
}
|
|
134
|
-
function ensureYamlExtension(filePath) {
|
|
135
|
-
const parsedPath = parse(filePath);
|
|
136
|
-
if (parsedPath.ext === '.yaml' || parsedPath.ext === '.yml')
|
|
137
|
-
return filePath;
|
|
138
|
-
const normalized = `${join(parsedPath.dir, parsedPath.name)}.yaml`;
|
|
139
|
-
warn(`Provided file path does not have a .yaml or .yml extension. Normalizing to ${normalized}`);
|
|
140
|
-
return normalized;
|
|
141
|
-
}
|
|
142
|
-
async function promptUntilUniqueFile(subjectName, filePath) {
|
|
143
|
-
const outputFile = filePath ??
|
|
144
|
-
(await input({
|
|
145
|
-
message: 'Enter a path for the test spec file',
|
|
146
|
-
validate(d) {
|
|
147
|
-
if (!d.length) {
|
|
148
|
-
return 'Path cannot be empty';
|
|
149
|
-
}
|
|
150
|
-
return true;
|
|
151
|
-
},
|
|
152
|
-
theme,
|
|
153
|
-
}));
|
|
154
|
-
const normalized = ensureYamlExtension(outputFile);
|
|
155
|
-
if (!existsSync(normalized)) {
|
|
156
|
-
return normalized;
|
|
157
|
-
}
|
|
158
|
-
const confirmation = await yesNoOrCancel({
|
|
159
|
-
message: `File ${normalized} already exists. Overwrite?`,
|
|
160
|
-
default: false,
|
|
161
|
-
});
|
|
162
|
-
if (confirmation === 'cancel') {
|
|
163
|
-
return;
|
|
164
|
-
}
|
|
165
|
-
if (!confirmation) {
|
|
166
|
-
return promptUntilUniqueFile(subjectName);
|
|
167
|
-
}
|
|
168
|
-
return normalized;
|
|
169
|
-
}
|
|
170
|
-
/**
|
|
171
|
-
* If the user provides the --force-overwrite flag, then we'll use the default file path (either the one provided by --output-file or the default path).
|
|
172
|
-
* If the user doesn't provide it, we'll prompt the user for a file path until they provide a unique one or cancel.
|
|
173
|
-
*/
|
|
174
|
-
async function determineFilePath(subjectName, outputFile, forceOverwrite) {
|
|
175
|
-
const defaultFile = ensureYamlExtension(outputFile ?? join('specs', `${subjectName}-testSpec.yaml`));
|
|
176
|
-
return forceOverwrite ? defaultFile : promptUntilUniqueFile(subjectName, defaultFile);
|
|
177
|
-
}
|
|
178
96
|
export default class AgentGenerateTestSpec extends SfCommand {
|
|
179
97
|
static summary = messages.getMessage('summary');
|
|
180
98
|
static description = messages.getMessage('description');
|
|
181
99
|
static examples = messages.getMessages('examples');
|
|
182
100
|
static enableJsonFlag = false;
|
|
183
101
|
static state = 'beta';
|
|
184
|
-
static flags = {
|
|
185
|
-
'from-definition': Flags.string({
|
|
186
|
-
char: 'd',
|
|
187
|
-
summary: messages.getMessage('flags.from-definition.summary'),
|
|
188
|
-
}),
|
|
189
|
-
'force-overwrite': Flags.boolean({
|
|
190
|
-
summary: messages.getMessage('flags.force-overwrite.summary'),
|
|
191
|
-
}),
|
|
192
|
-
'output-file': Flags.file({
|
|
193
|
-
char: 'f',
|
|
194
|
-
summary: messages.getMessage('flags.output-file.summary'),
|
|
195
|
-
parse: async (raw) => Promise.resolve(raw ? ensureYamlExtension(raw) : undefined),
|
|
196
|
-
}),
|
|
197
|
-
};
|
|
198
102
|
async run() {
|
|
199
|
-
const
|
|
200
|
-
const
|
|
201
|
-
const cs = await ComponentSetBuilder.build({
|
|
202
|
-
metadata: {
|
|
203
|
-
metadataEntries: ['GenAiPlanner', 'GenAiPlugin', 'Bot', 'AiEvaluationDefinition'],
|
|
204
|
-
directoryPaths,
|
|
205
|
-
},
|
|
206
|
-
});
|
|
207
|
-
if (flags['from-definition']) {
|
|
208
|
-
const aiEvalDefs = getMetadataFilePaths(cs, 'AiEvaluationDefinition');
|
|
209
|
-
if (!aiEvalDefs[flags['from-definition']]) {
|
|
210
|
-
throw new SfError(`AiEvaluationDefinition ${flags['from-definition']} not found`, 'AiEvalDefinitionNotFoundError');
|
|
211
|
-
}
|
|
212
|
-
const spec = await generateTestSpecFromAiEvalDefinition(aiEvalDefs[flags['from-definition']]);
|
|
213
|
-
const outputFile = await determineFilePath(spec.subjectName, flags['output-file'], flags['force-overwrite']);
|
|
214
|
-
if (!outputFile) {
|
|
215
|
-
this.log(messages.getMessage('info.cancel'));
|
|
216
|
-
return;
|
|
217
|
-
}
|
|
218
|
-
await writeTestSpec(spec, outputFile);
|
|
219
|
-
this.log(`Created ${outputFile}`);
|
|
220
|
-
return;
|
|
221
|
-
}
|
|
222
|
-
const bots = [
|
|
223
|
-
...cs
|
|
224
|
-
.filter((component) => component.type.name === 'Bot')
|
|
225
|
-
.map((c) => c.fullName)
|
|
226
|
-
.filter((n) => n !== '*'),
|
|
227
|
-
];
|
|
103
|
+
const botsDir = join('force-app', 'main', 'default', 'bots');
|
|
104
|
+
const bots = await readDir(botsDir);
|
|
228
105
|
if (bots.length === 0) {
|
|
229
|
-
throw new SfError(`No agents found in ${
|
|
106
|
+
throw new SfError(`No agents found in ${botsDir}`, 'NoAgentsFoundError');
|
|
230
107
|
}
|
|
231
108
|
const subjectType = await select({
|
|
232
109
|
message: 'What are you testing',
|
|
@@ -238,12 +115,6 @@ export default class AgentGenerateTestSpec extends SfCommand {
|
|
|
238
115
|
choices: bots,
|
|
239
116
|
theme,
|
|
240
117
|
});
|
|
241
|
-
const outputFile = await determineFilePath(subjectName, flags['output-file'], flags['force-overwrite']);
|
|
242
|
-
if (!outputFile) {
|
|
243
|
-
this.log(messages.getMessage('info.cancel'));
|
|
244
|
-
return;
|
|
245
|
-
}
|
|
246
|
-
const { genAiPlugins, genAiFunctions } = await getPluginsAndFunctions(subjectName, cs);
|
|
247
118
|
const name = await input({
|
|
248
119
|
message: 'Enter a name for the test definition',
|
|
249
120
|
validate(d) {
|
|
@@ -252,6 +123,12 @@ export default class AgentGenerateTestSpec extends SfCommand {
|
|
|
252
123
|
return 'Name cannot be empty';
|
|
253
124
|
}
|
|
254
125
|
return true;
|
|
126
|
+
// TODO: add back validation once we refine the regex
|
|
127
|
+
// check against FORTY_CHAR_API_NAME_REGEX
|
|
128
|
+
// if (!FORTY_CHAR_API_NAME_REGEX.test(d)) {
|
|
129
|
+
// return 'The non-namespaced portion an API name must begin with a letter, contain only letters, numbers, and underscores, not contain consecutive underscores, and not end with an underscore.';
|
|
130
|
+
// }
|
|
131
|
+
// return true;
|
|
255
132
|
},
|
|
256
133
|
theme,
|
|
257
134
|
});
|
|
@@ -259,26 +136,31 @@ export default class AgentGenerateTestSpec extends SfCommand {
|
|
|
259
136
|
message: 'Enter a description for test definition (optional)',
|
|
260
137
|
theme,
|
|
261
138
|
});
|
|
139
|
+
const genAiPluginDir = join('force-app', 'main', 'default', 'genAiPlugins');
|
|
140
|
+
const genAiPlugins = Object.fromEntries((await readDir(genAiPluginDir)).map((genAiPlugin) => [
|
|
141
|
+
genAiPlugin.replace('.genAiPlugin-meta.xml', ''),
|
|
142
|
+
join(genAiPluginDir, genAiPlugin),
|
|
143
|
+
]));
|
|
262
144
|
const testCases = [];
|
|
263
145
|
do {
|
|
264
146
|
this.log();
|
|
265
147
|
this.styledHeader(`Adding test case #${testCases.length + 1}`);
|
|
266
148
|
// eslint-disable-next-line no-await-in-loop
|
|
267
|
-
testCases.push(await promptForTestCase(genAiPlugins
|
|
149
|
+
testCases.push(await promptForTestCase(genAiPlugins));
|
|
268
150
|
} while ( // eslint-disable-next-line no-await-in-loop
|
|
269
151
|
await confirm({
|
|
270
152
|
message: 'Would you like to add another test case',
|
|
271
153
|
default: true,
|
|
272
154
|
}));
|
|
273
155
|
this.log();
|
|
274
|
-
await
|
|
156
|
+
await generateTestSpec({
|
|
275
157
|
name,
|
|
276
158
|
description,
|
|
277
159
|
subjectType,
|
|
278
160
|
subjectName,
|
|
279
161
|
testCases,
|
|
280
|
-
},
|
|
281
|
-
this.log(`Created ${
|
|
162
|
+
}, `${name}-test-spec.yaml`);
|
|
163
|
+
this.log(`Created ${name}-test-spec.yaml`);
|
|
282
164
|
}
|
|
283
165
|
}
|
|
284
166
|
//# sourceMappingURL=test-spec.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"test-spec.js","sourceRoot":"","sources":["../../../../src/commands/agent/generate/test-spec.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AACH,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"test-spec.js","sourceRoot":"","sources":["../../../../src/commands/agent/generate/test-spec.ts"],"names":[],"mappings":"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"}
|
|
@@ -11,12 +11,11 @@ export default class AgentTestCreate extends SfCommand<AgentTestCreateResult> {
|
|
|
11
11
|
static readonly examples: string[];
|
|
12
12
|
static readonly state = "beta";
|
|
13
13
|
static readonly flags: {
|
|
14
|
+
spec: import("@oclif/core/interfaces").OptionFlag<string, import("@oclif/core/interfaces").CustomOptions>;
|
|
14
15
|
'target-org': import("@oclif/core/interfaces").OptionFlag<import("@salesforce/core").Org, import("@oclif/core/interfaces").CustomOptions>;
|
|
15
16
|
'api-version': import("@oclif/core/interfaces").OptionFlag<string | undefined, import("@oclif/core/interfaces").CustomOptions>;
|
|
16
17
|
preview: import("@oclif/core/interfaces").BooleanFlag<boolean>;
|
|
17
|
-
'
|
|
18
|
-
spec: import("@oclif/core/interfaces").OptionFlag<string | undefined, import("@oclif/core/interfaces").CustomOptions>;
|
|
19
|
-
"test-api-name": import("@oclif/core/interfaces").OptionFlag<string | undefined, import("@oclif/core/interfaces").CustomOptions>;
|
|
18
|
+
'no-prompt': import("@oclif/core/interfaces").BooleanFlag<boolean>;
|
|
20
19
|
};
|
|
21
20
|
private mso?;
|
|
22
21
|
run(): Promise<AgentTestCreateResult>;
|