@pnp/cli-microsoft365 10.3.0-beta.ea113b7 → 10.3.0
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/.eslintrc.cjs +1 -0
- package/allCommands.json +1 -1
- package/allCommandsFull.json +1 -1
- package/dist/config.js +2 -1
- package/dist/m365/entra/commands/app/app-add.js +3 -0
- package/dist/m365/entra/commands/roledefinition/roledefinition-add.js +58 -0
- package/dist/m365/entra/commands/roledefinition/roledefinition-set.js +84 -0
- package/dist/m365/entra/commands/rolepermission/rolepermission-list.js +42 -0
- package/dist/m365/entra/commands.js +3 -0
- package/dist/m365/outlook/commands/mailbox/mailbox-settings-set.js +163 -0
- package/dist/m365/outlook/commands.js +1 -0
- package/dist/m365/spe/commands/container/container-activate.js +50 -0
- package/dist/m365/spe/commands.js +1 -0
- package/dist/m365/spo/commands/list/list-defaultvalue-clear.js +184 -0
- package/dist/m365/spo/commands/list/list-defaultvalue-remove.js +181 -0
- package/dist/m365/spo/commands/list/list-defaultvalue-set.js +210 -0
- package/dist/m365/spo/commands.js +3 -0
- package/dist/m365/tenant/commands/people/people-pronouns-set.js +46 -0
- package/dist/m365/tenant/commands/report/report-settings-set.js +47 -0
- package/dist/m365/tenant/commands.js +2 -0
- package/docs/docs/cmd/entra/app/app-add.mdx +1 -1
- package/docs/docs/cmd/entra/roledefinition/roledefinition-add.mdx +131 -0
- package/docs/docs/cmd/entra/roledefinition/roledefinition-set.mdx +64 -0
- package/docs/docs/cmd/entra/rolepermission/rolepermission-list.mdx +162 -0
- package/docs/docs/cmd/outlook/mailbox/mailbox-settings-set.mdx +166 -0
- package/docs/docs/cmd/spe/container/container-activate.mdx +34 -0
- package/docs/docs/cmd/spo/list/list-defaultvalue-clear.mdx +62 -0
- package/docs/docs/cmd/spo/list/list-defaultvalue-remove.mdx +62 -0
- package/docs/docs/cmd/spo/list/list-defaultvalue-set.mdx +112 -0
- package/docs/docs/cmd/tenant/people/people-pronouns-set.mdx +82 -0
- package/docs/docs/cmd/tenant/report/report-settings-set.mdx +32 -0
- package/npm-shrinkwrap.json +54 -74
- package/package.json +10 -10
|
@@ -0,0 +1,181 @@
|
|
|
1
|
+
import SpoCommand from '../../../base/SpoCommand.js';
|
|
2
|
+
import { globalOptionsZod } from '../../../../Command.js';
|
|
3
|
+
import { z } from 'zod';
|
|
4
|
+
import { zod } from '../../../../utils/zod.js';
|
|
5
|
+
import commands from '../../commands.js';
|
|
6
|
+
import { DOMParser } from '@xmldom/xmldom';
|
|
7
|
+
import { validation } from '../../../../utils/validation.js';
|
|
8
|
+
import { urlUtil } from '../../../../utils/urlUtil.js';
|
|
9
|
+
import request from '../../../../request.js';
|
|
10
|
+
import { formatting } from '../../../../utils/formatting.js';
|
|
11
|
+
import { cli } from '../../../../cli/cli.js';
|
|
12
|
+
const options = globalOptionsZod
|
|
13
|
+
.extend({
|
|
14
|
+
webUrl: zod.alias('u', z.string()
|
|
15
|
+
.refine(url => validation.isValidSharePointUrl(url) === true, url => ({
|
|
16
|
+
message: `'${url}' is not a valid SharePoint Online site URL.`
|
|
17
|
+
}))),
|
|
18
|
+
listId: zod.alias('i', z.string().optional()
|
|
19
|
+
.refine(id => id === undefined || validation.isValidGuid(id), id => ({
|
|
20
|
+
message: `'${id}' is not a valid GUID.`
|
|
21
|
+
}))),
|
|
22
|
+
listTitle: zod.alias('t', z.string().optional()),
|
|
23
|
+
listUrl: z.string().optional(),
|
|
24
|
+
fieldName: z.string(),
|
|
25
|
+
folderUrl: z.string().optional(),
|
|
26
|
+
force: zod.alias('f', z.boolean().optional())
|
|
27
|
+
})
|
|
28
|
+
.strict();
|
|
29
|
+
class SpoListDefaultValueRemoveCommand extends SpoCommand {
|
|
30
|
+
get name() {
|
|
31
|
+
return commands.LIST_DEFAULTVALUE_REMOVE;
|
|
32
|
+
}
|
|
33
|
+
get description() {
|
|
34
|
+
return 'Removes a specific default column value for a specific document library';
|
|
35
|
+
}
|
|
36
|
+
get schema() {
|
|
37
|
+
return options;
|
|
38
|
+
}
|
|
39
|
+
getRefinedSchema(schema) {
|
|
40
|
+
return schema
|
|
41
|
+
.refine(options => [options.listId, options.listTitle, options.listUrl].filter(o => o !== undefined).length === 1, {
|
|
42
|
+
message: 'Use one of the following options: listId, listTitle, listUrl.'
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
async commandAction(logger, args) {
|
|
46
|
+
if (!args.options.force) {
|
|
47
|
+
const result = await cli.promptForConfirmation({ message: `Are you sure you want to remove default column value '${args.options.fieldName}' from ${args.options.folderUrl ? `'${args.options.folderUrl}'` : 'the root of the list'}?` });
|
|
48
|
+
if (!result) {
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
try {
|
|
53
|
+
if (this.verbose) {
|
|
54
|
+
await logger.logToStderr(`Removing default column value '${args.options.fieldName}' from ${args.options.folderUrl ? `'${args.options.folderUrl}'` : 'the root of the list'}.`);
|
|
55
|
+
await logger.logToStderr(`Getting server-relative URL of the list...`);
|
|
56
|
+
}
|
|
57
|
+
const listServerRelUrl = await this.getServerRelativeListUrl(args.options);
|
|
58
|
+
let folderUrl = listServerRelUrl;
|
|
59
|
+
if (args.options.folderUrl) {
|
|
60
|
+
folderUrl = urlUtil.getServerRelativePath(args.options.webUrl, urlUtil.removeTrailingSlashes(args.options.folderUrl));
|
|
61
|
+
}
|
|
62
|
+
if (this.verbose) {
|
|
63
|
+
await logger.logToStderr(`Getting default column values...`);
|
|
64
|
+
}
|
|
65
|
+
const defaultValuesXml = await this.getDefaultColumnValuesXml(args.options.webUrl, listServerRelUrl);
|
|
66
|
+
const removeDefaultValueResult = this.removeFieldFromXml(defaultValuesXml, args.options.fieldName, folderUrl);
|
|
67
|
+
if (!removeDefaultValueResult.isFieldFound) {
|
|
68
|
+
throw `Default column value '${args.options.fieldName}' was not found.`;
|
|
69
|
+
}
|
|
70
|
+
if (this.verbose) {
|
|
71
|
+
await logger.logToStderr(`Uploading default column values to list...`);
|
|
72
|
+
}
|
|
73
|
+
await this.uploadDefaultColumnValuesXml(args.options.webUrl, listServerRelUrl, removeDefaultValueResult.xml);
|
|
74
|
+
}
|
|
75
|
+
catch (err) {
|
|
76
|
+
this.handleRejectedODataJsonPromise(err);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
async getServerRelativeListUrl(options) {
|
|
80
|
+
const requestOptions = {
|
|
81
|
+
url: `${options.webUrl}/_api/Web`,
|
|
82
|
+
headers: {
|
|
83
|
+
accept: 'application/json;odata=nometadata'
|
|
84
|
+
},
|
|
85
|
+
responseType: 'json'
|
|
86
|
+
};
|
|
87
|
+
if (options.listUrl) {
|
|
88
|
+
const serverRelativeUrl = urlUtil.getServerRelativePath(options.webUrl, options.listUrl);
|
|
89
|
+
requestOptions.url += `/GetList('${formatting.encodeQueryParameter(serverRelativeUrl)}')`;
|
|
90
|
+
}
|
|
91
|
+
else if (options.listId) {
|
|
92
|
+
requestOptions.url += `/Lists('${options.listId}')`;
|
|
93
|
+
}
|
|
94
|
+
else if (options.listTitle) {
|
|
95
|
+
requestOptions.url += `/Lists/GetByTitle('${formatting.encodeQueryParameter(options.listTitle)}')`;
|
|
96
|
+
}
|
|
97
|
+
requestOptions.url += '?$expand=RootFolder&$select=RootFolder/ServerRelativeUrl,BaseTemplate';
|
|
98
|
+
try {
|
|
99
|
+
const response = await request.get(requestOptions);
|
|
100
|
+
if (response.BaseTemplate !== 101) {
|
|
101
|
+
throw `The specified list is not a document library.`;
|
|
102
|
+
}
|
|
103
|
+
return response.RootFolder.ServerRelativeUrl;
|
|
104
|
+
}
|
|
105
|
+
catch (error) {
|
|
106
|
+
if (error.status === 404) {
|
|
107
|
+
throw `List '${options.listId || options.listTitle || options.listUrl}' was not found.`;
|
|
108
|
+
}
|
|
109
|
+
throw error;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
async getDefaultColumnValuesXml(webUrl, listServerRelUrl) {
|
|
113
|
+
try {
|
|
114
|
+
const requestOptions = {
|
|
115
|
+
url: `${webUrl}/_api/Web/GetFileByServerRelativePath(decodedUrl='${formatting.encodeQueryParameter(listServerRelUrl + '/Forms/client_LocationBasedDefaults.html')}')/$value`,
|
|
116
|
+
headers: {
|
|
117
|
+
accept: 'application/json;odata=nometadata'
|
|
118
|
+
},
|
|
119
|
+
responseType: 'json'
|
|
120
|
+
};
|
|
121
|
+
const defaultValuesXml = await request.get(requestOptions);
|
|
122
|
+
return defaultValuesXml;
|
|
123
|
+
}
|
|
124
|
+
catch (err) {
|
|
125
|
+
// For lists that have never had default column values set, the client_LocationBasedDefaults.html file does not exist.
|
|
126
|
+
if (err.status === 404) {
|
|
127
|
+
return null;
|
|
128
|
+
}
|
|
129
|
+
throw err;
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
removeFieldFromXml(xml, fieldName, folderUrl) {
|
|
133
|
+
if (xml === null) {
|
|
134
|
+
return { isFieldFound: false };
|
|
135
|
+
}
|
|
136
|
+
// Encode all spaces in the folder URL
|
|
137
|
+
const encodedFolderUrl = folderUrl.replace(/ /g, '%20');
|
|
138
|
+
const parser = new DOMParser();
|
|
139
|
+
const doc = parser.parseFromString(xml, 'application/xml');
|
|
140
|
+
const folderLinks = doc.getElementsByTagName('a');
|
|
141
|
+
for (let i = 0; i < folderLinks.length; i++) {
|
|
142
|
+
const folderNode = folderLinks[i];
|
|
143
|
+
const folderNodeUrl = folderNode.getAttribute('href');
|
|
144
|
+
if (encodedFolderUrl.toLowerCase() !== folderNodeUrl.toLowerCase()) {
|
|
145
|
+
continue;
|
|
146
|
+
}
|
|
147
|
+
const defaultValues = folderNode.getElementsByTagName('DefaultValue');
|
|
148
|
+
for (let j = 0; j < defaultValues.length; j++) {
|
|
149
|
+
const defaultValueNode = defaultValues[j];
|
|
150
|
+
const defaultValueNodeField = defaultValueNode.getAttribute('FieldName');
|
|
151
|
+
if (defaultValueNodeField !== fieldName) {
|
|
152
|
+
continue;
|
|
153
|
+
}
|
|
154
|
+
if (folderNode.childNodes.length === 1) {
|
|
155
|
+
// No other default values found in the folder, let's remove the folder node
|
|
156
|
+
folderNode.parentNode.removeChild(folderNode);
|
|
157
|
+
}
|
|
158
|
+
else {
|
|
159
|
+
// Default value node found, let's remove it
|
|
160
|
+
folderNode.removeChild(defaultValueNode);
|
|
161
|
+
}
|
|
162
|
+
return { isFieldFound: true, xml: doc.toString() };
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
return { isFieldFound: false };
|
|
166
|
+
}
|
|
167
|
+
async uploadDefaultColumnValuesXml(webUrl, listServerRelUrl, xml) {
|
|
168
|
+
const requestOptions = {
|
|
169
|
+
url: `${webUrl}/_api/Web/GetFileByServerRelativePath(decodedUrl='${formatting.encodeQueryParameter(listServerRelUrl + '/Forms/client_LocationBasedDefaults.html')}')/$value`,
|
|
170
|
+
headers: {
|
|
171
|
+
accept: 'application/json;odata=nometadata',
|
|
172
|
+
'content-type': 'text/plain'
|
|
173
|
+
},
|
|
174
|
+
responseType: 'json',
|
|
175
|
+
data: xml
|
|
176
|
+
};
|
|
177
|
+
await request.put(requestOptions);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
export default new SpoListDefaultValueRemoveCommand();
|
|
181
|
+
//# sourceMappingURL=list-defaultvalue-remove.js.map
|
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
import SpoCommand from '../../../base/SpoCommand.js';
|
|
2
|
+
import { globalOptionsZod } from '../../../../Command.js';
|
|
3
|
+
import { z } from 'zod';
|
|
4
|
+
import { zod } from '../../../../utils/zod.js';
|
|
5
|
+
import commands from '../../commands.js';
|
|
6
|
+
import { DOMParser } from '@xmldom/xmldom';
|
|
7
|
+
import { validation } from '../../../../utils/validation.js';
|
|
8
|
+
import { urlUtil } from '../../../../utils/urlUtil.js';
|
|
9
|
+
import request from '../../../../request.js';
|
|
10
|
+
import { formatting } from '../../../../utils/formatting.js';
|
|
11
|
+
const options = globalOptionsZod
|
|
12
|
+
.extend({
|
|
13
|
+
webUrl: zod.alias('u', z.string()
|
|
14
|
+
.refine(url => validation.isValidSharePointUrl(url) === true, url => ({
|
|
15
|
+
message: `'${url}' is not a valid SharePoint Online site URL.`
|
|
16
|
+
}))),
|
|
17
|
+
listId: zod.alias('i', z.string().optional()
|
|
18
|
+
.refine(id => id === undefined || validation.isValidGuid(id), id => ({
|
|
19
|
+
message: `'${id}' is not a valid GUID.`
|
|
20
|
+
}))),
|
|
21
|
+
listTitle: zod.alias('t', z.string().optional()),
|
|
22
|
+
listUrl: z.string().optional(),
|
|
23
|
+
fieldName: z.string(),
|
|
24
|
+
fieldValue: z.string()
|
|
25
|
+
.refine(value => value !== '', `The value cannot be empty. Use 'spo list defaultvalue remove' to remove a default column value.`),
|
|
26
|
+
folderUrl: z.string().optional()
|
|
27
|
+
.refine(url => url === undefined || (!url.includes('#') && !url.includes('%')), 'Due to limitations in SharePoint Online, setting default column values for folders with a # or % character in their path is not supported.')
|
|
28
|
+
})
|
|
29
|
+
.strict();
|
|
30
|
+
class SpoListDefaultValueSetCommand extends SpoCommand {
|
|
31
|
+
get name() {
|
|
32
|
+
return commands.LIST_DEFAULTVALUE_SET;
|
|
33
|
+
}
|
|
34
|
+
get description() {
|
|
35
|
+
return 'Sets default column values for a specific document library';
|
|
36
|
+
}
|
|
37
|
+
get schema() {
|
|
38
|
+
return options;
|
|
39
|
+
}
|
|
40
|
+
getRefinedSchema(schema) {
|
|
41
|
+
return schema
|
|
42
|
+
.refine(options => [options.listId, options.listTitle, options.listUrl].filter(o => o !== undefined).length === 1, {
|
|
43
|
+
message: 'Use one of the following options: listId, listTitle, listUrl.'
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
async commandAction(logger, args) {
|
|
47
|
+
try {
|
|
48
|
+
if (this.verbose) {
|
|
49
|
+
await logger.logToStderr(`Setting default column value '${args.options.fieldValue}' for field '${args.options.fieldName}'...`);
|
|
50
|
+
await logger.logToStderr(`Getting server-relative URL of the list...`);
|
|
51
|
+
}
|
|
52
|
+
const listServerRelUrl = await this.getServerRelativeListUrl(args.options);
|
|
53
|
+
let folderUrl = listServerRelUrl;
|
|
54
|
+
if (args.options.folderUrl) {
|
|
55
|
+
if (this.verbose) {
|
|
56
|
+
await logger.logToStderr(`Getting server-relative URL of folder '${args.options.folderUrl}'...`);
|
|
57
|
+
}
|
|
58
|
+
// Casing of the folder URL is important, let's retrieve the correct URL
|
|
59
|
+
const serverRelativeFolderUrl = urlUtil.getServerRelativePath(args.options.webUrl, urlUtil.removeTrailingSlashes(args.options.folderUrl));
|
|
60
|
+
folderUrl = await this.getCorrectFolderUrl(args.options.webUrl, serverRelativeFolderUrl);
|
|
61
|
+
}
|
|
62
|
+
if (this.verbose) {
|
|
63
|
+
await logger.logToStderr(`Getting default column values...`);
|
|
64
|
+
}
|
|
65
|
+
const defaultValuesXml = await this.ensureDefaultColumnValuesXml(args.options.webUrl, listServerRelUrl);
|
|
66
|
+
const modifiedXml = await this.updateFieldValueXml(logger, defaultValuesXml, args.options.fieldName, args.options.fieldValue, folderUrl);
|
|
67
|
+
await this.uploadDefaultColumnValuesXml(logger, args.options.webUrl, listServerRelUrl, modifiedXml);
|
|
68
|
+
}
|
|
69
|
+
catch (err) {
|
|
70
|
+
this.handleRejectedODataJsonPromise(err);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
async getServerRelativeListUrl(options) {
|
|
74
|
+
const requestOptions = {
|
|
75
|
+
url: `${options.webUrl}/_api/Web`,
|
|
76
|
+
headers: {
|
|
77
|
+
accept: 'application/json;odata=nometadata'
|
|
78
|
+
},
|
|
79
|
+
responseType: 'json'
|
|
80
|
+
};
|
|
81
|
+
if (options.listUrl) {
|
|
82
|
+
const serverRelativeUrl = urlUtil.getServerRelativePath(options.webUrl, options.listUrl);
|
|
83
|
+
requestOptions.url += `/GetList('${formatting.encodeQueryParameter(serverRelativeUrl)}')`;
|
|
84
|
+
}
|
|
85
|
+
else if (options.listId) {
|
|
86
|
+
requestOptions.url += `/Lists('${options.listId}')`;
|
|
87
|
+
}
|
|
88
|
+
else if (options.listTitle) {
|
|
89
|
+
requestOptions.url += `/Lists/GetByTitle('${formatting.encodeQueryParameter(options.listTitle)}')`;
|
|
90
|
+
}
|
|
91
|
+
requestOptions.url += '?$expand=RootFolder&$select=RootFolder/ServerRelativeUrl,BaseTemplate';
|
|
92
|
+
try {
|
|
93
|
+
const response = await request.get(requestOptions);
|
|
94
|
+
if (response.BaseTemplate !== 101) {
|
|
95
|
+
throw `The specified list is not a document library.`;
|
|
96
|
+
}
|
|
97
|
+
return response.RootFolder.ServerRelativeUrl;
|
|
98
|
+
}
|
|
99
|
+
catch (error) {
|
|
100
|
+
if (error.status === 404) {
|
|
101
|
+
throw `List '${options.listId || options.listTitle || options.listUrl}' was not found.`;
|
|
102
|
+
}
|
|
103
|
+
throw error;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
async getCorrectFolderUrl(webUrl, folderUrl) {
|
|
107
|
+
const requestOptions = {
|
|
108
|
+
// Using ListItemAllFields endpoint because GetFolderByServerRelativePath doesn't return the correctly cased URL
|
|
109
|
+
url: `${webUrl}/_api/Web/GetFolderByServerRelativePath(decodedUrl='${formatting.encodeQueryParameter(folderUrl)}')/ListItemAllFields?$select=FileRef`,
|
|
110
|
+
headers: {
|
|
111
|
+
accept: 'application/json;odata=nometadata'
|
|
112
|
+
},
|
|
113
|
+
responseType: 'json'
|
|
114
|
+
};
|
|
115
|
+
const response = await request.get(requestOptions);
|
|
116
|
+
if (!response.FileRef) {
|
|
117
|
+
throw `Folder '${folderUrl}' was not found.`;
|
|
118
|
+
}
|
|
119
|
+
return response.FileRef;
|
|
120
|
+
}
|
|
121
|
+
async ensureDefaultColumnValuesXml(webUrl, listServerRelUrl) {
|
|
122
|
+
try {
|
|
123
|
+
const requestOptions = {
|
|
124
|
+
url: `${webUrl}/_api/Web/GetFileByServerRelativePath(decodedUrl='${formatting.encodeQueryParameter(listServerRelUrl + '/Forms/client_LocationBasedDefaults.html')}')/$value`,
|
|
125
|
+
headers: {
|
|
126
|
+
accept: 'application/json;odata=nometadata'
|
|
127
|
+
},
|
|
128
|
+
responseType: 'json'
|
|
129
|
+
};
|
|
130
|
+
const defaultValuesXml = await request.get(requestOptions);
|
|
131
|
+
return defaultValuesXml;
|
|
132
|
+
}
|
|
133
|
+
catch (err) {
|
|
134
|
+
if (err.status !== 404) {
|
|
135
|
+
throw err;
|
|
136
|
+
}
|
|
137
|
+
// For lists that have never had default column values set, the client_LocationBasedDefaults.html file does not exist.
|
|
138
|
+
// In this case, we need to create the file with blank default metadata.
|
|
139
|
+
const requestOptions = {
|
|
140
|
+
url: `${webUrl}/_api/Web/GetFolderByServerRelativePath(decodedUrl='${formatting.encodeQueryParameter(listServerRelUrl + '/Forms')}')/Files/Add(url='client_LocationBasedDefaults.html', overwrite=false)`,
|
|
141
|
+
headers: {
|
|
142
|
+
accept: 'application/json;odata=nometadata',
|
|
143
|
+
'content-type': 'text/plain'
|
|
144
|
+
},
|
|
145
|
+
responseType: 'json',
|
|
146
|
+
data: '<MetadataDefaults />'
|
|
147
|
+
};
|
|
148
|
+
await request.post(requestOptions);
|
|
149
|
+
return requestOptions.data;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
async updateFieldValueXml(logger, xml, fieldName, fieldValue, folderUrl) {
|
|
153
|
+
if (this.verbose) {
|
|
154
|
+
await logger.logToStderr(`Modifying default column values...`);
|
|
155
|
+
}
|
|
156
|
+
// Encode all spaces in the folder URL
|
|
157
|
+
const encodedFolderUrl = folderUrl.replace(/ /g, '%20');
|
|
158
|
+
const parser = new DOMParser();
|
|
159
|
+
const doc = parser.parseFromString(xml, 'application/xml');
|
|
160
|
+
// Create a new DefaultValue node
|
|
161
|
+
const newDefaultValueNode = doc.createElement('DefaultValue');
|
|
162
|
+
newDefaultValueNode.setAttribute('FieldName', fieldName);
|
|
163
|
+
newDefaultValueNode.textContent = fieldValue;
|
|
164
|
+
const folderLinks = doc.getElementsByTagName('a');
|
|
165
|
+
for (let i = 0; i < folderLinks.length; i++) {
|
|
166
|
+
const folderNode = folderLinks[i];
|
|
167
|
+
const folderNodeUrl = folderNode.getAttribute('href');
|
|
168
|
+
if (encodedFolderUrl !== folderNodeUrl) {
|
|
169
|
+
continue;
|
|
170
|
+
}
|
|
171
|
+
const defaultValues = folderNode.getElementsByTagName('DefaultValue');
|
|
172
|
+
for (let j = 0; j < defaultValues.length; j++) {
|
|
173
|
+
const defaultValueNode = defaultValues[j];
|
|
174
|
+
const defaultValueNodeField = defaultValueNode.getAttribute('FieldName');
|
|
175
|
+
if (defaultValueNodeField !== fieldName) {
|
|
176
|
+
continue;
|
|
177
|
+
}
|
|
178
|
+
// Default value node found, let's update the value
|
|
179
|
+
defaultValueNode.textContent = fieldValue;
|
|
180
|
+
return doc.toString();
|
|
181
|
+
}
|
|
182
|
+
// Default value node not found, let's create it
|
|
183
|
+
folderNode.appendChild(newDefaultValueNode);
|
|
184
|
+
return doc.toString();
|
|
185
|
+
}
|
|
186
|
+
// Folder node was not found, let's create it
|
|
187
|
+
const newFolderNode = doc.createElement('a');
|
|
188
|
+
newFolderNode.setAttribute('href', encodedFolderUrl);
|
|
189
|
+
newFolderNode.appendChild(newDefaultValueNode);
|
|
190
|
+
doc.documentElement.appendChild(newFolderNode);
|
|
191
|
+
return doc.toString();
|
|
192
|
+
}
|
|
193
|
+
async uploadDefaultColumnValuesXml(logger, webUrl, listServerRelUrl, xml) {
|
|
194
|
+
if (this.verbose) {
|
|
195
|
+
await logger.logToStderr(`Uploading default column values to list...`);
|
|
196
|
+
}
|
|
197
|
+
const requestOptions = {
|
|
198
|
+
url: `${webUrl}/_api/Web/GetFileByServerRelativePath(decodedUrl='${formatting.encodeQueryParameter(listServerRelUrl + '/Forms/client_LocationBasedDefaults.html')}')/$value`,
|
|
199
|
+
headers: {
|
|
200
|
+
accept: 'application/json;odata=nometadata',
|
|
201
|
+
'content-type': 'text/plain'
|
|
202
|
+
},
|
|
203
|
+
responseType: 'json',
|
|
204
|
+
data: xml
|
|
205
|
+
};
|
|
206
|
+
await request.put(requestOptions);
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
export default new SpoListDefaultValueSetCommand();
|
|
210
|
+
//# sourceMappingURL=list-defaultvalue-set.js.map
|
|
@@ -137,7 +137,10 @@ export default {
|
|
|
137
137
|
LIST_CONTENTTYPE_LIST: `${prefix} list contenttype list`,
|
|
138
138
|
LIST_CONTENTTYPE_REMOVE: `${prefix} list contenttype remove`,
|
|
139
139
|
LIST_CONTENTTYPE_DEFAULT_SET: `${prefix} list contenttype default set`,
|
|
140
|
+
LIST_DEFAULTVALUE_CLEAR: `${prefix} list defaultvalue clear`,
|
|
140
141
|
LIST_DEFAULTVALUE_LIST: `${prefix} list defaultvalue list`,
|
|
142
|
+
LIST_DEFAULTVALUE_REMOVE: `${prefix} list defaultvalue remove`,
|
|
143
|
+
LIST_DEFAULTVALUE_SET: `${prefix} list defaultvalue set`,
|
|
141
144
|
LIST_GET: `${prefix} list get`,
|
|
142
145
|
LIST_LIST: `${prefix} list list`,
|
|
143
146
|
LIST_REMOVE: `${prefix} list remove`,
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
import { globalOptionsZod } from '../../../../Command.js';
|
|
3
|
+
import request from '../../../../request.js';
|
|
4
|
+
import { zod } from '../../../../utils/zod.js';
|
|
5
|
+
import GraphCommand from '../../../base/GraphCommand.js';
|
|
6
|
+
import commands from '../../commands.js';
|
|
7
|
+
const options = globalOptionsZod
|
|
8
|
+
.extend({
|
|
9
|
+
enabled: zod.alias('e', z.boolean())
|
|
10
|
+
})
|
|
11
|
+
.strict();
|
|
12
|
+
class TenantPeoplePronounsSetCommand extends GraphCommand {
|
|
13
|
+
get name() {
|
|
14
|
+
return commands.PEOPLE_PRONOUNS_SET;
|
|
15
|
+
}
|
|
16
|
+
get description() {
|
|
17
|
+
return 'Manage pronouns settings for an organization';
|
|
18
|
+
}
|
|
19
|
+
get schema() {
|
|
20
|
+
return options;
|
|
21
|
+
}
|
|
22
|
+
async commandAction(logger, args) {
|
|
23
|
+
try {
|
|
24
|
+
if (this.verbose) {
|
|
25
|
+
await logger.logToStderr('Updating pronouns settings...');
|
|
26
|
+
}
|
|
27
|
+
const requestOptions = {
|
|
28
|
+
url: `${this.resource}/v1.0/admin/people/pronouns`,
|
|
29
|
+
headers: {
|
|
30
|
+
accept: 'application/json;odata.metadata=none'
|
|
31
|
+
},
|
|
32
|
+
data: {
|
|
33
|
+
isEnabledInOrganization: args.options.enabled
|
|
34
|
+
},
|
|
35
|
+
responseType: 'json'
|
|
36
|
+
};
|
|
37
|
+
const pronouns = await request.patch(requestOptions);
|
|
38
|
+
await logger.log(pronouns);
|
|
39
|
+
}
|
|
40
|
+
catch (err) {
|
|
41
|
+
this.handleRejectedODataJsonPromise(err);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
export default new TenantPeoplePronounsSetCommand();
|
|
46
|
+
//# sourceMappingURL=people-pronouns-set.js.map
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import request from '../../../../request.js';
|
|
2
|
+
import GraphCommand from '../../../base/GraphCommand.js';
|
|
3
|
+
import commands from '../../commands.js';
|
|
4
|
+
import { z } from 'zod';
|
|
5
|
+
import { globalOptionsZod } from '../../../../Command.js';
|
|
6
|
+
import { zod } from '../../../../utils/zod.js';
|
|
7
|
+
const options = globalOptionsZod
|
|
8
|
+
.extend({
|
|
9
|
+
displayConcealedNames: zod.alias('d', z.boolean())
|
|
10
|
+
})
|
|
11
|
+
.strict();
|
|
12
|
+
class TenantReportSettingsSetCommand extends GraphCommand {
|
|
13
|
+
get name() {
|
|
14
|
+
return commands.REPORT_SETTINGS_SET;
|
|
15
|
+
}
|
|
16
|
+
get description() {
|
|
17
|
+
return 'Update tenant-level settings for Microsoft 365 reports';
|
|
18
|
+
}
|
|
19
|
+
get schema() {
|
|
20
|
+
return options;
|
|
21
|
+
}
|
|
22
|
+
async commandAction(logger, args) {
|
|
23
|
+
try {
|
|
24
|
+
const { displayConcealedNames } = args.options;
|
|
25
|
+
if (this.verbose) {
|
|
26
|
+
await logger.logToStderr(`Updating report setting displayConcealedNames to ${displayConcealedNames}`);
|
|
27
|
+
}
|
|
28
|
+
const requestOptions = {
|
|
29
|
+
url: `${this.resource}/v1.0/admin/reportSettings`,
|
|
30
|
+
headers: {
|
|
31
|
+
accept: 'application/json;odata.metadata=none',
|
|
32
|
+
'content-type': 'application/json'
|
|
33
|
+
},
|
|
34
|
+
responseType: 'json',
|
|
35
|
+
data: {
|
|
36
|
+
displayConcealedNames
|
|
37
|
+
}
|
|
38
|
+
};
|
|
39
|
+
await request.patch(requestOptions);
|
|
40
|
+
}
|
|
41
|
+
catch (err) {
|
|
42
|
+
this.handleRejectedODataJsonPromise(err);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
export default new TenantReportSettingsSetCommand();
|
|
47
|
+
//# sourceMappingURL=report-settings-set.js.map
|
|
@@ -8,12 +8,14 @@ export default {
|
|
|
8
8
|
PEOPLE_PROFILECARDPROPERTY_REMOVE: `${prefix} people profilecardproperty remove`,
|
|
9
9
|
PEOPLE_PROFILECARDPROPERTY_SET: `${prefix} people profilecardproperty set`,
|
|
10
10
|
PEOPLE_PRONOUNS_GET: `${prefix} people pronouns get`,
|
|
11
|
+
PEOPLE_PRONOUNS_SET: `${prefix} people pronouns set`,
|
|
11
12
|
REPORT_ACTIVEUSERCOUNTS: `${prefix} report activeusercounts`,
|
|
12
13
|
REPORT_ACTIVEUSERDETAIL: `${prefix} report activeuserdetail`,
|
|
13
14
|
REPORT_OFFICE365ACTIVATIONCOUNTS: `${prefix} report office365activationcounts`,
|
|
14
15
|
REPORT_OFFICE365ACTIVATIONSUSERDETAIL: `${prefix} report office365activationsuserdetail`,
|
|
15
16
|
REPORT_OFFICE365ACTIVATIONSUSERCOUNTS: `${prefix} report office365activationsusercounts`,
|
|
16
17
|
REPORT_SERVICESUSERCOUNTS: `${prefix} report servicesusercounts`,
|
|
18
|
+
REPORT_SETTINGS_SET: `${prefix} report settings set`,
|
|
17
19
|
SECURITY_ALERTS_LIST: `${prefix} security alerts list`,
|
|
18
20
|
SERVICEANNOUNCEMENT_HEALTHISSUE_GET: `${prefix} serviceannouncement healthissue get`,
|
|
19
21
|
SERVICEANNOUNCEMENT_HEALTH_GET: `${prefix} serviceannouncement health get`,
|
|
@@ -31,7 +31,7 @@ m365 entra appregistration add [options]
|
|
|
31
31
|
: Comma-separated list of redirect URIs. Requires `platform` to be specified.
|
|
32
32
|
|
|
33
33
|
`-p, --platform [platform]`
|
|
34
|
-
: Platform for which the `redirectUris` should be configured. Allowed values `spa`, `web`, `publicClient`.
|
|
34
|
+
: Platform for which the `redirectUris` should be configured. Allowed values `spa`, `web`, `publicClient`. Requires `redirectUris` to be specified.
|
|
35
35
|
|
|
36
36
|
`--implicitFlow`
|
|
37
37
|
: Specify, to indicate that the authorization endpoint should return ID and access tokens.
|
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
import Global from '/docs/cmd/_global.mdx';
|
|
2
|
+
import Tabs from '@theme/Tabs';
|
|
3
|
+
import TabItem from '@theme/TabItem';
|
|
4
|
+
|
|
5
|
+
# entra roledefinition add
|
|
6
|
+
|
|
7
|
+
Creates a custom Microsoft Entra ID role definition
|
|
8
|
+
|
|
9
|
+
## Usage
|
|
10
|
+
|
|
11
|
+
```sh
|
|
12
|
+
m365 entra roledefinition add [options]
|
|
13
|
+
```
|
|
14
|
+
|
|
15
|
+
## Options
|
|
16
|
+
|
|
17
|
+
```md definition-list
|
|
18
|
+
`-n, --displayName <displayName>`
|
|
19
|
+
: The display name for the role definition.
|
|
20
|
+
|
|
21
|
+
`-a, --allowedResourceActions <allowedResourceActions>`
|
|
22
|
+
: Comma-separated list of resource actions allowed for the role.
|
|
23
|
+
|
|
24
|
+
`-d, --description [description]`
|
|
25
|
+
: The description for the role definition.
|
|
26
|
+
|
|
27
|
+
`-e, --enabled [enabled]`
|
|
28
|
+
: Indicates if the role is enabled for the assignment. If not specified, the role is enabled by default.
|
|
29
|
+
|
|
30
|
+
`-v, --version [version]`
|
|
31
|
+
: The version of the role definition.
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
<Global />
|
|
35
|
+
|
|
36
|
+
## Remarks
|
|
37
|
+
|
|
38
|
+
Use the `m365 entra rolepermission list --resourceNamespace microsoft.directory` command to get a list of available resource actions.
|
|
39
|
+
|
|
40
|
+
## Examples
|
|
41
|
+
|
|
42
|
+
Create a custom Microsoft Entra ID role
|
|
43
|
+
|
|
44
|
+
```sh
|
|
45
|
+
m365 entra roledefinition add --displayName 'Application Remover' --description 'Allows to remove any Entra ID application' --allowedResourceActions 'microsoft.directory/applications/delete'
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
Create a custom Microsoft Entra ID role, but disable it for the assignment
|
|
49
|
+
|
|
50
|
+
```sh
|
|
51
|
+
m365 entra roledefinition add --displayName 'Application Remover' --version '1.0' --enabled false --allowedResourceActions 'microsoft.directory/applications/delete,microsoft.directory/applications/owners/update'
|
|
52
|
+
```
|
|
53
|
+
|
|
54
|
+
## Response
|
|
55
|
+
|
|
56
|
+
<Tabs>
|
|
57
|
+
<TabItem value="JSON">
|
|
58
|
+
|
|
59
|
+
```json
|
|
60
|
+
{
|
|
61
|
+
"id": "3844129d-f748-4c03-8165-4412ee9b4ceb",
|
|
62
|
+
"description": null,
|
|
63
|
+
"displayName": "Custom Role",
|
|
64
|
+
"isBuiltIn": false,
|
|
65
|
+
"isEnabled": true,
|
|
66
|
+
"resourceScopes": [
|
|
67
|
+
"/"
|
|
68
|
+
],
|
|
69
|
+
"templateId": "3844129d-f748-4c03-8165-4412ee9b4ceb",
|
|
70
|
+
"version": "1",
|
|
71
|
+
"rolePermissions": [
|
|
72
|
+
{
|
|
73
|
+
"allowedResourceActions": [
|
|
74
|
+
"microsoft.directory/groups.unified/create",
|
|
75
|
+
"microsoft.directory/groups.unified/delete"
|
|
76
|
+
],
|
|
77
|
+
"condition": null
|
|
78
|
+
}
|
|
79
|
+
]
|
|
80
|
+
}
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
</TabItem>
|
|
84
|
+
<TabItem value="Text">
|
|
85
|
+
|
|
86
|
+
```text
|
|
87
|
+
description : null
|
|
88
|
+
displayName : Custom Role
|
|
89
|
+
id : 3844129d-f748-4c03-8165-4412ee9b4ceb
|
|
90
|
+
isBuiltIn : false
|
|
91
|
+
isEnabled : true
|
|
92
|
+
resourceScopes : ["/"]
|
|
93
|
+
rolePermissions: [{"allowedResourceActions":["microsoft.directory/groups.unified/create","microsoft.directory/groups.unified/delete"],"condition":null}]
|
|
94
|
+
templateId : 3844129d-f748-4c03-8165-4412ee9b4ceb
|
|
95
|
+
version : 1
|
|
96
|
+
```
|
|
97
|
+
|
|
98
|
+
</TabItem>
|
|
99
|
+
<TabItem value="CSV">
|
|
100
|
+
|
|
101
|
+
```csv
|
|
102
|
+
id,description,displayName,isBuiltIn,isEnabled,templateId,version
|
|
103
|
+
3844129d-f748-4c03-8165-4412ee9b4ceb,,Custom Role,0,1,3844129d-f748-4c03-8165-4412ee9b4ceb,1
|
|
104
|
+
```
|
|
105
|
+
|
|
106
|
+
</TabItem>
|
|
107
|
+
<TabItem value="Markdown">
|
|
108
|
+
|
|
109
|
+
```md
|
|
110
|
+
# entra roledefinition add --displayName "Custom Role" --allowedResourceActions "microsoft.directory/groups.unified/create,microsoft.directory/groups.unified/delete" --version 1
|
|
111
|
+
|
|
112
|
+
Date: 12/15/2024
|
|
113
|
+
|
|
114
|
+
## Custom Role (3844129d-f748-4c03-8165-4412ee9b4ceb)
|
|
115
|
+
|
|
116
|
+
Property | Value
|
|
117
|
+
---------|-------
|
|
118
|
+
id | 3844129d-f748-4c03-8165-4412ee9b4ceb
|
|
119
|
+
displayName | Custom Role
|
|
120
|
+
isBuiltIn | false
|
|
121
|
+
isEnabled | true
|
|
122
|
+
templateId | 3844129d-f748-4c03-8165-4412ee9b4ceb
|
|
123
|
+
version | 1
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
</TabItem>
|
|
127
|
+
</Tabs>
|
|
128
|
+
|
|
129
|
+
## More information
|
|
130
|
+
|
|
131
|
+
- https://learn.microsoft.com/graph/api/rbacapplication-post-roledefinitions
|