@pnp/cli-microsoft365 5.0.0-beta.9b6e729 → 5.0.0-beta.ae24497

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 (98) hide show
  1. package/.devcontainer/Dockerfile +1 -1
  2. package/.devcontainer/devcontainer.json +9 -1
  3. package/.eslintrc.js +1 -0
  4. package/README.md +1 -1
  5. package/dist/Utils.js +7 -0
  6. package/dist/appInsights.js +5 -2
  7. package/dist/cli/Cli.js +24 -4
  8. package/dist/m365/aad/commands/app/app-add.js +58 -5
  9. package/dist/m365/aad/commands/app/app-get.js +97 -0
  10. package/dist/m365/aad/commands/group/group-list.js +41 -0
  11. package/dist/m365/aad/commands/o365group/{GroupUser.js → GroupExtended.js} +1 -1
  12. package/dist/m365/aad/commands/o365group/o365group-add.js +56 -50
  13. package/dist/m365/aad/commands/o365group/o365group-user-set.js +3 -3
  14. package/dist/m365/aad/commands/oauth2grant/oauth2grant-list.js +4 -4
  15. package/dist/m365/aad/commands/oauth2grant/oauth2grant-remove.js +36 -12
  16. package/dist/m365/aad/commands/user/user-get.js +33 -6
  17. package/dist/m365/aad/commands/user/user-hibp.js +67 -0
  18. package/dist/m365/aad/commands/user/user-password-validate.js +42 -0
  19. package/dist/m365/aad/commands.js +4 -0
  20. package/dist/m365/app/commands/permission/permission-list.js +266 -0
  21. package/dist/m365/app/commands.js +7 -0
  22. package/dist/m365/base/AppCommand.js +76 -0
  23. package/dist/m365/{aad/commands/o365group/Group.js → base/M365RcJson.js} +1 -1
  24. package/dist/m365/cli/commands/cli-doctor.js +2 -0
  25. package/dist/m365/cli/commands/config/config-set.js +4 -1
  26. package/dist/m365/file/commands/file-list.js +181 -0
  27. package/dist/m365/file/commands.js +2 -1
  28. package/dist/m365/pa/cds-project-mutator.js +1 -1
  29. package/dist/m365/pa/commands/app/app-list.js +28 -1
  30. package/dist/m365/planner/AppliedCategories.js +3 -0
  31. package/dist/m365/planner/commands/task/task-add.js +288 -0
  32. package/dist/m365/planner/commands/task/task-set.js +357 -0
  33. package/dist/m365/planner/commands.js +3 -1
  34. package/dist/m365/search/commands/externalconnection/externalconnection-add.js +99 -0
  35. package/dist/m365/search/commands.js +7 -0
  36. package/dist/m365/spfx/commands/project/project-upgrade/rules/FN006005_CFG_PS_metadata.js +63 -0
  37. package/dist/m365/spfx/commands/project/project-upgrade/rules/FN006006_CFG_PS_features.js +60 -0
  38. package/dist/m365/spfx/commands/project/project-upgrade/upgrade-1.13.1.js +53 -0
  39. package/dist/m365/spfx/commands/project/project-upgrade/upgrade-1.14.0-beta.4.js +57 -0
  40. package/dist/m365/spfx/commands/project/project-upgrade.js +17 -13
  41. package/dist/m365/spfx/commands/spfx-doctor.js +176 -62
  42. package/dist/m365/spo/commands/contenttype/contenttype-list.js +52 -0
  43. package/dist/m365/spo/commands/group/group-user-add.js +64 -13
  44. package/dist/m365/spo/commands/list/list-get.js +6 -2
  45. package/dist/m365/spo/commands/page/Page.js +3 -1
  46. package/dist/m365/spo/commands/page/page-add.js +7 -10
  47. package/dist/m365/spo/commands/page/page-set.js +7 -10
  48. package/dist/m365/spo/commands/site/site-ensure.js +1 -1
  49. package/dist/m365/spo/commands/site/site-recyclebinitem-list.js +76 -0
  50. package/dist/m365/spo/commands/site/site-remove.js +98 -30
  51. package/dist/m365/spo/commands/web/web-installedlanguage-list.js +48 -0
  52. package/dist/m365/spo/commands.js +4 -1
  53. package/dist/m365/teams/commands/app/app-list.js +9 -6
  54. package/dist/m365/teams/commands/chat/chat-list.js +43 -0
  55. package/dist/m365/teams/commands/chat/chat-member-list.js +42 -0
  56. package/dist/m365/teams/commands/chat/chat-message-list.js +60 -0
  57. package/dist/m365/teams/commands/message/message-get.js +1 -1
  58. package/dist/m365/teams/commands/report/report-directroutingcalls.js +1 -1
  59. package/dist/m365/teams/commands/tab/tab-get.js +9 -6
  60. package/dist/m365/teams/commands.js +3 -0
  61. package/dist/request.js +9 -4
  62. package/dist/settingsNames.js +6 -1
  63. package/docs/docs/cmd/_global.md +2 -2
  64. package/docs/docs/cmd/aad/app/app-add.md +11 -0
  65. package/docs/docs/cmd/aad/app/app-get.md +48 -0
  66. package/docs/docs/cmd/aad/group/group-list.md +21 -0
  67. package/docs/docs/cmd/aad/o365group/o365group-add.md +1 -0
  68. package/docs/docs/cmd/aad/oauth2grant/oauth2grant-list.md +2 -2
  69. package/docs/docs/cmd/aad/oauth2grant/oauth2grant-remove.md +9 -0
  70. package/docs/docs/cmd/aad/sp/sp-add.md +1 -1
  71. package/docs/docs/cmd/aad/user/user-get.md +13 -4
  72. package/docs/docs/cmd/aad/user/user-hibp.md +46 -0
  73. package/docs/docs/cmd/aad/user/user-password-validate.md +29 -0
  74. package/docs/docs/cmd/app/permission/permission-list.md +36 -0
  75. package/docs/docs/cmd/file/file-list.md +46 -0
  76. package/docs/docs/cmd/pa/app/app-list.md +17 -1
  77. package/docs/docs/cmd/planner/task/task-add.md +78 -0
  78. package/docs/docs/cmd/planner/task/task-set.md +99 -0
  79. package/docs/docs/cmd/search/externalconnection/externalconnection-add.md +43 -0
  80. package/docs/docs/cmd/spfx/project/project-externalize.md +1 -1
  81. package/docs/docs/cmd/spfx/project/project-rename.md +1 -1
  82. package/docs/docs/cmd/spfx/project/project-upgrade.md +1 -1
  83. package/docs/docs/cmd/spfx/spfx-doctor.md +1 -1
  84. package/docs/docs/cmd/spo/contenttype/contenttype-list.md +33 -0
  85. package/docs/docs/cmd/spo/group/group-user-add.md +24 -6
  86. package/docs/docs/cmd/spo/list/list-get.md +9 -0
  87. package/docs/docs/cmd/spo/page/page-add.md +2 -2
  88. package/docs/docs/cmd/spo/page/page-set.md +3 -3
  89. package/docs/docs/cmd/spo/site/site-recyclebinitem-list.md +40 -0
  90. package/docs/docs/cmd/spo/site/site-remove.md +3 -1
  91. package/docs/docs/cmd/spo/web/web-installedlanguage-list.md +24 -0
  92. package/docs/docs/cmd/teams/chat/chat-list.md +30 -0
  93. package/docs/docs/cmd/teams/chat/chat-member-list.md +24 -0
  94. package/docs/docs/cmd/teams/chat/chat-message-list.md +24 -0
  95. package/docs/docs/cmd/teams/message/message-get.md +0 -3
  96. package/docs/docs/cmd/teams/report/report-directroutingcalls.md +0 -3
  97. package/npm-shrinkwrap.json +1137 -1756
  98. package/package.json +37 -29
