by joe.pesch
4. November 2013 06:12
Below is a sample app.config using a custom dictionary section to store simple key/value pairs:
<configuration>
<configSections>
<section name="CustomDictionarySection" type="System.Configuration.DictionarySectionHandler"/>
</configSections>
<startup>
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5" />
</startup>
<CustomDictionarySection>
<add key="Key1" value="Value 1" />
<add key="Key2" value="Value 2" />
<add key="Key3" value="Value 3" />
</CustomDictionarySection>
</configuration>
class Program
{
static void Main(string[] args)
{
string key = "Key2";
var section = (ConfigurationManager.GetSection("CustomDictionarySection") as System.Collections.Hashtable)
.Cast<System.Collections.DictionaryEntry>()
.Where(t => t.Key.ToString() == key);
if (section.Count() > 0)
Console.WriteLine(string.Format("Found section with key: {0} containing value: {1}", key, section.First().Value));
else
Console.WriteLine(string.Format("Could not find section with key: {0}", key));
Console.Read();
}
}
32bc91ab-430f-4a16-8611-06c544d59009|1|1.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
C#
by joe.pesch
26. October 2013 12:58
Using plUpload I was getting IO Error 2032 on larger file uploads. Two settings that seemed to be associated to this. First, if I added the resize attribute (i.e. resize: { width: 1000, height: 1000, quality: 90 }, ); however, I didn't really want to resize and was concerned that could have browser dependencies that may not be well supported in all cases. The second fix I actually opted for in this case was adding the chunk_size attribute. The interesting issue I didn't figure out yet was adding as chunk_size: '1mb', worked even though my original attempt adding chunk_size: '64kb', failed.
4f308dd3-03d8-405e-9feb-ecffbc27fa3c|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
JQuery | plUpload
by joe.pesch
27. September 2013 20:21
When using jQuery and $(document).ready() event along with an ASP.Net UpdatePanel the $(document).ready() event will only fire on the initial page load and will not fire on subsequent AJAX postbacks via the UpdatePanel. There are a couple of common solutions posted to deal with this (shown below). My preference is the first option which seems cleaner and has proven more reliable for me.
Solution 1: Wrap the $(document).ready() event inside a pageLoad() function since the AJAX postback will always fire the pageLoad() event function.
function pageLoad() {
$(document).ready(function () {
DoPageInitialization();
});
}
Solution 2: Add a reference to the PageRequestManager and register your function to the add_endrequest() event.
$(document).ready(function() {
// Still needed as the PageRequestManager below only fires on the postback...
DoPageInitialization();
});
var prm = Sys.WebForms.PageRequestManager.getInstance();
prm.add_endRequest(function() {
// Secondary binding for all AJAX postbacks through the UpdatePanel
DoPageInitialization();
});
function DoPageInitialization() {
... Document Ready Code Here ...
}
Here is a good post that goes into more details on both $(document).ready() and pageLoad()
http://encosia.com/document-ready-and-pageload-are-not-the-same/
27e5c2e2-37a5-4f3f-aac7-8350a5948f90|3|4.7|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
ASP.Net | JavaScript | JQuery
by joe.pesch
5. September 2013 12:09
Issue with SharePoint is that the WebParts already in place on an existing page don't indicate what type of web part they actually are. This is frustrating if it is not obvious what type of web part you are looking at. In order to get a better view of the WebPart in question you can navigate to the WebPart Maintenance Page by entering a URL like this: http://yoursite/whatever/default.aspx?contents=1
77adccea-7d9e-47c9-be7d-06555ccb4b62|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
Sharepoint
by joe.pesch
30. August 2013 14:15
NameValueCollection Inputs = new NameValueCollection();
Inputs.Add("param1", "something");
Inputs.Add("param2", "somethingelse");
Http.RemotePost("http://google.com", Inputs);
using System.Collections.Specialized;
using System.Net;
public static class Http
{
public static void RemotePost(string Url, NameValueCollection Inputs)
{
System.Web.HttpContext.Current.Response.Clear();
System.Web.HttpContext.Current.Response.Write("<html><head>");
System.Web.HttpContext.Current.Response.Write(string.Format("</head><body onload=\"document.{0}.submit()\">", "form1"));
System.Web.HttpContext.Current.Response.Write(string.Format("<form name=\"{0}\" method=\"{1}\" action=\"{2}\" >", "form1", "post", Url));
for (int i = 0; i < Inputs.Keys.Count; i++)
{
System.Web.HttpContext.Current.Response.Write(string.Format("<input name=\"{0}\" type=\"hidden\" value=\"{1}\">", Inputs.Keys[i], Inputs[Inputs.Keys[i]]));
}
System.Web.HttpContext.Current.Response.Write("</form>");
System.Web.HttpContext.Current.Response.Write("</body></html>");
System.Web.HttpContext.Current.Response.End();
}
}
8f7f7dc2-5da2-43d9-8f82-5565a423f9c0|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
C#
by joe.pesch
26. August 2013 15:54
We had issues with PDF documents on Sharepoint that were prompting the average "View Only" users to check-out the document. If they actually attempted to check-out the document they could not; however, this was cofusing to the users. The desired effect was to have the document open directly into the PDF viewer. Below are some registry settings that seemed to clear this issue up in most cases.
Windows Registry Editor Version 5.00
[HKEY_LOCAL_MACHINE\SOFTWARE\Policies\Adobe\Adobe Acrobat\11.0\FeatureLockDown\cSharePoint]
"bDisableSharePointFeatures"=dword:00000001
[HKEY_LOCAL_MACHINE\SOFTWARE\Policies\Adobe\Adobe Acrobat\10.0\FeatureLockDown\cSharePoint]
"bDisableSharePointFeatures"=dword:00000001
4ce4c04f-f6a4-46fd-a6d1-8b06a0d156a6|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
Sharepoint
by joe.pesch
22. August 2013 12:41
When attempting to run "DBCC ShrinkFile" SQL command to shrink a transaction log file on a database that has replication enabled it may appear to have run successfully; however, the log file may not actually shrink. To resolve this issue you can add a call to "EXEC sp_repldone" to allow the server to take action on the log file. Below is a sample process that I have added to a nightly job to take place after the full DB backup executes.
use DBName;
go
alter database DBName set recovery simple;
go
exec sp_repldone @xactid = NULL, @xact_segno = NULL, @numtrans = 0, @time = 0, @reset = 1
go
dbcc SHRINKFILE (DBName_log, 1);
go
alter database DBName set recovery full;
go
8f3efc39-b05a-4ead-95d9-6c3a86578a0e|1|1.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
SQL Server
by joe.pesch
19. August 2013 20:43
Put the following content in the .htaccess file in the root folder:
<IfModule mod_rewrite.c>
RewriteEngine On
RewriteBase /blog/
RewriteRule ^index\.php$ - [L]
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule . /blog/index.php [L]
</IfModule>
Add Rewrite to web.config:
<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<system.webServer>
<directoryBrowse enabled="false" />
<defaultDocument>
<files>
<clear />
<add value="index.php" />
</files>
</defaultDocument>
<rewrite>
<rules>
<rule name="Main Rule" stopProcessing="true">
<match url=".*" />
<conditions logicalGrouping="MatchAll">
<add input="{REQUEST_FILENAME}" matchType="IsFile" negate="true" />
<add input="{REQUEST_FILENAME}" matchType="IsDirectory" negate="true" />
</conditions>
<action type="Rewrite" url="index.php/{R:0}" />
</rule>
</rules>
</rewrite>
</system.webServer>
</configuration>
0ae8adbd-43d4-416f-9d02-52d721be0438|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
PHP | Wordpress
by joe.pesch
28. July 2013 08:45
267c1b0c-f0ab-4a31-b650-ae6b20dc27c6|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
by joe.pesch
27. July 2013 21:47
When using an ASP.Net UpdatePanel you may need to use the PageRequestManager to bind functions that would otherwise be bound on the page load or jQuery Document.Ready function (see below). Link to blog article on this topic: http://stackoverflow.com/questions/256195/jquery-document-ready-and-updatepanels
$(document).ready(function() {
// bind your jQuery events here initially
});
var prm = Sys.WebForms.PageRequestManager.getInstance();
prm.add_endRequest(function() {
// re-bind your jQuery events here
});
024f9071-0128-437d-896b-0159d2e00460|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
ASP.Net | JQuery
by joe.pesch
22. July 2013 07:09
Excerpt from this article: http://msdn.microsoft.com/en-us/library/dd203052.aspx
From the point of view of REST, a “real” Web service is a service designed to embrace the “Web” from the ground up. The Web has become the most successful distributed computing platform ever developed, thanks to years of investment and widespread usage. The Web platform includes concepts that you’re probably already familiar with like URIs, HTTP, and common data formats like XML, RSS/ATOM, and JSON. Ultimately, services that embrace these core concepts will immediately derive the valuable benefits they provide.
REST defines an architectural style based on a set of constraints for building things the “Web” way. REST is not tied to any particular technology or platform – it’s simply a way to design things to work like the Web. People often refer to services that follow this philosophy as“RESTful services.” In this whitepaper, we’ll cover the fundamental REST design principles and show you how to build RESTful services with Windows Communication Foundation (WCF), the WCF REST Starter Kit, and ADO.NET Data Services.
fdd54cbe-1cc4-450d-bd19-dcb079e8654c|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
ASP.Net | REST
by joe.pesch
22. July 2013 07:01
Sharepoint 2010 Registry Entries
I recently installed a development machine with the necessary tools to perform SharePoint development on a Windows 7 64bit virtual machine. The machine was then used as a master for a set of 7 virtual desktop developer workstations. The issue; however, was that the inital intallations created several registry entries that referenced the original machine name. This rendered the Sharepoint environment broken on all the virtual desktops (as each one had a different unique machine name). To resolve this, I went back through the registry on the master and change the original machine name to "LOCALHOST" virtual name on all the following registry entries. NOTE: The initial symptom manifest itself initially when opening the browser and attempting to browse to the local sharepoint site (in this case htt://localhost). The page threw the follwing error: "Cannot connect to the configuration database." Upon reviewing the log Sharepoint files (C:\Program Files\Common Files\Microsoft Shared\Web Server Extensions\14\LOGS\...) I noticed the original machine name in the failed connection attempt log entry.
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Office Server\14.0\LauncherSettings\LoadBalancerUrl
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Office Server\14.0\Search\Applications\c0e071e1-533e-457a-8a61-8a86a6b78dcd\Gather\Portal_Content\ContentSources\0\StartPages\0\URL
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Office Server\14.0\Search\Applications\c0e071e1-533e-457a-8a61-8a86a6b78dcd\Gather\Portal_Content\ContentSources\0\StartPages\1\URL
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Office Server\14.0\Search\Applications\c0e071e1-533e-457a-8a61-8a86a6b78dcd\ResourceManager\ (several entries)
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Shared Tools\Web Server Extensions\14.0\Search\Applications\fd2e5092-25e3-46cf-8fc6-d62c85334b3d\Gather\Search\ContentSources\0\StartPages\0\URL
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Shared Tools\Web Server Extensions\14.0\Search\Applications\fd2e5092-25e3-46cf-8fc6-d62c85334b3d\ResourceManager\ (several entries)
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Shared Tools\Web Server Extensions\14.0\Secure\ConfigDB\dsn
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Shared Tools\Web Server Extensions\14.0\WSS\CentralAdministrationURL
by joe.pesch
16. July 2013 13:14
fd97a69f-820d-443b-8367-eb750e7d16ec|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
ASP.Net | Sharepoint
by joe.pesch
1. July 2013 07:03
Here is some code to remove the titlebar from a jQuery dialog and also remove the borders and make the background transparent (effectively making the dialog itself invisible so only the actual content of the dialog is visible.
var $obj = $('selector');
$obj.dialog({ });
$obj.parents(".ui-dialog")
.css("border", "0 none")
.css("background", "transparent")
.find(".ui-dialog-titlebar").remove();
6665cc45-dc4c-4e71-9edc-b1a6905409b7|1|5.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
JQuery
by joe.pesch
27. June 2013 08:47
This approach to fix the issue with using external assemblies (i.e. assemblies not in the project folder structure). In this example the project references the "Encompass" assemblies from the "C:\Program Files (x86)\Encompass" folder. I wanted to use the program file reference so it’s easy to upgrade SDK; however, Visual Studio Project always have relative paths which doesn’t get resolved in TFS Build. To resolve this I have add new registry key for the Encompass folder path on the TFS build server so that it looks into this location when it compiles any project. Now I don’t need to change the project file.
Registry key: HKEY_LOCAL_MACHINE > SOFTWARE > Microsoft > .NET Framework > AssemblyFolders
Add new key (in this case "Encompass") and then update the "(default)" string value to be the path to the assemblies (in this case "C:\Program Files (x86)\Encompass").
cd0518d7-324e-45b5-93f1-1d34eda9687a|1|5.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
ASP.Net | Visual Studio | TFS
by joe.pesch
24. June 2013 13:43
In some cases depending on the usage you may have a jQuery dialog open after the user clicks the Ok button; however, you may want to disable the button to prevent the user from double-clicking it. Here are a couple of methods to achive this typ of behavior.
Just disable the Ok button:
$(".ui-dialog-buttonpane button:contains('Ok')").attr("disabled", true).addClass("ui-state-disabled");
Remove all buttons and replace with a processing indicator:
$('#selector').dialog({
title: 'Sample Dialog',
modal: true,
buttons: {
'Cancel': function() { $(this).dialog('close'); },
'Ok': function() { $('.ui-dialog-buttonpane').html('<div style="width: 100%; text-align: center; padding: 20px; font-size: 20px; background: url(processing.gif) no-repeat;">Processing please wait...</div>');
}
});
76be6030-8170-4d86-8b8f-299652e97cf5|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
JQuery
by joe.pesch
20. June 2013 08:09
Windows event logs will typically capture user login/logout activity; however, it does not capture user unlocking/locking their computer (which quite often is the more frequent activity). This C# code will run as a Windows service and add Information entries into the Windows Application log accessible via the system Event Viewer. Currently the code currently has an error in implementing "System.Security.Principal.WindowsIdentity.GetCurrent().Name" as an attempt to log the user name performing the activity; however, that is actually simply logging the user name that the Windows service is running as. In a future update I plan to fix this issue and add the actual user name to the log entry.
UserLockUnlockProfiler.zipx (41.86 kb)
Links regarding service interacting with user desktop:
https://stackoverflow.com/questions/5200341/capture-screen-on-server-desktop-session/12851218#12851218
https://stackoverflow.com/questions/5200341/capture-screen-on-server-desktop-session/12851218#12851218
Main Service Code:
using System;
using System.Diagnostics;
using System.Management;
using System.ServiceProcess;
namespace ULUPSvc
{
public partial class ULUPSvc : ServiceBase
{
public ULUPSvc()
{
InitializeComponent();
this.CanHandleSessionChangeEvent = true;
}
public void LogEvent(string msg)
{
msg += " " + DateTime.Now.ToString();
ManagementScope ms = new ManagementScope("\\\\.\\root\\cimv2");
ObjectQuery query = new ObjectQuery("SELECT * FROM Win32_ComputerSystem");
ManagementObjectSearcher searcher = new ManagementObjectSearcher(ms, query);
foreach (ManagementObject mo in searcher.Get())
{
msg += " (" + mo["UserName"].ToString() + ")";
}
EventLog eventLog = new EventLog();
eventLog.Log = "Application";
eventLog.Source = "ULUPSvc";
eventLog.WriteEntry(msg, EventLogEntryType.Information);
}
protected override void OnSessionChange(SessionChangeDescription changeDescription)
{
switch (changeDescription.Reason)
{
case SessionChangeReason.SessionLogon:
LogEvent("Logon");
break;
case SessionChangeReason.SessionLogoff:
LogEvent("Logoff");
break;
case SessionChangeReason.SessionLock:
LogEvent("Lock");
break;
case SessionChangeReason.SessionUnlock:
LogEvent("Unlock");
break;
}
base.OnSessionChange(changeDescription);
}
protected override void OnStart(string[] args)
{
// Make sure we have our application in the event log.
if (!EventLog.SourceExists("ULUPSvc"))
EventLog.CreateEventSource("ULUPSvc", "Application");
}
protected override void OnStop()
{
LogEvent("Stopping");
}
}
}
0d1ce6e4-9e71-4796-96e8-7417c08f7066|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
C#
by joe.pesch
15. April 2013 12:45
The following is a SQL statement that will return a list of active connections and the login authentication schema being used by the connection. This is useful in validating that connections are using the expected protocol.
select
s.session_id,
s.login_name,
s.host_name,
c.auth_scheme
from
sys.dm_exec_connections c
inner join
sys.dm_exec_sessions s
on c.session_id = s.session_id
af990970-6e5d-4230-9a3b-ca3f72905a6f|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
SQL Server
by joe.pesch
13. April 2013 23:18
Suffering issue wtih IE9 and Uploadify where the button that usually triggers the browse for file dialog does nothing. It works if I switch to compatability mode but does nothing in standard mode. I found some updated code below that fixes the issue, just replace the existing code in the jquery.uploadify.js and jquery.uploadify.min.js files.
SWFUpload.prototype.getFlashHTML = function () {
// Flash Satay object syntax: http://www.alistapart.com/articles/flashsatay
var obj = ['<object id="', this.movieName, '" type="application/x-shockwave-flash" data="', this.settings.flash_url, '" width="', this.settings.button_width, '" height="', this.settings.button_height, '" class="swfupload">'].join(""),
params = [
'<param name="wmode" value="', this.settings.button_window_mode, '" />',
'<param name="movie" value="', this.settings.flash_url, '" />',
'<param name="quality" value="high" />',
'<param name="menu" value="false" />',
'<param name="allowScriptAccess" value="always" />',
'<param name="flashvars" value="', this.getFlashVars(), '" />'
].join("");
if (navigator.userAgent.search(/MSIE/) > -1) {
obj = ['<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" id="', this.movieName, '" width="', this.settings.button_width, '" height="', this.settings.button_height, '" class="swfupload">'].join("");
params += '<param name="src" value="' + this.settings.flash_url + '" />';
}
return [obj, params, '</object>'].join("");
};
3f2f2323-d8d7-4905-ab22-2c571ab9aca0|1|1.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
JavaScript
by joe.pesch
9. April 2013 10:52
When the user information is synchronized from Active Directory (AD) into Sharepoint there are two basic processes.
First, User Profile Service Application running under the Sharepoint Services on Server is what actually imports data from AD into Sharepoint. This can be customized to select specific AD attributes and can also be configured to export data back from Sharepoint to Acitve Directory. This could be useful if you want to allow Sharepoint users to update selected attributes directly in Sharepoint and publish them back to Active Directory. You can view the current status of an active synchronization by using the Forefront Identity Manager (located at [InstatllationDrive]:\Program Files\Microsoft Office Servers\14.0\Synchronization Service\UIShell\miisclient.exe). The Sharepoint Service relies on the "Forefront Identity Manager Service" which must also be running on the Sharepoint server that is running the synchronization service.
That process simply creates, updates and deletes the records in the Sharepoint User Database by inspecting the current state of Active Directory. To have the information visible in the Sharepoint People search and liked visually to organizational heirarchy it must also be propogated into the corresponding Sharepoint content databases. That portion of the process occurs through the User Profile Sync Timer Job (running under the Sharepoint Timer service) which typically runs at the top of each hour.
IMPORTANT NOTE: If you use the People Picker in Sharepoint to locate user it will locate users directly in Active Directory (i.e. even if they have not been synchronized or setup in Sharepoint previously). If you then add a user to a library, list, etc. (i.e. grant them permissions, add to a group, etc.) they Sharepoint will create a user stub record in the User Database with minimal data (basically just the user id and guid). Next time the synchronization process runs it will update all the additional user attributes based on how the service is configured. In my case the synchronization process was failing so there was some confusion as to how some users were getting into the Sharepoint environment with minimal data (i.e. they were getting added when they were granted permissions to resources in Sharepoint).
A couple of maintenance items that can help if these processes start failing:
1) Clearing the User Profile Sync Table - Over time the data in this table will get corrupted specifically when dropping and attaching databases this is a known issue. In order to see a list of the entries you can use the listolddatabases commande below. To clean up the corrupted data you can run the deleteolddatabases command below which will delete all entries that have not been successfully updated since 1 day ago (i.e. the last parameter is the number of days since last update).
STSADM -o sync -listolddatabases 0
STSADM -o sync -deleteolddatabases 1
2) Clear the configuration cache on the servers in the Sharepoint server farm:
To resolve this issue, clear the file system cache on all servers in the server farm on which the Windows SharePoint Services Timer service is running. To do this, follow these steps:
- Stop the Timer service. To do this, follow these steps:
- Click Start, point to Administrative Tools, and then click Services.
- Right-click Windows SharePoint Services Timer, and then click Stop.
- Close the Services console.
- On the computer that is running Microsoft Office SharePoint Server 2007 and on which the Central Administration site is hosted, click Start, click Run, type explorer, and then press ENTER.
- In Windows Explorer, locate and then double-click the following folder:
Drive:\Documents and Settings\All Users\Application Data\Microsoft\SharePoint\Config\GUID
Notes
- Back up the Cache.ini file.
- Delete all the XML configuration files in the GUID folder. Do this so that you can verify that the GUID folder is replaced by new XML configuration files when the cache is rebuilt.
Note When you empty the configuration cache in the GUID folder, make sure that you do not delete the GUID folder and the Cache.ini file that is located in the GUID folder.
- Double-click the Cache.ini file.
- On the Edit menu, click Select All.
- On the Edit menu, click Delete.
- Type 1, and then click Save on the File menu.
- On the File menu, click Exit.
- Start the Timer service. To do this, follow these steps:
- Click Start, point to Administrative Tools, and then click Services.
- Right-click Windows SharePoint Services Timer, and then click Start.
- Close the Services console.
Note The file system cache is re-created after you perform this procedure. Make sure that you perform this procedure on all servers in the server farm.
- Make sure that the Cache.ini file has been updated. For example it should no longer be 1 if the cache has been updated.
- Click Start, point to Programs, point to Administrative Tools, and then click SharePoint 3.0 Central Administration.
- Click the Operations tab, and then click Timer job status under Global Configuration.
- In the list of timer jobs, verify that the status of the Config Refresh entry is Succeeded.
- On the File menu, click Close.
Original link to these clean up steps:
http://support.microsoft.com/kb/939308
ff802396-f615-4722-b2b9-748166f56539|0|.0|96d5b379-7e1d-4dac-a6ba-1e50db561b04
Tags:
Sharepoint