
Tool SharePoint API Demo (Vanilla JS)
Precedentemente ho pubblicato un tool Tool SharePoint API Demo (SPFx) in SPFx da usare per testare le API di SharePoint Online, ma purtroppo alcuni clienti non permettono di installare SPFx nel loro tenant.
Per questo ho realizzato un tool alternativo in JavaScript puro (Vanilla JS) che può essere incollato nella Developer toolbar (F12 o CTRL+SHFT+I) del Browser all'interno di una pagina SharePoint.
Interfaccia utente
La Developer Toolbar può essere chiusa per aumentare l'area di lavoro.
Su molti Browser, quando si cerca di incollare nella Developer Toolbar, compare questo messaggio:
e creare un segnalibro nel Browser con questo codice:
Cliccando sul segnalibro, in una pagina SharePoint, si fa partire il tool.
Vedi anche Tool SharePoint API Demo (SPFx).
Per questo ho realizzato un tool alternativo in JavaScript puro (Vanilla JS) che può essere incollato nella Developer toolbar (F12 o CTRL+SHFT+I) del Browser all'interno di una pagina SharePoint.

Il vantaggio di questo approccio è che non richiede login aggiuntivi o AppRegistration, lavora nel contesto utente loggato in SharePoint.
Come usarlo
il codice da incollare nella Developer toolbar è questo:JavaScript: Codice JavaScript da incollare nel browser
(function () {
/*
SharePoint Tool Api Demo (Sgart.it)
javascript:(function(){var s=document.createElement('script');s.src='/SiteAssets/ToolApiDemo/sgart-sp-tool-api-demo.js?t='+(new Date()).getTime();document.head.appendChild(s);})();
*/
let serverRelativeUrlPrefix = "/";
const VERSION = "1.1.2025-10-27";
function injectStyle() {
const css = `
.sgart-content {
font-family: Arial, sans-serif;
border: 0;
display: flex;
flex-direction: column;
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: white;
margin: 0;
padding: 0;
z-index: 10000;
}
.sgart-content input, .sgart-content textarea, .sgart-content select, .sgart-content .sgart-button {
font-family: Arial, sans-serif;
font-size: 14px;
height: 32px;
padding: 0 10px;
border: 1px solid rgb(149, 60, 15);
background-color: white;
}
.sgart-content select {
width: 110px;
}
.sgart-content #sgart-api-demo {
width: 200px;
}
.sgart-content .sgart-button {
background-color: rgb(149, 60, 15);
color: white;
padding: 0px 20px;
cursor: pointer;
}
.sgart-content .sgart-separator{
margin: 0px 10px;
}
.sgart-header {
background-color: #080808;
color: white;
padding: 10px;
border-bottom: 1px solid #ccc;
height: 40px;
display: flex;
flwex-direction: row;
align-items: center;
justify-content: space-between;
}
.sgart-header .sgart-button {
background-color: #080808;
color: white;
padding: 0px 20px;
}
.sgart-header .logo {
height: 33px;
margin-right: 10px;
}
.sgart-toolbar{
display:flex;
flex-direction: row;
align-items: center;
justify-content: space-between;
}
.sgart-toolbar-left{
}
.sgart-toolbar-right{
justify-content: right;
}
.sgart-body {
display: flex;
flex-direction: column;
flex-grow: 1;
padding: 10px;
gap: 10px;
}
.sgart-input-area {
display: flex;
gap: 10px;
align-items: center;
jsutify-content: space-between;
}
#sgart-input {
flex-grow: 1;
}
.sgart-output-area {
flex-grow: 1;
display: flex;
}
#sgart-output {
width: 100%;
height: 100%;
flex-grow: 1;
gap: 10px;
font-family: monospace;
resize: none;
}
`;
const stylePrev = document.head.getElementsByClassName('sgart-inject-style')[0];
if (stylePrev) {
document.head.removeChild(stylePrev);
}
const style = document.createElement('style');
style.className = 'sgart-inject-style';
//style.type = 'text/css';
style.appendChild(document.createTextNode(css));
document.head.appendChild(style);
}
function copyToClipboard(text) {
navigator.clipboard.writeText(text)
.then(() => {
//console.log('Text copied to clipboard:', text);
})
.catch(err => {
console.error('Failed to copy text: ', err);
});
}
function getQueryParam(query) {
if (!query) return "";
const q = "?"
+ Object.keys(query)
//.map(k => "$" + encodeURIComponent(k) + "=" + encodeURIComponent(query[k]))
.map(k => "$" + k + "=" + query[k])
.join("&");
return q;
}
function loadAPiUrlOptions() {
const configs = {
groups: [
{
id: "site",
title: "Site",
actions: [
{
id: "getSite",
title: "Get site",
url: "site"
},
{
id: "getSiteId",
title: "Get site id",
url: "site/id"
}
]
},
{
id: "web",
title: "Web",
actions: [
{
id: "getWeb",
title: "Get web",
url: "web"
},
{
id: "getWebById",
title: "Get sub webs",
url: "web/webs"
},
{
id: "getWebSiteUsers",
title: "Get site users",
url: "web/siteusers"
},
{
id: "getWebSiteGrous",
title: "Get site groups",
url: "web/sitegroups"
},
{
id: "getWebRoleDefinitions",
title: "Get site role definitions",
url: "web/roledefinitions"
}
]
},
{
id: "user",
title: "User",
actions: [
{
id: "getCurrentUser",
title: "Get current user",
url: "web/CurrentUser"
},
{
id: "getUsers",
title: "Get users",
url: "web/SiteUsers"
},
{
id: "getUserById",
title: "Get user by id",
url: "web/GetUserById(1)"
},
{
id: "getGroups",
title: "Get groups",
url: "web/sitegroups"
},
{
id: "getGroupsMembers",
title: "Get group members",
url: "web/sitegroups(1)/users"
},
]
},
{
id: "list",
title: "List",
actions: [
{
id: "getLists",
title: "Get lists",
url: "web/lists"
},
{
id: "getListByGuid",
title: "Get list by guid",
url: "web/lists(guid'00000000-0000-0000-0000-000000000000')"
},
{
id: "getListByTitle",
title: "Get list by title",
url: "web/lists/getbytitle('Documents')"
},
{
id: "getListByTitleRootFolder",
title: "Get list by title RootFolder",
url: "web/lists/getbytitle('Documents')/RootFolder"
},
{
id: "getFields",
title: "Get fields",
url: "web/lists/getbytitle('Documents')/fields"
},
{
id: "getViews",
title: "Get views",
url: "web/lists/getbytitle('Documents')/views?$top=100&$orderBy=Title"
},
{
id: "getViewsHidden",
title: "Get views hidden",
url: "web/lists/getbytitle('Documents')/views?$select=Id,Title,ServerRelativeUrl,ViewQuery&$top=100&$orderBy=Title&$filter=Hidden eq true"
},
{
id: "getContenttypes",
title: "Get content types",
url: "web/lists/getbytitle('Documents')/contenttypes?$select=*&$orderBy=Name"
},
]
},
{
id: "item",
title: "Item",
actions: [
{
id: "getItems",
title: "Get items",
url: "web/lists/getbytitle('Documents')/items"
},
{
id: "getItemById",
title: "Get item by id",
url: "web/lists/getbytitle('Documents')/items(1)"
},
{
id: "getItemByIdWithSelect",
title: "Get item by id with select and expand",
url: "web/lists/getbytitle('Documents')/items(1)?$select=Title,Id,Created,Modified,Author/Title,Editor/Title&$expand=Author,Editor"
},
{
id: "getItemAttachments",
title: "Get item attachments",
url: "web/lists/getbytitle('Documents')/items(1)/AttachmentFiles",
}
]
},
{
id: "file",
title: "File",
actions: [
{
id: "getFileById",
title: "Get file by id",
url: "web/getfilebyid('00000000-0000-0000-0000-000000000000')"
},
{
id: "getFileByServerRelativeUrl",
title: "Get file by server relative url",
url: "web/getfilebyserverrelativeurl('/sites/someSite/Shared Documents/file.txt')"
},
{
id: "getFileContent",
title: "Get file content",
url: "web/getfilebyserverrelativeurl('/sites/someSite/Shared Documents/file.txt')/$value"
}
]
},
{
id: "folder",
title: "Folder",
actions: [
{
id: "getFolderByServerRelativeUrl",
title: "Get folder by server relative url",
url: "web/getfolderbyserverrelativeurl('Shared Documents')"
},
{
id: "getFolderById",
title: "Get folder by id",
url: "web/getfolderbyid('00000000-0000-0000-0000-000000000000')"
},
{
id: "getFolderFiles",
title: "Get folder files",
url: "web/getfolderbyserverrelativeurl('Shared Documents')/files"
},
{
id: "getFolderFileContent",
title: "Get folder files",
url: "web/getfolderbyserverrelativeurl('/sites/someSite/Shared Documents/file name')/$value"
}
]
},
{
id: "search",
title: "Search",
actions: [
{
id: "searchSites",
title: "Search sites",
url: "search/query",
query: {
mode: "search",
filter: "sharepoint (contentclass:STS_Site) Path:\"https://sgart.sharepoint.com/*\"",
select: "Title,Path,Description,SiteLogo,WebTemplate,WebId,SiteId,Created,LastModifiedTime"
}
}
]
},
{
id: "userProfile",
title: "User Profile",
actions: [
{
id: "getPMInstance",
title: "PeopleManager instance",
url: "SP.UserProfiles.PeopleManager"
},
{
id: "getPMFollowedByMe",
title: "Followed by ME",
url: "SP.UserProfiles.PeopleManager/getpeoplefollowedbyme",
query: {
select: "*"
}
},
{
id: "getPMFollowedBy",
title: "Followed by ...",
url: "SP.UserProfiles.PeopleManager/getpeoplefollowedby(@v)?@v='i%3A0%23.f%7Cmembership%7Cuser%40domain.onme",
query: {
select: "*"
}
}
]
},
{
id: "taxonomy",
title: "Taxonomy",
actions: [
{
id: "getTermStoreGroups",
title: "Get groups",
url: "v2.1/termStore/groups"
},
{
id: "getTermStoreTermGroups",
title: "Get term groups",
url: "v2.1/termStore/termGroups"
},
{
id: "getTermStoreSets",
title: "Get term sets",
url: "v2.1/termStore/groups/{groupid}/sets"
},
{
id: "getTermStoreSetById",
title: "Get terms by set id",
url: "v2.1/termStore/groups/{groupid}/sets/{termSetId}/terms"
},
{
id: "getTermById",
title: "Get terms by set id next level",
url: "v2.1/termStore/groups/{groupid}/sets/{termSetId}/terms/{termId}/terms"
}
]
},
{
id: "webhooks",
title: "Webhooks",
actions: [
{
id: "getWebhooks",
title: "Get webhooks",
url: "web/lists/getbytitle('Documents')/subscriptions"
}
]
},
{
id: "tenant",
title: "Tenant",
actions: [
{
id: "getTenantAppCatalog",
title: "Get tenant app catalog",
url: "SP_TenantSettings_Current"
}
]
}
]
};
const select = document.getElementById('sgart-api-demo');
const optionEmpty = document.createElement('option');
optionEmpty.value = "";
optionEmpty.text = "API URL";
select.appendChild(optionEmpty);
configs.groups.forEach(group => {
const optGroup = document.createElement('optgroup');
optGroup.label = group.title;
group.actions.forEach(action => {
const relativeUrl = '_api/' + action.url + getQueryParam(action.query);
const option = document.createElement('option');
option.value = serverRelativeUrlPrefix + relativeUrl;
option.text = action.title + ": [ " + relativeUrl + " ]";
option.setAttribute('data-group', group.id);
option.setAttribute('data-action', action.id);
optGroup.appendChild(option);
});
select.appendChild(optGroup);
});
select.onchange = function () {
document.getElementById('sgart-input').value = this.value;
document.getElementById('sgart-output').value = "";
};
document.querySelector('#sgart-api-demo [data-action=getWeb]').selected = true;
select.onchange();
executeApiCall();
}
function showInterface() {
const interfaceDivPrev = document.getElementById('sgart-content');
if (interfaceDivPrev) {
document.body.removeChild(interfaceDivPrev);
}
const interfaceDiv = document.createElement('div');
interfaceDiv.id = 'sgart-content';
interfaceDiv.className = 'sgart-content';
interfaceDiv.innerHTML = `
<div class="sgart-header">
<a href="https://www.sgart.it" target="_blank"><img alt="Logo Sgart.it" class="logo" src=""></a>
<h3>Tool SharePoint API Demo (Vanilla JS)</h3>
<button id="sgart-close" class="sgart-button">Close</button>
</div>
<div class="sgart-body">
<div class="sgart-input-area">
<label for="sgart-input">API url:</label>
<input type="text" id="sgart-input" value="web/lists">
<select id="sgart-api-demo" title="API Demo URLs">
</select>
<select id="sgart-odata-mode" title="OData http header accept">
<option value="nometadata" selected>No Metadata [accept:application/json; odata=nometadata]</option>
<option value="verbose">OData Verbose [accept:application/json; odata=verbose]</option>
</select>
</div>
<div class="sgart-toolbar">
<div class="sgart-toolbar-left">
<button id="sgart-execute" class="sgart-button">Execute</button>
<span class="sgart-separator">|</span>
<button id="sgart-clear-output" class="sgart-button">Clear</button>
<button id="sgart-copy-output" class="sgart-button">Copy</button>
<!--
<span class="sgart-separator">|</span>
<button id="sgart-tab-response" class="sgart-button">Response</button>
<button id="sgart-tab-table" class="sgart-button">Table</button>
-->
</div>
<div class="sgart-toolbar-right">v. ${VERSION}</div>
</div>
<div class="sgart-output-area">
<textarea id="sgart-output"></textarea>
</div>
</div>
`;
interfaceDiv.querySelector('#sgart-close').onclick = function () {
document.body.removeChild(interfaceDiv);
const style = document.head.getElementsByClassName('sgart-inject-style')[0];
if (style) {
document.head.removeChild(style);
}
};
interfaceDiv.querySelector('#sgart-clear-output').onclick = function () {
document.getElementById('sgart-output').value = "";
};
interfaceDiv.querySelector('#sgart-copy-output').onclick = function () {
copyToClipboard(document.getElementById('sgart-output').value);
};
document.body.appendChild(interfaceDiv);
loadAPiUrlOptions();
}
const fetchGetJson = async (url, odataVerbose, outputNormal) => {
/* to paste into the 'Browser Developer Console' in another Tenant */
const ct = "application/json; odata=" + (odataVerbose ? "verbose" : "nometadata");
const response = await fetch(url, { method: "GET", headers: { "Accept": ct, "Content-Type": ct } });
if (!response.ok) {
const txt = await response.json();
throw new Error(`Response status: ${response.status}, ${response.statusText}, ${txt}`);
}
const data = await response.json();
if (outputNormal)
return data;
if (odataVerbose) {
if (data.d && data.d.results) return data.d.results;
else if (data.d) return data.d;
return data;
}
return data.value ?? data;
};
function executeApiCall() {
const input = document.getElementById('sgart-input').value;
const outputArea = document.getElementById('sgart-output');
const modeVerbose = document.getElementById('sgart-odata-mode').value === 'verbose';
outputArea.value = "Executing...";
fetchGetJson(input, modeVerbose).then(data => {
outputArea.value = JSON.stringify(data, null, 2);
}).catch(error => {
outputArea.value = "Error: " + error.message;
});
}
function init() {
const i = window.location.pathname.toLocaleLowerCase().indexOf('/sites/');
if (i >= 0) {
serverRelativeUrlPrefix = window.location.pathname.substring(0, window.location.pathname.indexOf('/', i + 7)) + "/";
} else {
serverRelativeUrlPrefix = "/";
}
injectStyle();
showInterface();
document.getElementById('sgart-execute').onclick = executeApiCall;
console.log("Sgart.it SharePoint API Test Interface initialized v." + VERSION);
//console.log(fetchGetJson.toString());
}
init();
})();
L'ultima versione del codice JavaScript può essere scaricato da GitHub: sgart-vanillajs-sp-api-demo.
Il tool rimane attivo finchè non si cambia pagina oppure si preme il pulsante Close.La Developer Toolbar può essere chiusa per aumentare l'area di lavoro.
Su molti Browser, quando si cerca di incollare nella Developer Toolbar, compare questo messaggio:
Warning: Don’t paste code into the DevTools Console that you don’t understand or haven’t reviewed yourself. This could allow attackers to steal your identity or take control of your computer. Please type ‘allow pasting’ below and press Enter to allow pasting.
per abilitare l'incolla, scrivere allow pasting e premere invio.Alternativa
In alternativa si può caricare il file JavaScript in una document library SharePoint, ad esempioURL
/SiteAssets/ToolApiDemo/sgart-sp-tool-api-demo.js
JavaScript
javascript:(function(){var s=document.createElement('script');s.src='/SiteAssets/ToolApiDemo/sgart-sp-tool-api-demo.js?t='+(new Date()).getTime();document.head.appendChild(s);})();
Attualmente è possibile incollare del codice JavaScript da un preferito in una pagina.
Non è escluso che in futuro, per motivi di sicurezza, disabilitino questa possibilità.
Non è escluso che in futuro, per motivi di sicurezza, disabilitino questa possibilità.
N.B. Le content security policy di SharePoint Online non permettono di salvare lo script in un sito differente e richiamarlo da un preferito..
Content-Security-Policy: Le impostazioni della pagina hanno bloccato l’esecuzione di uno script (script-src-elem) da https://raw.githubusercontent.com/sg...761601961831 in quanto viola la seguente direttiva: “script-src https://res-1.cdn.office.net https://res-1.cdn.office.net 'self' https://contentstorage.osi.office.net https://swx.cdn.skype.com https://res.delve.office.com https://lpcres.delve.office.com https://widget.uservoice.com https://by2.uservoice.com https://www.bing.com/api/maps/ https://www.bing.com/rms/ https://fabriciss.azureedge.net https://public-cdn.sharepointonline.com https://ajax.aspnetcdn.com https://res-1.cdn.office.net https://res-1.cdn.office.net https://res-2.cdn.office.net https://webshell.suite.office.com https://amcdn.msftauth.net https://*.cdn.office.net https://*.fluidpreview.office.net https://*.onecdn.static.microsoft https://res-1.cdn.office.net https://teams.microsoft.com https://js.monitor.azure.com https://r4.res.office365.com https://res.public.onecdn.static.microsoft https://c1-word-view-15.cdn.office.net https://loki.delve.office.com https://res.cdn.office.net/midgard/ https://substrate.office.com 'unsafe-eval' 'nonce-p3gvvusv0lk'”
Content-Security-Policy: Le impostazioni della pagina hanno bloccato l’esecuzione di uno script (script-src-elem) da https://raw.githubusercontent.com/sg...761601961831 in quanto viola la seguente direttiva: “script-src https://res-1.cdn.office.net https://res-1.cdn.office.net 'self' https://contentstorage.osi.office.net https://swx.cdn.skype.com https://res.delve.office.com https://lpcres.delve.office.com https://widget.uservoice.com https://by2.uservoice.com https://www.bing.com/api/maps/ https://www.bing.com/rms/ https://fabriciss.azureedge.net https://public-cdn.sharepointonline.com https://ajax.aspnetcdn.com https://res-1.cdn.office.net https://res-1.cdn.office.net https://res-2.cdn.office.net https://webshell.suite.office.com https://amcdn.msftauth.net https://*.cdn.office.net https://*.fluidpreview.office.net https://*.onecdn.static.microsoft https://res-1.cdn.office.net https://teams.microsoft.com https://js.monitor.azure.com https://r4.res.office365.com https://res.public.onecdn.static.microsoft https://c1-word-view-15.cdn.office.net https://loki.delve.office.com https://res.cdn.office.net/midgard/ https://substrate.office.com 'unsafe-eval' 'nonce-p3gvvusv0lk'”
Vedi anche Tool SharePoint API Demo (SPFx).