@pnp/cli-microsoft365 9.0.0-beta.2f8dd1e → 9.0.0-beta.62575a1

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.
Files changed (47) hide show
  1. package/.eslintrc.cjs +1 -0
  2. package/allCommands.json +1 -1
  3. package/allCommandsFull.json +1 -1
  4. package/dist/Auth.js +1 -9
  5. package/dist/chili/chili.js +0 -23
  6. package/dist/cli/cli.js +1 -63
  7. package/dist/m365/commands/setup.js +0 -4
  8. package/dist/m365/entra/commands/m365group/m365group-set.js +66 -29
  9. package/dist/m365/entra/commands/multitenant/MultitenantOrganization.js +2 -0
  10. package/dist/m365/entra/commands/multitenant/multitenant-get.js +32 -0
  11. package/dist/m365/entra/commands.js +1 -0
  12. package/dist/m365/external/commands/connection/connection-doctor.js +10 -24
  13. package/dist/m365/flow/commands/flow-list.js +23 -24
  14. package/dist/m365/graph/commands/subscription/subscription-add.js +4 -2
  15. package/dist/m365/spfx/commands/project/base-project-command.js +36 -126
  16. package/dist/m365/spo/commands/cdn/cdn-get.js +12 -15
  17. package/dist/m365/spo/commands/cdn/cdn-set.js +6 -4
  18. package/dist/m365/spo/commands/contenttype/contenttype-field-list.js +124 -0
  19. package/dist/m365/spo/commands/field/field-list.js +1 -1
  20. package/dist/m365/spo/commands/group/group-member-add.js +103 -99
  21. package/dist/m365/spo/commands/page/page-clientsidewebpart-add.js +2 -3
  22. package/dist/m365/spo/commands/page/page-text-add.js +2 -3
  23. package/dist/m365/spo/commands/spo-search.js +3 -4
  24. package/dist/m365/spo/commands.js +1 -0
  25. package/dist/m365/teams/commands/meeting/meeting-attendancereport-get.js +119 -0
  26. package/dist/m365/teams/commands/message/message-remove.js +112 -0
  27. package/dist/m365/teams/commands.js +2 -0
  28. package/dist/m365/viva/commands/engage/engage-community-add.js +166 -0
  29. package/dist/m365/viva/commands.js +1 -0
  30. package/dist/utils/formatting.js +14 -1
  31. package/dist/utils/teams.js +49 -0
  32. package/dist/utils/validation.js +19 -0
  33. package/docs/docs/cmd/entra/m365group/m365group-set.mdx +37 -7
  34. package/docs/docs/cmd/entra/multitenant/multitenant-get.mdx +94 -0
  35. package/docs/docs/cmd/external/connection/connection-doctor.mdx +9 -9
  36. package/docs/docs/cmd/flow/flow-list.mdx +114 -56
  37. package/docs/docs/cmd/graph/subscription/subscription-add.mdx +18 -0
  38. package/docs/docs/cmd/spo/cdn/cdn-set.mdx +3 -3
  39. package/docs/docs/cmd/spo/contenttype/contenttype-field-list.mdx +172 -0
  40. package/docs/docs/cmd/spo/contenttype/contenttype-list.mdx +3 -3
  41. package/docs/docs/cmd/spo/field/field-list.mdx +3 -3
  42. package/docs/docs/cmd/spo/group/group-member-add.mdx +34 -27
  43. package/docs/docs/cmd/teams/meeting/meeting-attendancereport-get.mdx +138 -0
  44. package/docs/docs/cmd/teams/message/message-remove.mdx +63 -0
  45. package/docs/docs/cmd/viva/engage/engage-community-add.mdx +168 -0
  46. package/npm-shrinkwrap.json +0 -195
  47. package/package.json +2 -2
package/dist/Auth.js CHANGED
@@ -354,15 +354,7 @@ export class Auth {
354
354
  await logger.logToStderr(response);
355
355
  await logger.logToStderr('');
356
356
  }
357
- cli.spinner.text = response.message;
358
- cli.spinner.spinner = {
359
- frames: ['🌶️ ']
360
- };
361
- // don't show spinner if running tests
362
- /* c8 ignore next 3 */
363
- if (!cli.spinner.isSpinning && typeof global.it === 'undefined') {
364
- cli.spinner.start();
365
- }
357
+ await logger.logToStderr(`🌶️ ${response.message}`);
366
358
  if (cli.getSettingWithDefaultValue(settingsNames.autoOpenLinksInBrowser, false)) {
367
359
  await browserUtil.open(response.verificationUri);
368
360
  }