@@ -3,8 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  const cli_1 = require("../../../../cli");
4
4
  const Command_1 = require("../../../../Command");
5
5
  const request_1 = require("../../../../request");
6
- const SpoCommand_1 = require("../../../base/SpoCommand");
7
6
  const AadUserGetCommand = require("../../../aad/commands/user/user-get");
7
+ const SpoCommand_1 = require("../../../base/SpoCommand");
8
8
  const commands_1 = require("../../commands");
9
9
  class SpoGroupUserAddCommand extends SpoCommand_1.default {
10
10
  get name() {
@@ -17,15 +17,21 @@ class SpoGroupUserAddCommand extends SpoCommand_1.default {
17
17
  return ['DisplayName', 'Email'];
18
18
  }
19
19
  commandAction(logger, args, cb) {
20
- this.getOnlyActiveUsers(args, logger)
20
+ let groupId = 0;
21
+ this
22
+ .getGroupId(args)
23
+ .then((_groupId) => {
24
+ groupId = _groupId;
25
+ return this.getOnlyActiveUsers(args, logger);
26
+ })
21
27
  .then((resolvedUsernameList) => {
22
28
  if (this.verbose) {
23
- logger.logToStderr(`Start adding Active user/s to SharePoint Group ${args.options.groupId}...`);
29
+ logger.logToStderr(`Start adding Active user/s to SharePoint Group ${args.options.groupId ? args.options.groupId : args.options.groupName}`);
24
30
  }
25
31
  const data = {
26
32
  url: args.options.webUrl,
27
33
  peoplePickerInput: this.getFormattedUserList(resolvedUsernameList),
28
- roleValue: `group:${args.options.groupId}`
34
+ roleValue: `group:${groupId}`
29
35
  };
30
36
  const requestOptions = {
31
37
  url: `${args.options.webUrl}/_api/SP.Web.ShareObject`,
@@ -46,24 +52,51 @@ class SpoGroupUserAddCommand extends SpoCommand_1.default {
46
52
  cb();
47
53
  }, (err) => this.handleRejectedODataJsonPromise(err, logger, cb));
48
54
  }
55
+ getGroupId(args) {
56
+ if (args.options.groupId) {
57
+ return Promise.resolve(args.options.groupId);
58
+ }
59
+ const requestOptions = {
60
+ url: `${args.options.webUrl}/_api/web/sitegroups/GetByName('${encodeURIComponent(args.options.groupName)}')`,
61
+ headers: {
62
+ 'accept': 'application/json;odata=nometadata'
63
+ },
64
+ responseType: 'json'
65
+ };
66
+ return request_1.default
67
+ .get(requestOptions)
68
+ .then(response => {
69
+ const groupId = response.Id;
70
+ if (!groupId) {
71
+ return Promise.reject(`The specified group not exist in the SharePoint site`);
72
+ }
73
+ return Promise.resolve(groupId);
74
+ });
75
+ }
49
76
  getOnlyActiveUsers(args, logger) {
50
77
  if (this.verbose) {
51
78
  logger.logToStderr(`Removing Users which are not active from the original list`);
52
79
  }
53
- const activeUsernamelist = [];
54
- return Promise.all(args.options.userName.split(",").map(singleUsername => {
80
+ const activeUserNameList = [];
81
+ const userInfo = args.options.userName ? args.options.userName : args.options.email;
82
+ return Promise.all(userInfo.split(',').map(singleUserName => {
55
83
  const options = {
56
- userName: singleUsername.trim(),
57
84
  output: 'json',
58
85
  debug: args.options.debug,
59
86
  verbose: args.options.verbose
60
87
  };
88
+ if (args.options.userName) {
89
+ options.userName = singleUserName.trim();
90
+ }
91
+ else {
92
+ options.email = singleUserName.trim();
93
+ }
61
94
  return cli_1.Cli.executeCommandWithOutput(AadUserGetCommand, { options: Object.assign(Object.assign({}, options), { _: [] }) })
62
95
  .then((getUserGetOutput) => {
63
96
  if (this.debug) {
64
97
  logger.logToStderr(getUserGetOutput.stderr);
65
98
  }
66
- activeUsernamelist.push(JSON.parse(getUserGetOutput.stdout).userPrincipalName);
99
+ activeUserNameList.push(JSON.parse(getUserGetOutput.stdout).userPrincipalName);
67
100
  }, (err) => {
68
101
  if (this.debug) {
69
102
  logger.logToStderr(err.stderr);
@@ -71,7 +104,7 @@ class SpoGroupUserAddCommand extends SpoCommand_1.default {
71
104
  });
72
105
  }))
73
106
  .then(() => {
74
- return Promise.resolve(activeUsernamelist);
107
+ return Promise.resolve(activeUserNameList);
75
108
  });
76
109
  }
77
110
  getFormattedUserList(activeUserList) {
@@ -86,10 +119,16 @@ class SpoGroupUserAddCommand extends SpoCommand_1.default {
86
119
  option: '-u, --webUrl <webUrl>'
87
120
  },
88
121
  {
89
- option: '--groupId <groupId>'
122
+ option: '--groupId [groupId]'
123
+ },
124
+ {
125
+ option: '--groupName [groupName]'
126
+ },
127
+ {
128
+ option: '--userName [userName]'
90
129
  },
91
130
  {
92
- option: '--userName <userName>'
131
+ option: '--email [email]'
93
132
  }
94
133
  ];
95
134
  const parentOptions = super.options();
@@ -100,8 +139,20 @@ class SpoGroupUserAddCommand extends SpoCommand_1.default {
100
139
  if (isValidSharePointUrl !== true) {
101
140
  return isValidSharePointUrl;
102
141
  }
103
- if (typeof args.options.groupId !== 'number') {
104
- return `Group Id : ${args.options.groupId} is not a number`;
142
+ if (!args.options.groupId && !args.options.groupName) {
143
+ return 'Specify either groupId or groupName';
144
+ }
145
+ if (args.options.groupId && args.options.groupName) {
146
+ return 'Specify either groupId or groupName but not both';
147
+ }
148
+ if (!args.options.userName && !args.options.email) {
149
+ return 'Specify either userName or email';
150
+ }
151
+ if (args.options.userName && args.options.email) {
152
+ return 'Specify either userName or email but not both';
153
+ }
154
+ if (args.options.groupId && isNaN(args.options.groupId)) {
155
+ return `Specified groupId ${args.options.groupId} is not a number`;
105
156
  }
106
157
  return true;
107
158
  }
@@ -16,6 +16,7 @@ class SpoListGetCommand extends SpoCommand_1.default {
16
16
  telemetryProps.id = (!(!args.options.id)).toString();
17
17
  telemetryProps.title = (!(!args.options.title)).toString();
18
18
  telemetryProps.properties = (!(!args.options.properties)).toString();
19
+ telemetryProps.withPermissions = typeof args.options.withPermissions !== 'undefined';
19
20
  return telemetryProps;
20
21
  }
21
22
  commandAction(logger, args, cb) {
@@ -29,10 +30,10 @@ class SpoListGetCommand extends SpoCommand_1.default {
29
30
  else {
30
31
  requestUrl = `${args.options.webUrl}/_api/web/lists/GetByTitle('${encodeURIComponent(args.options.title)}')`;
31
32
  }
32
- const propertiesSelect = args.options.properties ? `?$select=${encodeURIComponent(args.options.properties)}` : ``;
33
+ let propertiesSelect = args.options.properties ? `?$select=${encodeURIComponent(args.options.properties)}` : ``;
34
+ propertiesSelect += args.options.withPermissions ? `${args.options.properties ? '&' : '?'}$expand=HasUniqueRoleAssignments,RoleAssignments/Member,RoleAssignments/RoleDefinitionBindings` : ``;
33
35
  const requestOptions = {
34
36
  url: requestUrl + propertiesSelect,
35
- method: 'GET',
36
37
  headers: {
37
38
  'accept': 'application/json;odata=nometadata'
38
39
  },
@@ -58,6 +59,9 @@ class SpoListGetCommand extends SpoCommand_1.default {
58
59
  },
59
60
  {
60
61
  option: '-p, --properties [properties]'
62
+ },
63
+ {
64
+ option: '--withPermissions'
61
65
  }
62
66
  ];
63
67
  const parentOptions = super.options();
@@ -1,10 +1,12 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.Page = void 0;
3
+ exports.Page = exports.supportedPromoteAs = exports.supportedPageLayouts = void 0;
4
4
  const request_1 = require("../../../../request");
5
5
  const Utils_1 = require("../../../../Utils");
6
6
  const clientsidepages_1 = require("./clientsidepages");
7
7
  const pageMethods_1 = require("./pageMethods");
8
+ exports.supportedPageLayouts = ['Article', 'Home', 'SingleWebPartAppPage', 'RepostPage', 'HeaderlessSearchResults', 'Spaces', 'Topic'];
9
+ exports.supportedPromoteAs = ['HomePage', 'NewsPage', 'Template'];
8
10
  class Page {
9
11
  static getPage(name, webUrl, logger, debug, verbose) {
10
12
  return new Promise((resolve, reject) => {
@@ -284,11 +284,11 @@ class SpoPageAddCommand extends SpoCommand_1.default {
284
284
  },
285
285
  {
286
286
  option: '-l, --layoutType [layoutType]',
287
- autocomplete: ['Article', 'Home']
287
+ autocomplete: Page_1.supportedPageLayouts
288
288
  },
289
289
  {
290
290
  option: '-p, --promoteAs [promoteAs]',
291
- autocomplete: ['HomePage', 'NewsPage', 'Template']
291
+ autocomplete: Page_1.supportedPromoteAs
292
292
  },
293
293
  {
294
294
  option: '--commentsEnabled'
@@ -312,20 +312,17 @@ class SpoPageAddCommand extends SpoCommand_1.default {
312
312
  return isValidSharePointUrl;
313
313
  }
314
314
  if (args.options.layoutType &&
315
- args.options.layoutType !== 'Article' &&
316
- args.options.layoutType !== 'Home') {
317
- return `${args.options.layoutType} is not a valid option for layoutType. Allowed values Article|Home`;
315
+ Page_1.supportedPageLayouts.indexOf(args.options.layoutType) < 0) {
316
+ return `${args.options.layoutType} is not a valid option for layoutType. Allowed values ${Page_1.supportedPageLayouts.join(', ')}`;
318
317
  }
319
318
  if (args.options.promoteAs &&
320
- args.options.promoteAs !== 'HomePage' &&
321
- args.options.promoteAs !== 'NewsPage' &&
322
- args.options.promoteAs !== 'Template') {
323
- return `${args.options.promoteAs} is not a valid option for promoteAs. Allowed values HomePage|NewsPage|Template`;
319
+ Page_1.supportedPromoteAs.indexOf(args.options.promoteAs) < 0) {
320
+ return `${args.options.promoteAs} is not a valid option for promoteAs. Allowed values ${Page_1.supportedPromoteAs.join(', ')}`;
324
321
  }
325
322
  if (args.options.promoteAs === 'HomePage' && args.options.layoutType !== 'Home') {
326
323
  return 'You can only promote home pages as site home page';
327
324
  }
328
- if (args.options.promoteAs === 'NewsPage' && args.options.layoutType === 'Home') {
325
+ if (args.options.promoteAs === 'NewsPage' && args.options.layoutType && args.options.layoutType !== 'Article') {
329
326
  return 'You can only promote article pages as news article';
330
327
  }
331
328
  return true;
@@ -283,11 +283,11 @@ class SpoPageSetCommand extends SpoCommand_1.default {
283
283
  },
284
284
  {
285
285
  option: '-l, --layoutType [layoutType]',
286
- autocomplete: ['Article', 'Home']
286
+ autocomplete: Page_1.supportedPageLayouts
287
287
  },
288
288
  {
289
289
  option: '-p, --promoteAs [promoteAs]',
290
- autocomplete: ['HomePage', 'NewsPage', 'Template']
290
+ autocomplete: Page_1.supportedPromoteAs
291
291
  },
292
292
  {
293
293
  option: '--commentsEnabled [commentsEnabled]',
@@ -315,20 +315,17 @@ class SpoPageSetCommand extends SpoCommand_1.default {
315
315
  return isValidSharePointUrl;
316
316
  }
317
317
  if (args.options.layoutType &&
318
- args.options.layoutType !== 'Article' &&
319
- args.options.layoutType !== 'Home') {
320
- return `${args.options.layoutType} is not a valid option for layoutType. Allowed values Article|Home`;
318
+ Page_1.supportedPageLayouts.indexOf(args.options.layoutType) < 0) {
319
+ return `${args.options.layoutType} is not a valid option for layoutType. Allowed values ${Page_1.supportedPageLayouts.join(', ')}`;
321
320
  }
322
321
  if (args.options.promoteAs &&
323
- args.options.promoteAs !== 'HomePage' &&
324
- args.options.promoteAs !== 'NewsPage' &&
325
- args.options.promoteAs !== 'Template') {
326
- return `${args.options.promoteAs} is not a valid option for promoteAs. Allowed values HomePage|NewsPage|Template`;
322
+ Page_1.supportedPromoteAs.indexOf(args.options.promoteAs) < 0) {
323
+ return `${args.options.promoteAs} is not a valid option for promoteAs. Allowed values ${Page_1.supportedPromoteAs.join(', ')}`;
327
324
  }
328
325
  if (args.options.promoteAs === 'HomePage' && args.options.layoutType !== 'Home') {
329
326
  return 'You can only promote home pages as site home page';
330
327
  }
331
- if (args.options.promoteAs === 'NewsPage' && args.options.layoutType === 'Home') {
328
+ if (args.options.promoteAs === 'NewsPage' && args.options.layoutType && args.options.layoutType !== 'Article') {
332
329
  return 'You can only promote article pages as news article';
333
330
  }
334
331
  if (typeof args.options.commentsEnabled !== 'undefined' &&
@@ -59,7 +59,7 @@ class SpoSiteEnsureCommand extends SpoCommand_1.default {
59
59
  if (this.debug) {
60
60
  logger.logToStderr(err.stderr);
61
61
  }
62
- if (err.error.message !== 'Request failed with status code 404') {
62
+ if (err.error.message !== '404 FILE NOT FOUND') {
63
63
  return Promise.reject(err);
64
64
  }
65
65
  if (this.verbose) {
@@ -0,0 +1,76 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const request_1 = require("../../../../request");
4
+ const SpoCommand_1 = require("../../../base/SpoCommand");
5
+ const commands_1 = require("../../commands");
6
+ class SpoSiteRecycleBinItemListCommand extends SpoCommand_1.default {
7
+ get name() {
8
+ return commands_1.default.SITE_RECYCLEBINITEM_LIST;
9
+ }
10
+ get description() {
11
+ return 'Lists items from recycle bin';
12
+ }
13
+ defaultProperties() {
14
+ return ['Id', 'Title', 'DirName'];
15
+ }
16
+ commandAction(logger, args, cb) {
17
+ if (this.verbose) {
18
+ logger.logToStderr(`Retrieving all items from recycle bin at ${args.options.siteUrl}...`);
19
+ }
20
+ const state = args.options.secondary ? '2' : '1';
21
+ let requestUrl = `${args.options.siteUrl}/_api/site/RecycleBin?$filter=(ItemState eq ${state})`;
22
+ if (typeof args.options.type !== 'undefined') {
23
+ const type = SpoSiteRecycleBinItemListCommand.recycleBinItemType.find(item => item.value === args.options.type);
24
+ if (typeof type !== 'undefined') {
25
+ requestUrl += ` and (ItemType eq ${type.id})`;
26
+ }
27
+ }
28
+ const requestOptions = {
29
+ url: requestUrl,
30
+ headers: {
31
+ 'accept': 'application/json;odata=nometadata'
32
+ },
33
+ responseType: 'json'
34
+ };
35
+ request_1.default
36
+ .get(requestOptions)
37
+ .then((response) => {
38
+ logger.log(response.value);
39
+ cb();
40
+ }, (err) => this.handleRejectedODataJsonPromise(err, logger, cb));
41
+ }
42
+ options() {
43
+ const options = [
44
+ {
45
+ option: '-u, --siteUrl <siteUrl>'
46
+ },
47
+ {
48
+ option: '-t, --type [type]',
49
+ autocomplete: SpoSiteRecycleBinItemListCommand.recycleBinItemType.map(item => item.value)
50
+ },
51
+ {
52
+ option: '-s, --secondary'
53
+ }
54
+ ];
55
+ const parentOptions = super.options();
56
+ return options.concat(parentOptions);
57
+ }
58
+ validate(args) {
59
+ const isValidSharePointUrl = SpoCommand_1.default.isValidSharePointUrl(args.options.siteUrl);
60
+ if (isValidSharePointUrl !== true) {
61
+ return isValidSharePointUrl;
62
+ }
63
+ if (typeof args.options.type !== 'undefined' &&
64
+ !SpoSiteRecycleBinItemListCommand.recycleBinItemType.some(item => item.value === args.options.type)) {
65
+ return `${args.options.type} is not a valid value. Allowed values are ${SpoSiteRecycleBinItemListCommand.recycleBinItemType.map(item => item.value).join(', ')}`;
66
+ }
67
+ return true;
68
+ }
69
+ }
70
+ SpoSiteRecycleBinItemListCommand.recycleBinItemType = [
71
+ { id: 1, value: 'files' },
72
+ { id: 3, value: 'listItems' },
73
+ { id: 5, value: 'folders' }
74
+ ];
75
+ module.exports = new SpoSiteRecycleBinItemListCommand();
76
+ //# sourceMappingURL=site-recyclebinitem-list.js.map
@@ -25,26 +25,65 @@ class SpoSiteRemoveCommand extends SpoCommand_1.default {
25
25
  commandAction(logger, args, cb) {
26
26
  const removeSite = () => {
27
27
  this.dots = '';
28
- this
29
- .getSiteGroupId(args.options.url, logger)
30
- .then((_groupId) => {
31
- if (_groupId === '00000000-0000-0000-0000-000000000000') {
32
- if (this.debug) {
33
- logger.logToStderr('Site is not groupified. Going ahead with the conventional site deletion options');
34
- }
35
- return this.deleteSiteWithoutGroup(logger, args);
36
- }
37
- else {
38
- if (this.debug) {
39
- logger.logToStderr(`Site attached to group ${_groupId}. Initiating group delete operation via Graph API`);
28
+ if (args.options.fromRecycleBin) {
29
+ this
30
+ .deleteSiteWithoutGroup(logger, args)
31
+ .then(_ => cb(), (err) => this.handleRejectedPromise(err, logger, cb));
32
+ }
33
+ else {
34
+ this
35
+ .getSiteGroupId(args.options.url, logger)
36
+ .then((groupId) => {
37
+ if (groupId === '00000000-0000-0000-0000-000000000000') {
38
+ if (this.debug) {
39
+ logger.logToStderr('Site is not groupified. Going ahead with the conventional site deletion options');
40
+ }
41
+ return this.deleteSiteWithoutGroup(logger, args);
40
42
  }
41
- if (args.options.fromRecycleBin || args.options.skipRecycleBin || args.options.wait) {
42
- logger.log(chalk.yellow(`Entered site is a groupified site. Hence, the parameters 'fromRecycleBin' or 'skipRecycleBin' or 'wait' will not be applicable.`));
43
+ else {
44
+ if (this.debug) {
45
+ logger.logToStderr(`Site attached to group ${groupId}. Initiating group delete operation via Graph API`);
46
+ }
47
+ return this
48
+ .getSiteGroup(groupId)
49
+ .then((group) => {
50
+ if (args.options.skipRecycleBin || args.options.wait) {
51
+ logger.logToStderr(chalk.yellow(`Entered site is a groupified site. Hence, the parameters 'skipRecycleBin' and 'wait' will not be applicable.`));
52
+ }
53
+ return this.deleteGroupifiedSite(group.id, logger);
54
+ })
55
+ .catch((err) => {
56
+ if (err.response.status === 404) {
57
+ if (this.verbose) {
58
+ logger.logToStderr(`Site group doesn't exist. Searching in the Microsoft 365 deleted groups.`);
59
+ }
60
+ return this
61
+ .isSiteGroupDeleted(groupId)
62
+ .then((deletedGroups) => {
63
+ if (deletedGroups.value.length === 0) {
64
+ if (this.verbose) {
65
+ logger.logToStderr("Site group doesn't exist anymore. Deleting the site.");
66
+ }
67
+ if (args.options.wait) {
68
+ logger.logToStderr(chalk.yellow(`Entered site is a groupified site. Hence, the parameter 'wait' will not be applicable.`));
69
+ }
70
+ return Promise.resolve();
71
+ }
72
+ else {
73
+ return Promise.reject(`Site group still exists in the deleted groups. The site won't be removed.`);
74
+ }
75
+ })
76
+ .then(_ => this.deleteOrphanedSite(logger, args.options.url))
77
+ .catch((err) => Promise.reject(err));
78
+ }
79
+ else {
80
+ return Promise.reject(err);
81
+ }
82
+ });
43
83
  }
44
- return this.deleteGroupifiedSite(_groupId, logger);
45
- }
46
- })
47
- .then(_ => cb(), (err) => this.handleRejectedPromise(err, logger, cb));
84
+ })
85
+ .then(_ => cb(), (err) => this.handleRejectedPromise(err, logger, cb));
86
+ }
48
87
  };
49
88
  if (args.options.confirm) {
50
89
  removeSite();
@@ -65,32 +104,64 @@ class SpoSiteRemoveCommand extends SpoCommand_1.default {
65
104
  });
66
105
  }
67
106
  }
107
+ getSiteGroup(groupId) {
108
+ const requestOptions = {
109
+ url: `https://graph.microsoft.com/v1.0/groups/${groupId}`,
110
+ headers: {
111
+ accept: 'application/json;odata.metadata=none'
112
+ },
113
+ responseType: 'json'
114
+ };
115
+ return request_1.default.get(requestOptions);
116
+ }
117
+ isSiteGroupDeleted(groupId) {
118
+ const requestOptions = {
119
+ url: `https://graph.microsoft.com/v1.0/directory/deletedItems/Microsoft.Graph.Group?$select=id&$filter=groupTypes/any(c:c+eq+'Unified') and startswith(id, '${groupId}')`,
120
+ headers: {
121
+ accept: 'application/json;odata.metadata=none'
122
+ },
123
+ responseType: 'json'
124
+ };
125
+ return request_1.default.get(requestOptions);
126
+ }
127
+ deleteOrphanedSite(logger, url) {
128
+ return this
129
+ .getSpoAdminUrl(logger, this.debug)
130
+ .then((spoAdminUrl) => {
131
+ const requestOptions = {
132
+ url: `${spoAdminUrl}/_api/GroupSiteManager/Delete?siteUrl='${url}'`,
133
+ headers: {
134
+ 'content-type': 'application/json;odata=nometadata',
135
+ accept: 'application/json;odata=nometadata'
136
+ },
137
+ responseType: 'json'
138
+ };
139
+ return request_1.default.post(requestOptions);
140
+ });
141
+ }
68
142
  deleteSiteWithoutGroup(logger, args) {
69
143
  return this
70
144
  .getSpoAdminUrl(logger, this.debug)
71
- .then((_spoAdminUrl) => {
72
- this.spoAdminUrl = _spoAdminUrl;
145
+ .then((spoAdminUrl) => {
146
+ this.spoAdminUrl = spoAdminUrl;
73
147
  return this.ensureFormDigest(this.spoAdminUrl, logger, this.context, this.debug);
74
148
  })
75
149
  .then((res) => {
76
150
  this.context = res;
77
151
  if (args.options.fromRecycleBin) {
78
152
  if (this.verbose) {
79
- logger.logToStderr(`Deleting site collection from recycle bin ${args.options.url}...`);
153
+ logger.logToStderr(`Deleting site from recycle bin ${args.options.url}...`);
80
154
  }
81
155
  return this.deleteSiteFromTheRecycleBin(args.options.url, args.options.wait, logger);
82
156
  }
83
157
  else {
84
- if (this.verbose) {
85
- logger.logToStderr(`Deleting site collection ${args.options.url}...`);
86
- }
87
158
  return this.deleteSite(args.options.url, args.options.wait, logger);
88
159
  }
89
160
  })
90
161
  .then(() => {
91
162
  if (args.options.skipRecycleBin) {
92
163
  if (this.verbose) {
93
- logger.logToStderr(`Also deleting site collection from recycle bin ${args.options.url}...`);
164
+ logger.logToStderr(`Also deleting site from tenant recycle bin ${args.options.url}...`);
94
165
  }
95
166
  return this.deleteSiteFromTheRecycleBin(args.options.url, args.options.wait, logger);
96
167
  }
@@ -106,7 +177,7 @@ class SpoSiteRemoveCommand extends SpoCommand_1.default {
106
177
  .then((res) => {
107
178
  this.context = res;
108
179
  if (this.verbose) {
109
- logger.logToStderr(`Deleting site ${url} ...`);
180
+ logger.logToStderr(`Deleting site ${url}...`);
110
181
  }
111
182
  const requestOptions = {
112
183
  url: `${this.spoAdminUrl}/_vti_bin/client.svc/ProcessQuery`,
@@ -143,9 +214,6 @@ class SpoSiteRemoveCommand extends SpoCommand_1.default {
143
214
  .ensureFormDigest(this.spoAdminUrl, logger, this.context, this.debug)
144
215
  .then((res) => {
145
216
  this.context = res;
146
- if (this.verbose) {
147
- logger.logToStderr(`Deleting site ${url} from the recycle bin...`);
148
- }
149
217
  const requestOptions = {
150
218
  url: `${this.spoAdminUrl}/_vti_bin/client.svc/ProcessQuery`,
151
219
  headers: {
@@ -185,7 +253,7 @@ class SpoSiteRemoveCommand extends SpoCommand_1.default {
185
253
  .then((res) => {
186
254
  this.context = res;
187
255
  if (this.verbose) {
188
- logger.logToStderr(`Retrieving the GroupId of the site ${url}`);
256
+ logger.logToStderr(`Retrieving the group Id of the site ${url}`);
189
257
  }
190
258
  const requestOptions = {
191
259
  url: `${this.spoAdminUrl}/_vti_bin/client.svc/ProcessQuery`,
@@ -0,0 +1,48 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const request_1 = require("../../../../request");
4
+ const SpoCommand_1 = require("../../../base/SpoCommand");
5
+ const commands_1 = require("../../commands");
6
+ class SpoWebInstalledLanguageListCommand extends SpoCommand_1.default {
7
+ get name() {
8
+ return commands_1.default.WEB_INSTALLEDLANGUAGE_LIST;
9
+ }
10
+ get description() {
11
+ return 'Lists all installed languages on site';
12
+ }
13
+ defaultProperties() {
14
+ return ['DisplayName', 'LanguageTag', 'Lcid'];
15
+ }
16
+ commandAction(logger, args, cb) {
17
+ if (this.verbose) {
18
+ logger.logToStderr(`Retrieving all installed languages on site ${args.options.webUrl}...`);
19
+ }
20
+ const requestOptions = {
21
+ url: `${args.options.webUrl}/_api/web/RegionalSettings/InstalledLanguages`,
22
+ headers: {
23
+ 'accept': 'application/json;odata=nometadata'
24
+ },
25
+ responseType: 'json'
26
+ };
27
+ request_1.default
28
+ .get(requestOptions)
29
+ .then((webInstalledLanguageProperties) => {
30
+ logger.log(webInstalledLanguageProperties.Items);
31
+ cb();
32
+ }, (err) => this.handleRejectedODataJsonPromise(err, logger, cb));
33
+ }
34
+ options() {
35
+ const options = [
36
+ {
37
+ option: '-u, --webUrl <webUrl>'
38
+ }
39
+ ];
40
+ const parentOptions = super.options();
41
+ return options.concat(parentOptions);
42
+ }
43
+ validate(args) {
44
+ return SpoCommand_1.default.isValidSharePointUrl(args.options.webUrl);
45
+ }
46
+ }
47
+ module.exports = new SpoWebInstalledLanguageListCommand();
48
+ //# sourceMappingURL=web-installedlanguage-list.js.map
@@ -22,9 +22,10 @@ exports.default = {
22
22
  CDN_POLICY_SET: `${prefix} cdn policy set`,
23
23
  CDN_SET: `${prefix} cdn set`,
24
24
  CONTENTTYPE_ADD: `${prefix} contenttype add`,
25
- CONTENTTYPE_GET: `${prefix} contenttype get`,
26
25
  CONTENTTYPE_FIELD_REMOVE: `${prefix} contenttype field remove`,
27
26
  CONTENTTYPE_FIELD_SET: `${prefix} contenttype field set`,
27
+ CONTENTTYPE_GET: `${prefix} contenttype get`,
28
+ CONTENTTYPE_LIST: `${prefix} contenttype list`,
28
29
  CONTENTTYPE_REMOVE: `${prefix} contenttype remove`,
29
30
  CONTENTTYPEHUB_GET: `${prefix} contenttypehub get`,
30
31
  CUSTOMACTION_ADD: `${prefix} customaction add`,
@@ -182,6 +183,7 @@ exports.default = {
182
183
  SITE_GROUPIFY: `${prefix} site groupify`,
183
184
  SITE_LIST: `${prefix} site list`,
184
185
  SITE_INPLACERECORDSMANAGEMENT_SET: `${prefix} site inplacerecordsmanagement set`,
186
+ SITE_RECYCLEBINITEM_LIST: `${prefix} site recyclebinitem list`,
185
187
  SITE_REMOVE: `${prefix} site remove`,
186
188
  SITE_RENAME: `${prefix} site rename`,
187
189
  SITE_SET: `${prefix} site set`,
@@ -245,6 +247,7 @@ exports.default = {
245
247
  WEB_ADD: `${prefix} web add`,
246
248
  WEB_CLIENTSIDEWEBPART_LIST: `${prefix} web clientsidewebpart list`,
247
249
  WEB_GET: `${prefix} web get`,
250
+ WEB_INSTALLEDLANGUAGE_LIST: `${prefix} web installedlanguage list`,
248
251
  WEB_LIST: `${prefix} web list`,
249
252
  WEB_REINDEX: `${prefix} web reindex`,
250
253
  WEB_REMOVE: `${prefix} web remove`,
@@ -25,24 +25,27 @@ class TeamsAppListCommand extends GraphItemsListCommand_1.GraphItemsListCommand
25
25
  if (args.options.teamId) {
26
26
  return Promise.resolve(args.options.teamId);
27
27
  }
28
- const teamRequestOptions = {
29
- url: `${this.resource}/v1.0/me/joinedTeams?$filter=displayName eq '${encodeURIComponent(args.options.teamName)}'`,
28
+ const requestOptions = {
29
+ url: `${this.resource}/v1.0/groups?$filter=displayName eq '${encodeURIComponent(args.options.teamName)}'`,
30
30
  headers: {
31
31
  accept: 'application/json;odata.metadata=none'
32
32
  },
33
33
  responseType: 'json'
34
34
  };
35
35
  return request_1.default
36
- .get(teamRequestOptions)
36
+ .get(requestOptions)
37
37
  .then(response => {
38
- const teamItem = response.value[0];
39
- if (!teamItem) {
38
+ const groupItem = response.value[0];
39
+ if (!groupItem) {
40
+ return Promise.reject(`The specified team does not exist in the Microsoft Teams`);
41
+ }
42
+ if (groupItem.resourceProvisioningOptions.indexOf('Team') === -1) {
40
43
  return Promise.reject(`The specified team does not exist in the Microsoft Teams`);
41
44
  }
42
45
  if (response.value.length > 1) {
43
46
  return Promise.reject(`Multiple Microsoft Teams teams with name ${args.options.teamName} found: ${response.value.map(x => x.id)}`);
44
47
  }
45
- return Promise.resolve(teamItem.id);
48
+ return Promise.resolve(groupItem.id);
46
49
  });
47
50
  }
48
51
  getEndpointUrl(args) {