@@ -1,22 +1,17 @@
1
1
  import fs from 'fs';
2
- import ora from 'ora';
3
2
  import path from 'path';
4
3
  import url from 'url';
5
- import { cli } from '../cli/cli.js';
6
4
  import request from '../request.js';
7
- import { settingsNames } from '../settingsNames.js';
8
5
  import { md } from '../utils/md.js';
9
6
  import { prompt } from '../utils/prompt.js';
10
7
  const __dirname = url.fileURLToPath(new URL('.', import.meta.url));
11
8
  const mendableBaseUrl = 'https://api.mendable.ai/v1';
12
9
  const mendableApiKey = 'd3313d54-6f8e-40e0-90d3-4095019d4be7';
13
- const spinner = ora({ discardStdin: false });
14
10
  let showHelp = false;
15
11
  let debug = false;
16
12
  let conversationId = 0;
17
13
  let initialPrompt = '';
18
14
  let history = [];
19
- const showSpinner = cli.getSettingWithDefaultValue(settingsNames.showSpinner, true) && typeof global.it === 'undefined';
20
15
  request.logger = {
21
16
  /* c8 ignore next 3 */
22
17
  log: async (msg) => console.log(msg),
@@ -74,20 +69,11 @@ async function promptForPrompt() {
74
69
  }
75
70
  async function runConversationTurn(conversationId, question) {
76
71
  console.log('');
77
- /* c8 ignore next 4 */
78
- if (showSpinner) {
79
- spinner.text = 'Searching documentation...';
80
- spinner.start();
81
- }
82
72
  const response = await runMendableChat(conversationId, question);
83
73
  history.push({
84
74
  prompt: question,
85
75
  response: response.answer.text
86
76
  });
87
- /* c8 ignore next 3 */
88
- if (showSpinner) {
89
- spinner.stop();
90
- }
91
77
  console.log(md.md2plain(response.answer.text, ''));
92
78
  console.log('');
93
79
  console.log('Source:');
@@ -141,16 +127,7 @@ async function endConversation(conversationId) {
141
127
  conversation_id: conversationId
142
128
  }
143
129
  };
144
- /* c8 ignore next 4 */
145
- if (showSpinner) {
146
- spinner.text = 'Ending conversation...';
147
- spinner.start();
148
- }
149
130
  await request.post(requestOptions);
150
- /* c8 ignore next 3 */
151
- if (showSpinner) {
152
- spinner.stop();
153
- }
154
131
  }
155
132
  async function runMendableChat(conversationId, question) {
156
133
  const requestOptions = {
package/dist/cli/cli.js CHANGED
@@ -2,7 +2,6 @@ import Configstore from 'configstore';
2
2
  import fs from 'fs';
3
3
  import minimist from 'minimist';
4
4
  import { createRequire } from 'module';
5
- import ora from 'ora';
6
5
  import os from 'os';
7
6
  import path from 'path';
8
7
  import { fileURLToPath, pathToFileURL } from 'url';
@@ -21,9 +20,6 @@ import { browserUtil } from '../utils/browserUtil.js';
21
20
  const require = createRequire(import.meta.url);
22
21
  const __dirname = fileURLToPath(new URL('.', import.meta.url));
23
22
  let _config;
24
- // we assign it through exported function to support mocking
25
- // eslint-disable-next-line prefer-const
26
- let spinner = ora();
27
23
  const commands = [];
28
24
  /**
29
25
  * Command to execute
@@ -190,12 +186,6 @@ async function executeCommand(command, args) {
190
186
  // the command to execute
191
187
  const parentCommandName = cli.currentCommandName;
192
188
  cli.currentCommandName = command.getCommandName(cli.currentCommandName);
193
- const showSpinner = cli.getSettingWithDefaultValue(settingsNames.showSpinner, true) && args.options.output !== 'none';
194
- // don't show spinner if running tests
195
- /* c8 ignore next 3 */
196
- if (showSpinner && typeof global.it === 'undefined') {
197
- cli.spinner.start();
198
- }
199
189
  const startCommand = process.hrtime.bigint();
200
190
  try {
201
191
  await command.action(logger, args);
@@ -207,10 +197,6 @@ async function executeCommand(command, args) {
207
197
  finally {
208
198
  // restore the original command name
209
199
  cli.currentCommandName = parentCommandName;
210
- /* c8 ignore next 3 */
211
- if (cli.spinner.isSpinning) {
212
- cli.spinner.stop();
213
- }
214
200
  const endCommand = process.hrtime.bigint();
215
201
  timings.command.push(Number(endCommand - startCommand));
216
202
  }
@@ -758,29 +744,14 @@ async function closeWithError(error, args, showHelpIfEnabled = false) {
758
744
  /* c8 ignore next */
759
745
  }
760
746
  function log(message, ...optionalParams) {
761
- const spinnerSpinning = cli.spinner.isSpinning;
762
- /* c8 ignore next 3 */
763
- if (spinnerSpinning) {
764
- cli.spinner.stop();
765
- }
766
747
  if (message) {
767
748
  console.log(message, ...optionalParams);
768
749
  }
769
750
  else {
770
751
  console.log();
771
752
  }
772
- // Restart the spinner if it was running before the log
773
- /* c8 ignore next 3 */
774
- if (spinnerSpinning) {
775
- cli.spinner.start();
776
- }
777
753
  }
778
754
  async function error(message, ...optionalParams) {
779
- const spinnerSpinning = cli.spinner.isSpinning;
780
- /* c8 ignore next 3 */
781
- if (spinnerSpinning) {
782
- cli.spinner.stop();
783
- }
784
755
  const errorOutput = cli.getSettingWithDefaultValue(settingsNames.errorOutput, 'stderr');
785
756
  if (errorOutput === 'stdout') {
786
757
  console.log(message, ...optionalParams);
@@ -788,40 +759,15 @@ async function error(message, ...optionalParams) {
788
759
  else {
789
760
  console.error(message, ...optionalParams);
790
761
  }
791
- // Restart the spinner if it was running before the log
792
- /* c8 ignore next 3 */
793
- if (spinnerSpinning) {
794
- cli.spinner.start();
795
- }
796
762
  }
797
763
  async function promptForSelection(config) {
798
- const spinnerSpinning = cli.spinner.isSpinning;
799
- /* c8 ignore next 3 */
800
- if (spinnerSpinning) {
801
- cli.spinner.stop();
802
- }
803
764
  const answer = await prompt.forSelection(config);
804
765
  await cli.error('');
805
- // Restart the spinner if it was running before the prompt
806
- /* c8 ignore next 3 */
807
- if (spinnerSpinning) {
808
- cli.spinner.start();
809
- }
810
766
  return answer;
811
767
  }
812
768
  async function promptForConfirmation(config) {
813
- const spinnerSpinning = cli.spinner.isSpinning;
814
- /* c8 ignore next 3 */
815
- if (spinnerSpinning) {
816
- cli.spinner.stop();
817
- }
818
769
  const answer = await prompt.forConfirmation(config);
819
770
  await cli.error('');
820
- // Restart the spinner if it was running before the prompt
821
- /* c8 ignore next 3 */
822
- if (spinnerSpinning) {
823
- cli.spinner.start();
824
- }
825
771
  return answer;
826
772
  }
827
773
  async function handleMultipleResultsFound(message, values) {
@@ -885,14 +831,6 @@ export const cli = {
885
831
  printAvailableCommands,
886
832
  promptForConfirmation,
887
833
  promptForSelection,
888
- shouldTrimOutput,
889
- spinner
890
- };
891
- const spinnerOptions = {
892
- text: 'Running command...',
893
- /* c8 ignore next 1 */
894
- stream: cli.getSettingWithDefaultValue('errorOutput', 'stderr') === 'stderr' ? process.stderr : process.stdout,
895
- discardStdin: false
834
+ shouldTrimOutput
896
835
  };
897
- cli.spinner = ora(spinnerOptions);
898
836
  //# sourceMappingURL=cli.js.map
@@ -42,8 +42,6 @@ class SetupCommand extends AnonymousCommand {
42
42
  await this.configureSettings(settings, true, logger);
43
43
  return;
44
44
  }
45
- // stop the spinner. Fixes #5598
46
- cli.spinner.stop();
47
45
  await logger.logToStderr(`Welcome to the CLI for Microsoft 365 setup!`);
48
46
  await logger.logToStderr(`This command will guide you through the process of configuring the CLI for your needs.`);
49
47
  await logger.logToStderr(`Please, answer the following questions and we'll define a set of settings to best match how you intend to use the CLI.`);
@@ -85,8 +83,6 @@ class SetupCommand extends AnonymousCommand {
85
83
  await logger.logToStderr('');
86
84
  await logger.logToStderr('Configuring settings...');
87
85
  await logger.logToStderr('');
88
- // start the spinner. Fixes #5598
89
- cli.spinner.start();
90
86
  await this.configureSettings(settings, false, logger);
91
87
  if (!this.verbose) {
92
88
  await logger.logToStderr('');
@@ -13,6 +13,8 @@ import GraphCommand from '../../../base/GraphCommand.js';
13
13
  import commands from '../../commands.js';
14
14
  import { entraGroup } from '../../../../utils/entraGroup.js';
15
15
  import aadCommands from '../../aadCommands.js';
16
+ import { accessToken } from '../../../../utils/accessToken.js';
17
+ import auth from '../../../../Auth.js';
16
18
  class EntraM365GroupSetCommand extends GraphCommand {
17
19
  get name() {
18
20
  return commands.M365GROUP_SET;
@@ -35,22 +37,22 @@ class EntraM365GroupSetCommand extends GraphCommand {
35
37
  async commandAction(logger, args) {
36
38
  await this.showDeprecationWarning(logger, aadCommands.M365GROUP_SET, commands.M365GROUP_SET);
37
39
  try {
40
+ if ((args.options.allowExternalSenders !== undefined || args.options.autoSubscribeNewMembers !== undefined) && accessToken.isAppOnlyAccessToken(auth.connection.accessTokens[auth.defaultResource].accessToken)) {
41
+ throw `Option 'allowExternalSenders' and 'autoSubscribeNewMembers' can only be used when using delegated permissions.`;
42
+ }
38
43
  const isUnifiedGroup = await entraGroup.isUnifiedGroup(args.options.id);
39
44
  if (!isUnifiedGroup) {
40
45
  throw Error(`Specified group with id '${args.options.id}' is not a Microsoft 365 group.`);
41
46
  }
42
- if (args.options.displayName || args.options.description || typeof args.options.isPrivate !== 'undefined') {
43
- if (this.verbose) {
44
- await logger.logToStderr(`Updating Microsoft 365 Group ${args.options.id}...`);
45
- }
46
- const update = {};
47
- if (args.options.displayName) {
48
- update.displayName = args.options.displayName;
49
- }
50
- if (args.options.description) {
51
- update.description = args.options.description;
52
- }
53
- if (typeof args.options.isPrivate !== 'undefined') {
47
+ if (this.verbose) {
48
+ await logger.logToStderr(`Updating Microsoft 365 Group ${args.options.id}...`);
49
+ }
50
+ if (args.options.displayName || args.options.description !== undefined || args.options.isPrivate !== undefined) {
51
+ const update = {
52
+ displayName: args.options.displayName,
53
+ description: args.options.description !== '' ? args.options.description : null
54
+ };
55
+ if (args.options.isPrivate !== undefined) {
54
56
  update.visibility = args.options.isPrivate ? 'Private' : 'Public';
55
57
  }
56
58
  const requestOptions = {
@@ -63,6 +65,24 @@ class EntraM365GroupSetCommand extends GraphCommand {
63
65
  };
64
66
  await request.patch(requestOptions);
65
67
  }
68
+ // This has to be a separate request due to some Graph API limitations. Otherwise it will throw an error.
69
+ if (args.options.allowExternalSenders !== undefined || args.options.autoSubscribeNewMembers !== undefined || args.options.hideFromAddressLists !== undefined || args.options.hideFromOutlookClients !== undefined) {
70
+ const requestBody = {
71
+ allowExternalSenders: args.options.allowExternalSenders,
72
+ autoSubscribeNewMembers: args.options.autoSubscribeNewMembers,
73
+ hideFromAddressLists: args.options.hideFromAddressLists,
74
+ hideFromOutlookClients: args.options.hideFromOutlookClients
75
+ };
76
+ const requestOptions = {
77
+ url: `${this.resource}/v1.0/groups/${args.options.id}`,
78
+ headers: {
79
+ accept: 'application/json;odata.metadata=none'
80
+ },
81
+ responseType: 'json',
82
+ data: requestBody
83
+ };
84
+ await request.patch(requestOptions);
85
+ }
66
86
  if (args.options.logoPath) {
67
87
  const fullPath = path.resolve(args.options.logoPath);
68
88
  if (this.verbose) {
@@ -172,8 +192,12 @@ _EntraM365GroupSetCommand_instances = new WeakSet(), _EntraM365GroupSetCommand_i
172
192
  description: typeof args.options.description !== 'undefined',
173
193
  owners: typeof args.options.owners !== 'undefined',
174
194
  members: typeof args.options.members !== 'undefined',
175
- isPrivate: args.options.isPrivate,
176
- logoPath: typeof args.options.logoPath !== 'undefined'
195
+ isPrivate: !!args.options.isPrivate,
196
+ logoPath: typeof args.options.logoPath !== 'undefined',
197
+ allowExternalSenders: !!args.options.allowExternalSenders,
198
+ autoSubscribeNewMembers: !!args.options.autoSubscribeNewMembers,
199
+ hideFromAddressLists: !!args.options.hideFromAddressLists,
200
+ hideFromOutlookClients: !!args.options.hideFromOutlookClients
177
201
  });
178
202
  });
179
203
  }, _EntraM365GroupSetCommand_initOptions = function _EntraM365GroupSetCommand_initOptions() {
@@ -192,36 +216,49 @@ _EntraM365GroupSetCommand_instances = new WeakSet(), _EntraM365GroupSetCommand_i
192
216
  autocomplete: ['true', 'false']
193
217
  }, {
194
218
  option: '-l, --logoPath [logoPath]'
219
+ }, {
220
+ option: '--allowExternalSenders [allowExternalSenders]',
221
+ autocomplete: ['true', 'false']
222
+ }, {
223
+ option: '--autoSubscribeNewMembers [autoSubscribeNewMembers]',
224
+ autocomplete: ['true', 'false']
225
+ }, {
226
+ option: '--hideFromAddressLists [hideFromAddressLists]',
227
+ autocomplete: ['true', 'false']
228
+ }, {
229
+ option: '--hideFromOutlookClients [hideFromOutlookClients]',
230
+ autocomplete: ['true', 'false']
195
231
  });
196
232
  }, _EntraM365GroupSetCommand_initTypes = function _EntraM365GroupSetCommand_initTypes() {
197
- this.types.boolean.push('isPrivate');
233
+ this.types.boolean.push('isPrivate', 'allowEternalSenders', 'autoSubscribeNewMembers', 'hideFromAddressLists', 'hideFromOutlookClients');
234
+ this.types.string.push('id', 'displayName', 'description', 'owners', 'members', 'logoPath');
198
235
  }, _EntraM365GroupSetCommand_initValidators = function _EntraM365GroupSetCommand_initValidators() {
199
236
  this.validators.push(async (args) => {
200
237
  if (!args.options.displayName &&
201
- !args.options.description &&
238
+ args.options.description === undefined &&
202
239
  !args.options.members &&
203
240
  !args.options.owners &&
204
- typeof args.options.isPrivate === 'undefined' &&
205
- !args.options.logoPath) {
206
- return 'Specify at least one property to update';
241
+ args.options.isPrivate === undefined &&
242
+ !args.options.logoPath &&
243
+ args.options.allowExternalSenders === undefined &&
244
+ args.options.autoSubscribeNewMembers === undefined &&
245
+ args.options.hideFromAddressLists === undefined &&
246
+ args.options.hideFromOutlookClients === undefined) {
247
+ return 'Specify at least one option to update.';
207
248
  }
208
249
  if (!validation.isValidGuid(args.options.id)) {
209
250
  return `${args.options.id} is not a valid GUID`;
210
251
  }
211
252
  if (args.options.owners) {
212
- const owners = args.options.owners.split(',').map(o => o.trim());
213
- for (let i = 0; i < owners.length; i++) {
214
- if (owners[i].indexOf('@') < 0) {
215
- return `${owners[i]} is not a valid userPrincipalName`;
216
- }
253
+ const isValidArray = validation.isValidUserPrincipalNameArray(args.options.owners);
254
+ if (isValidArray !== true) {
255
+ return `Option 'owners' contains one or more invalid UPNs: ${isValidArray}.`;
217
256
  }
218
257
  }
219
258
  if (args.options.members) {
220
- const members = args.options.members.split(',').map(m => m.trim());
221
- for (let i = 0; i < members.length; i++) {
222
- if (members[i].indexOf('@') < 0) {
223
- return `${members[i]} is not a valid userPrincipalName`;
224
- }
259
+ const isValidArray = validation.isValidUserPrincipalNameArray(args.options.members);
260
+ if (isValidArray !== true) {
261
+ return `Option 'members' contains one or more invalid UPNs: ${isValidArray}.`;
225
262
  }
226
263
  }
227
264
  if (args.options.logoPath) {
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=MultitenantOrganization.js.map
@@ -0,0 +1,32 @@
1
+ import request from '../../../../request.js';
2
+ import GraphCommand from '../../../base/GraphCommand.js';
3
+ import commands from '../../commands.js';
4
+ class EntraMultitenantGetCommand extends GraphCommand {
5
+ get name() {
6
+ return commands.MULTITENANT_GET;
7
+ }
8
+ get description() {
9
+ return 'Gets properties of the multitenant organization';
10
+ }
11
+ async commandAction(logger) {
12
+ const requestOptions = {
13
+ url: `${this.resource}/v1.0/tenantRelationships/multiTenantOrganization`,
14
+ headers: {
15
+ accept: 'application/json;odata.metadata=none'
16
+ },
17
+ responseType: 'json'
18
+ };
19
+ try {
20
+ if (this.verbose) {
21
+ await logger.logToStderr('Retrieving multitenant organization...');
22
+ }
23
+ const multitenantOrg = await request.get(requestOptions);
24
+ await logger.log(multitenantOrg);
25
+ }
26
+ catch (err) {
27
+ this.handleRejectedODataJsonPromise(err);
28
+ }
29
+ }
30
+ }
31
+ export default new EntraMultitenantGetCommand();
32
+ //# sourceMappingURL=multitenant-get.js.map
@@ -72,6 +72,7 @@ export default {
72
72
  M365GROUP_USER_LIST: `${prefix} m365group user list`,
73
73
  M365GROUP_USER_REMOVE: `${prefix} m365group user remove`,
74
74
  M365GROUP_USER_SET: `${prefix} m365group user set`,
75
+ MULTITENANT_GET: `${prefix} multitenant get`,
75
76
  OAUTH2GRANT_ADD: `${prefix} oauth2grant add`,
76
77
  OAUTH2GRANT_LIST: `${prefix} oauth2grant list`,
77
78
  OAUTH2GRANT_REMOVE: `${prefix} oauth2grant remove`,
@@ -5,11 +5,10 @@ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (
5
5
  };
6
6
  var _ExternalConnectionDoctorCommand_instances, _a, _ExternalConnectionDoctorCommand_initOptions, _ExternalConnectionDoctorCommand_initValidators;
7
7
  import os from 'os';
8
- import { cli } from '../../../../cli/cli.js';
9
8
  import request from '../../../../request.js';
10
- import { settingsNames } from '../../../../settingsNames.js';
11
9
  import GraphCommand from '../../../base/GraphCommand.js';
12
10
  import commands from '../../commands.js';
11
+ import { CheckStatus, formatting } from '../../../../utils/formatting.js';
13
12
  class ExternalConnectionDoctorCommand extends GraphCommand {
14
13
  get name() {
15
14
  return commands.CONNECTION_DOCTOR;
@@ -28,9 +27,7 @@ class ExternalConnectionDoctorCommand extends GraphCommand {
28
27
  const ux = args.options.ux ?? 'all';
29
28
  const output = args.options.output;
30
29
  this.checksStatus = [];
31
- const showSpinner = cli.getSettingWithDefaultValue(settingsNames.showSpinner, true) &&
32
- output === 'text' &&
33
- typeof global.it === 'undefined';
30
+ const show = output === 'text';
34
31
  let checks = [
35
32
  {
36
33
  id: 'loadExternalConnection',
@@ -114,17 +111,10 @@ class ExternalConnectionDoctorCommand extends GraphCommand {
114
111
  if (this.debug) {
115
112
  await logger.logToStderr(`Running check ${check.id}...`);
116
113
  }
117
- // don't show spinner if running tests
118
- /* c8 ignore next 3 */
119
- if (showSpinner) {
120
- cli.spinner.start(check.text);
121
- }
122
114
  // only automated checks have functions
123
115
  if (!check.fn) {
124
- // don't show spinner if running tests
125
- /* c8 ignore next 3 */
126
- if (showSpinner) {
127
- cli.spinner.info(`${check.text} (manual)`);
116
+ if (show) {
117
+ await logger.log(formatting.getStatus(CheckStatus.Information, `${check.text} (manual)`));
128
118
  }
129
119
  this.checksStatus.push({
130
120
  ...check,
@@ -135,23 +125,19 @@ class ExternalConnectionDoctorCommand extends GraphCommand {
135
125
  const result = await check.fn.bind(this)(check.id, args);
136
126
  this.checksStatus.push({ ...check, ...result });
137
127
  if (result.status === 'passed') {
138
- // don't show spinner if running tests
139
- /* c8 ignore next 3 */
140
- if (showSpinner) {
141
- cli.spinner.succeed();
128
+ if (show) {
129
+ await logger.log(formatting.getStatus(CheckStatus.Success, check.text));
142
130
  }
143
131
  continue;
144
132
  }
145
133
  if (result.status === 'failed') {
146
- // don't show spinner if running tests
147
- /* c8 ignore next 9 */
148
- if (showSpinner) {
134
+ if (show) {
149
135
  const message = `${check.text}: ${result.errorMessage}`;
150
136
  if (check.type === 'required') {
151
- cli.spinner.fail(message);
137
+ await logger.log(formatting.getStatus(CheckStatus.Failure, message));
152
138
  }
153
139
  else {
154
- cli.spinner.warn(message);
140
+ await logger.log(formatting.getStatus(CheckStatus.Warning, check.text));
155
141
  }
156
142
  }
157
143
  if (result.shouldStop) {
@@ -159,7 +145,7 @@ class ExternalConnectionDoctorCommand extends GraphCommand {
159
145
  }
160
146
  }
161
147
  }
162
- if (output === 'text' || output === 'none') {
148
+ if (show || output === 'none') {
163
149
  return;
164
150
  }
165
151
  this.checksStatus.forEach(s => {
@@ -3,7 +3,7 @@ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (
3
3
  if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
4
4
  return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
5
5
  };
6
- var _FlowListCommand_instances, _FlowListCommand_initTelemetry, _FlowListCommand_initOptions, _FlowListCommand_initValidators;
6
+ var _FlowListCommand_instances, _FlowListCommand_initTelemetry, _FlowListCommand_initOptions, _FlowListCommand_initValidators, _FlowListCommand_initTypes;
7
7
  import { formatting } from '../../../utils/formatting.js';
8
8
  import { odata } from '../../../utils/odata.js';
9
9
  import PowerAutomateCommand from '../../base/PowerAutomateCommand.js';
@@ -13,7 +13,7 @@ class FlowListCommand extends PowerAutomateCommand {
13
13
  return commands.LIST;
14
14
  }
15
15
  get description() {
16
- return 'Lists Microsoft Flows in the given environment';
16
+ return 'Lists Power Automate flows in the given environment';
17
17
  }
18
18
  defaultProperties() {
19
19
  return ['name', 'displayName'];
@@ -25,8 +25,12 @@ class FlowListCommand extends PowerAutomateCommand {
25
25
  __classPrivateFieldGet(this, _FlowListCommand_instances, "m", _FlowListCommand_initTelemetry).call(this);
26
26
  __classPrivateFieldGet(this, _FlowListCommand_instances, "m", _FlowListCommand_initOptions).call(this);
27
27
  __classPrivateFieldGet(this, _FlowListCommand_instances, "m", _FlowListCommand_initValidators).call(this);
28
+ __classPrivateFieldGet(this, _FlowListCommand_instances, "m", _FlowListCommand_initTypes).call(this);
28
29
  }
29
30
  async commandAction(logger, args) {
31
+ if (this.verbose) {
32
+ await logger.logToStderr(`Getting Power Automate flows${args.options.asAdmin && ' as admin'} in environment '${args.options.environmentName}'...`);
33
+ }
30
34
  try {
31
35
  const { environmentName, asAdmin, sharingStatus, includeSolutions } = args.options;
32
36
  let items = [];
@@ -42,7 +46,8 @@ class FlowListCommand extends PowerAutomateCommand {
42
46
  let url = this.getApiUrl(environmentName, asAdmin, includeSolutions, 'personal');
43
47
  items = await odata.getAllItems(url);
44
48
  url = this.getApiUrl(environmentName, asAdmin, includeSolutions, 'team');
45
- items = await odata.getAllItems(url);
49
+ const teamFlows = await odata.getAllItems(url);
50
+ items = items.concat(teamFlows);
46
51
  }
47
52
  else {
48
53
  const url = this.getApiUrl(environmentName, asAdmin, includeSolutions);
@@ -50,34 +55,25 @@ class FlowListCommand extends PowerAutomateCommand {
50
55
  }
51
56
  // Remove duplicates
52
57
  items = items.filter((flow, index, self) => index === self.findIndex(f => f.id === flow.id));
53
- if (items.length > 0) {
54
- items.forEach(i => {
55
- i.displayName = i.properties.displayName;
58
+ if (args.options.output && args.options.output !== 'json') {
59
+ items.forEach(flow => {
60
+ flow.displayName = flow.properties.displayName;
56
61
  });
57
- await logger.log(items);
58
- }
59
- else {
60
- if (this.verbose) {
61
- await logger.logToStderr('No Flows found');
62
- }
63
62
  }
63
+ await logger.log(items);
64
64
  }
65
65
  catch (err) {
66
66
  this.handleRejectedODataJsonPromise(err);
67
67
  }
68
68
  }
69
69
  getApiUrl(environmentName, asAdmin, includeSolutionFlows, filter) {
70
- let url = `${this.resource}/providers/Microsoft.ProcessSimple${asAdmin ? '/scopes/admin' : ''}/environments/${formatting.encodeQueryParameter(environmentName)}/flows?api-version=2016-11-01`;
71
- if (filter === 'personal') {
72
- url += `&$filter=search('personal')`;
73
- }
74
- else if (filter === 'team') {
75
- url += `&$filter=search('team')`;
76
- }
77
- if (includeSolutionFlows) {
78
- url += '&include=includeSolutionCloudFlows';
79
- }
80
- return url;
70
+ const baseEndpoint = `${this.resource}/providers/Microsoft.ProcessSimple`;
71
+ const environmentSegment = `/environments/${formatting.encodeQueryParameter(environmentName)}`;
72
+ const adminSegment = `/scopes/admin${environmentSegment}/v2`;
73
+ const flowsEndpoint = '/flows?api-version=2016-11-01';
74
+ const filterQuery = filter === 'personal' || filter === 'team' ? `&$filter=search('${filter}')` : '';
75
+ const includeQuery = includeSolutionFlows ? '&include=includeSolutionCloudFlows' : '';
76
+ return `${baseEndpoint}${asAdmin ? adminSegment : environmentSegment}${flowsEndpoint}${filterQuery}${includeQuery}`;
81
77
  }
82
78
  }
83
79
  _FlowListCommand_instances = new WeakSet(), _FlowListCommand_initTelemetry = function _FlowListCommand_initTelemetry() {
@@ -105,10 +101,13 @@ _FlowListCommand_instances = new WeakSet(), _FlowListCommand_initTelemetry = fun
105
101
  return `The options asAdmin and sharingStatus cannot be specified together.`;
106
102
  }
107
103
  if (args.options.sharingStatus && !this.allowedSharingStatuses.some(status => status === args.options.sharingStatus)) {
108
- return `${args.options.sharingStatus} is not a valid sharing status. Allowed values are: ${this.allowedSharingStatuses.join(',')}`;
104
+ return `${args.options.sharingStatus} is not a valid sharing status. Allowed values are: ${this.allowedSharingStatuses.join(', ')}`;
109
105
  }
110
106
  return true;
111
107
  });
108
+ }, _FlowListCommand_initTypes = function _FlowListCommand_initTypes() {
109
+ this.types.string.push('environmentName', 'sharingStatus');
110
+ this.types.boolean.push('includeSolutions', 'asAdmin');
112
111
  };
113
112
  export default new FlowListCommand();
114
113
  //# sourceMappingURL=flow-list.js.map
@@ -141,8 +141,10 @@ _GraphSubscriptionAddCommand_instances = new WeakSet(), _GraphSubscriptionAddCom
141
141
  });
142
142
  }, _GraphSubscriptionAddCommand_initValidators = function _GraphSubscriptionAddCommand_initValidators() {
143
143
  this.validators.push(async (args) => {
144
- if (args.options.notificationUrl.indexOf('https://') !== 0) {
145
- return `The specified notification URL '${args.options.notificationUrl}' does not start with 'https://'`;
144
+ if (!args.options.notificationUrl.toLowerCase().startsWith('https://')
145
+ && !args.options.notificationUrl.toLowerCase().startsWith('eventhub:https://')
146
+ && !args.options.notificationUrl.toLowerCase().startsWith('eventgrid:?azuresubscriptionid=')) {
147
+ return `The specified notification URL '${args.options.notificationUrl}' does not start with either 'https://' or 'EventHub:https://' or 'EventGrid:?azuresubscriptionid='`;
146
148
  }
147
149
  if (!this.isValidChangeTypes(args.options.changeTypes)) {
148
150
  return `The specified changeTypes are invalid. Valid options are 'created', 'updated' and 'deleted'`;