Skip to content

Commit 69522e6

Browse files
author
Henrik Frystyk Nielsen
committedDec 30, 2016
Fixed Salesforce receiver to return failure responses as XML in most cases and added Salesforce sample.
1 parent 28e44e4 commit 69522e6

19 files changed

+472
-12
lines changed
 

‎WebHooks.sln

+9
Original file line numberDiff line numberDiff line change
@@ -160,6 +160,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "VstsReceiver", "samples\Vst
160160
EndProject
161161
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "GenericReceivers.DependencyInjection", "samples\GenericReceivers.DependencyInjection\GenericReceivers.DependencyInjection.csproj", "{3EE37ECD-B24C-4096-950B-7C9D73B0E4BB}"
162162
EndProject
163+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SalesforceReceiver", "samples\SalesforceReceiver\SalesforceReceiver.csproj", "{3B7DEFC0-1032-448B-98B6-4B875A8E4285}"
164+
EndProject
163165
Global
164166
GlobalSection(SolutionConfigurationPlatforms) = preSolution
165167
CodeAnalysis|Any CPU = CodeAnalysis|Any CPU
@@ -587,6 +589,12 @@ Global
587589
{3EE37ECD-B24C-4096-950B-7C9D73B0E4BB}.Debug|Any CPU.Build.0 = Debug|Any CPU
588590
{3EE37ECD-B24C-4096-950B-7C9D73B0E4BB}.Release|Any CPU.ActiveCfg = Release|Any CPU
589591
{3EE37ECD-B24C-4096-950B-7C9D73B0E4BB}.Release|Any CPU.Build.0 = Release|Any CPU
592+
{3B7DEFC0-1032-448B-98B6-4B875A8E4285}.CodeAnalysis|Any CPU.ActiveCfg = Release|Any CPU
593+
{3B7DEFC0-1032-448B-98B6-4B875A8E4285}.CodeAnalysis|Any CPU.Build.0 = Release|Any CPU
594+
{3B7DEFC0-1032-448B-98B6-4B875A8E4285}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
595+
{3B7DEFC0-1032-448B-98B6-4B875A8E4285}.Debug|Any CPU.Build.0 = Debug|Any CPU
596+
{3B7DEFC0-1032-448B-98B6-4B875A8E4285}.Release|Any CPU.ActiveCfg = Release|Any CPU
597+
{3B7DEFC0-1032-448B-98B6-4B875A8E4285}.Release|Any CPU.Build.0 = Release|Any CPU
590598
EndGlobalSection
591599
GlobalSection(SolutionProperties) = preSolution
592600
HideSolutionNode = FALSE
@@ -662,6 +670,7 @@ Global
662670
{B27CC8B5-BF38-434B-B5CE-42ACEC40624C} = {9575CB90-BC4B-43BB-8AEA-82C53FDA4187}
663671
{88FB9535-BBC9-48FE-AC34-4C3E9A4073F3} = {E957C8D9-B4A0-488B-838F-BAB4DE080A76}
664672
{3EE37ECD-B24C-4096-950B-7C9D73B0E4BB} = {E957C8D9-B4A0-488B-838F-BAB4DE080A76}
673+
{3B7DEFC0-1032-448B-98B6-4B875A8E4285} = {E957C8D9-B4A0-488B-838F-BAB4DE080A76}
665674
EndGlobalSection
666675
GlobalSection(ExtensibilityGlobals) = postSolution
667676
EnterpriseLibraryConfigurationToolBinariesPathV6 = packages\EnterpriseLibrary.TransientFaultHandling.6.0.1304.0\lib\portable-net45+win+wp8;packages\EnterpriseLibrary.TransientFaultHandling.Data.6.0.1304.1\lib\NET45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
using System.Web.Http;
2+
3+
namespace SalesforceReceiver
4+
{
5+
public static class WebApiConfig
6+
{
7+
public static void Register(HttpConfiguration config)
8+
{
9+
// Web API configuration and services
10+
11+
// Web API routes
12+
config.MapHttpAttributeRoutes();
13+
14+
config.Routes.MapHttpRoute(
15+
name: "DefaultApi",
16+
routeTemplate: "api/{controller}/{id}",
17+
defaults: new { id = RouteParameter.Optional }
18+
);
19+
20+
// Initialize Salesforce WebHook receiver
21+
config.InitializeReceiveSalesforceWebHooks();
22+
}
23+
}
24+
}
+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
<%@ Application Codebehind="Global.asax.cs" Inherits="SalesforceReceiver.WebApiApplication" Language="C#" %>
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
using System.Web.Http;
2+
3+
namespace SalesforceReceiver
4+
{
5+
public class WebApiApplication : System.Web.HttpApplication
6+
{
7+
protected void Application_Start()
8+
{
9+
GlobalConfiguration.Configure(WebApiConfig.Register);
10+
}
11+
}
12+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
using System.Reflection;
2+
using System.Runtime.InteropServices;
3+
4+
// General Information about an assembly is controlled through the following
5+
// set of attributes. Change these attribute values to modify the information
6+
// associated with an assembly.
7+
[assembly: AssemblyTitle("SalesforceReceiver")]
8+
[assembly: AssemblyDescription("")]
9+
[assembly: AssemblyConfiguration("")]
10+
[assembly: AssemblyCompany("")]
11+
[assembly: AssemblyProduct("SalesforceReceiver")]
12+
[assembly: AssemblyCopyright("Copyright © 2015")]
13+
[assembly: AssemblyTrademark("")]
14+
[assembly: AssemblyCulture("")]
15+
16+
// Setting ComVisible to false makes the types in this assembly not visible
17+
// to COM components. If you need to access a type in this assembly from
18+
// COM, set the ComVisible attribute to true on that type.
19+
[assembly: ComVisible(false)]
20+
21+
// The following GUID is for the ID of the typelib if this project is exposed to COM
22+
[assembly: Guid("3f9f62dd-c365-4fec-a10e-5314d111ebf5")]
23+
24+
// Version information for an assembly consists of the following four values:
25+
//
26+
// Major Version
27+
// Minor Version
28+
// Build Number
29+
// Revision
30+
//
31+
// You can specify all the values or you can default the Revision and Build Numbers
32+
// by using the '*' as shown below:
33+
[assembly: AssemblyVersion("1.0.0.0")]
34+
[assembly: AssemblyFileVersion("1.0.0.0")]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,164 @@
1+
<?xml version="1.0" encoding="utf-8"?>
2+
<Project ToolsVersion="12.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
3+
<Import Project="..\..\packages\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.1.0.0\build\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.props" Condition="Exists('..\..\packages\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.1.0.0\build\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.props')" />
4+
<Import Project="..\..\packages\Microsoft.Net.Compilers.1.0.0\build\Microsoft.Net.Compilers.props" Condition="Exists('..\..\packages\Microsoft.Net.Compilers.1.0.0\build\Microsoft.Net.Compilers.props')" />
5+
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
6+
<PropertyGroup>
7+
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
8+
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
9+
<ProductVersion>
10+
</ProductVersion>
11+
<SchemaVersion>2.0</SchemaVersion>
12+
<ProjectGuid>{3B7DEFC0-1032-448B-98B6-4B875A8E4285}</ProjectGuid>
13+
<ProjectTypeGuids>{349c5851-65df-11da-9384-00065b846f21};{fae04ec0-301f-11d3-bf4b-00c04f79efbc}</ProjectTypeGuids>
14+
<OutputType>Library</OutputType>
15+
<AppDesignerFolder>Properties</AppDesignerFolder>
16+
<RootNamespace>SalesforceReceiver</RootNamespace>
17+
<AssemblyName>SalesforceReceiver</AssemblyName>
18+
<TargetFrameworkVersion>v4.5.1</TargetFrameworkVersion>
19+
<UseIISExpress>true</UseIISExpress>
20+
<IISExpressSSLPort />
21+
<IISExpressAnonymousAuthentication />
22+
<IISExpressWindowsAuthentication />
23+
<IISExpressUseClassicPipelineMode />
24+
<UseGlobalApplicationHostFile />
25+
<NuGetPackageImportStamp>
26+
</NuGetPackageImportStamp>
27+
<TargetFrameworkProfile />
28+
</PropertyGroup>
29+
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
30+
<DebugSymbols>true</DebugSymbols>
31+
<DebugType>full</DebugType>
32+
<Optimize>false</Optimize>
33+
<OutputPath>bin\</OutputPath>
34+
<DefineConstants>DEBUG;TRACE</DefineConstants>
35+
<ErrorReport>prompt</ErrorReport>
36+
<WarningLevel>4</WarningLevel>
37+
</PropertyGroup>
38+
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
39+
<DebugType>pdbonly</DebugType>
40+
<Optimize>true</Optimize>
41+
<OutputPath>bin\</OutputPath>
42+
<DefineConstants>TRACE</DefineConstants>
43+
<ErrorReport>prompt</ErrorReport>
44+
<WarningLevel>4</WarningLevel>
45+
</PropertyGroup>
46+
<ItemGroup>
47+
<Reference Include="Microsoft.CodeDom.Providers.DotNetCompilerPlatform, Version=1.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
48+
<HintPath>..\..\packages\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.1.0.0\lib\net45\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.dll</HintPath>
49+
<Private>True</Private>
50+
</Reference>
51+
<Reference Include="Microsoft.CSharp" />
52+
<Reference Include="System.Data.DataSetExtensions" />
53+
<Reference Include="System.Net.Http" />
54+
<Reference Include="System.Net.Http.Formatting, Version=5.2.2.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
55+
<HintPath>..\..\packages\Microsoft.AspNet.WebApi.Client.5.2.2\lib\net45\System.Net.Http.Formatting.dll</HintPath>
56+
<Private>True</Private>
57+
</Reference>
58+
<Reference Include="System.Web.DynamicData" />
59+
<Reference Include="System.Web.Entity" />
60+
<Reference Include="System.Web.ApplicationServices" />
61+
<Reference Include="System.ComponentModel.DataAnnotations" />
62+
<Reference Include="System" />
63+
<Reference Include="System.Data" />
64+
<Reference Include="System.Web.Extensions" />
65+
<Reference Include="System.Web.Http, Version=5.2.2.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
66+
<HintPath>..\..\packages\Microsoft.AspNet.WebApi.Core.5.2.2\lib\net45\System.Web.Http.dll</HintPath>
67+
<Private>True</Private>
68+
</Reference>
69+
<Reference Include="System.Web.Http.WebHost, Version=5.2.2.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
70+
<HintPath>..\..\packages\Microsoft.AspNet.WebApi.WebHost.5.2.2\lib\net45\System.Web.Http.WebHost.dll</HintPath>
71+
<Private>True</Private>
72+
</Reference>
73+
<Reference Include="System.Drawing" />
74+
<Reference Include="System.Web" />
75+
<Reference Include="System.Xml" />
76+
<Reference Include="System.Configuration" />
77+
<Reference Include="System.Web.Services" />
78+
<Reference Include="System.EnterpriseServices" />
79+
<Reference Include="System.Xml.Linq" />
80+
</ItemGroup>
81+
<ItemGroup>
82+
<Reference Include="Newtonsoft.Json">
83+
<HintPath>..\..\packages\Newtonsoft.Json.6.0.4\lib\net45\Newtonsoft.Json.dll</HintPath>
84+
</Reference>
85+
</ItemGroup>
86+
<ItemGroup>
87+
<Content Include="Global.asax" />
88+
<Content Include="index.html" />
89+
<Content Include="Web.config" />
90+
</ItemGroup>
91+
<ItemGroup>
92+
<Compile Include="App_Start\WebApiConfig.cs" />
93+
<Compile Include="Global.asax.cs">
94+
<DependentUpon>Global.asax</DependentUpon>
95+
</Compile>
96+
<Compile Include="Properties\AssemblyInfo.cs" />
97+
<Compile Include="WebHooks\SalesforceWebHookHandler.cs" />
98+
</ItemGroup>
99+
<ItemGroup>
100+
<Content Include="packages.config" />
101+
<None Include="Web.Debug.config">
102+
<DependentUpon>Web.config</DependentUpon>
103+
</None>
104+
<None Include="Web.Release.config">
105+
<DependentUpon>Web.config</DependentUpon>
106+
</None>
107+
</ItemGroup>
108+
<ItemGroup>
109+
<Folder Include="App_Data\" />
110+
</ItemGroup>
111+
<ItemGroup>
112+
<ProjectReference Include="..\..\Src\Microsoft.AspNet.WebHooks.Common\Microsoft.AspNet.WebHooks.Common.csproj">
113+
<Project>{f7dd0935-6320-4efc-9464-d5a2d2b8c2f7}</Project>
114+
<Name>Microsoft.AspNet.WebHooks.Common</Name>
115+
</ProjectReference>
116+
<ProjectReference Include="..\..\src\Microsoft.AspNet.WebHooks.Receivers.Salesforce\Microsoft.AspNet.WebHooks.Receivers.Salesforce.csproj">
117+
<Project>{2331e6c8-2235-4ff6-8429-4b2b9cc21273}</Project>
118+
<Name>Microsoft.AspNet.WebHooks.Receivers.Salesforce</Name>
119+
</ProjectReference>
120+
<ProjectReference Include="..\..\Src\Microsoft.AspNet.WebHooks.Receivers\Microsoft.AspNet.WebHooks.Receivers.csproj">
121+
<Project>{8ced31fb-32f2-4ffb-9997-452fb9728577}</Project>
122+
<Name>Microsoft.AspNet.WebHooks.Receivers</Name>
123+
</ProjectReference>
124+
</ItemGroup>
125+
<PropertyGroup>
126+
<VisualStudioVersion Condition="'$(VisualStudioVersion)' == ''">10.0</VisualStudioVersion>
127+
<VSToolsPath Condition="'$(VSToolsPath)' == ''">$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(VisualStudioVersion)</VSToolsPath>
128+
</PropertyGroup>
129+
<Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" />
130+
<Import Project="$(VSToolsPath)\WebApplications\Microsoft.WebApplication.targets" Condition="'$(VSToolsPath)' != ''" />
131+
<Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v10.0\WebApplications\Microsoft.WebApplication.targets" Condition="false" />
132+
<ProjectExtensions>
133+
<VisualStudio>
134+
<FlavorProperties GUID="{349c5851-65df-11da-9384-00065b846f21}">
135+
<WebProjectProperties>
136+
<UseIIS>True</UseIIS>
137+
<AutoAssignPort>True</AutoAssignPort>
138+
<DevelopmentServerPort>15572</DevelopmentServerPort>
139+
<DevelopmentServerVPath>/</DevelopmentServerVPath>
140+
<IISUrl>http://localhost:50014</IISUrl>
141+
<NTLMAuthentication>False</NTLMAuthentication>
142+
<UseCustomServer>False</UseCustomServer>
143+
<CustomServerUrl>
144+
</CustomServerUrl>
145+
<SaveServerSettingsInUserFile>False</SaveServerSettingsInUserFile>
146+
</WebProjectProperties>
147+
</FlavorProperties>
148+
</VisualStudio>
149+
</ProjectExtensions>
150+
<Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild">
151+
<PropertyGroup>
152+
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
153+
</PropertyGroup>
154+
<Error Condition="!Exists('..\..\packages\Microsoft.Net.Compilers.1.0.0\build\Microsoft.Net.Compilers.props')" Text="$([System.String]::Format('$(ErrorText)', '..\..\packages\Microsoft.Net.Compilers.1.0.0\build\Microsoft.Net.Compilers.props'))" />
155+
<Error Condition="!Exists('..\..\packages\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.1.0.0\build\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.props')" Text="$([System.String]::Format('$(ErrorText)', '..\..\packages\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.1.0.0\build\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.props'))" />
156+
</Target>
157+
<!-- To modify your build process, add your task inside one of the targets below and uncomment it.
158+
Other similar extension points exist, see Microsoft.Common.targets.
159+
<Target Name="BeforeBuild">
160+
</Target>
161+
<Target Name="AfterBuild">
162+
</Target>
163+
-->
164+
</Project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
<?xml version="1.0" encoding="utf-8"?>
2+
3+
<!-- For more information on using web.config transformation visit http://go.microsoft.com/fwlink/?LinkId=125889 -->
4+
5+
<configuration xmlns:xdt="http://schemas.microsoft.com/XML-Document-Transform">
6+
<!--
7+
In the example below, the "SetAttributes" transform will change the value of
8+
"connectionString" to use "ReleaseSQLServer" only when the "Match" locator
9+
finds an attribute "name" that has a value of "MyDB".
10+
11+
<connectionStrings>
12+
<add name="MyDB"
13+
connectionString="Data Source=ReleaseSQLServer;Initial Catalog=MyReleaseDB;Integrated Security=True"
14+
xdt:Transform="SetAttributes" xdt:Locator="Match(name)"/>
15+
</connectionStrings>
16+
-->
17+
<system.web>
18+
<!--
19+
In the example below, the "Replace" transform will replace the entire
20+
<customErrors> section of your web.config file.
21+
Note that because there is only one customErrors section under the
22+
<system.web> node, there is no need to use the "xdt:Locator" attribute.
23+
24+
<customErrors defaultRedirect="GenericError.htm"
25+
mode="RemoteOnly" xdt:Transform="Replace">
26+
<error statusCode="500" redirect="InternalError.htm"/>
27+
</customErrors>
28+
-->
29+
</system.web>
30+
</configuration>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
<?xml version="1.0" encoding="utf-8"?>
2+
3+
<!-- For more information on using web.config transformation visit http://go.microsoft.com/fwlink/?LinkId=125889 -->
4+
5+
<configuration xmlns:xdt="http://schemas.microsoft.com/XML-Document-Transform">
6+
<!--
7+
In the example below, the "SetAttributes" transform will change the value of
8+
"connectionString" to use "ReleaseSQLServer" only when the "Match" locator
9+
finds an attribute "name" that has a value of "MyDB".
10+
11+
<connectionStrings>
12+
<add name="MyDB"
13+
connectionString="Data Source=ReleaseSQLServer;Initial Catalog=MyReleaseDB;Integrated Security=True"
14+
xdt:Transform="SetAttributes" xdt:Locator="Match(name)"/>
15+
</connectionStrings>
16+
-->
17+
<system.web>
18+
<compilation xdt:Transform="RemoveAttributes(debug)" />
19+
<!--
20+
In the example below, the "Replace" transform will replace the entire
21+
<customErrors> section of your web.config file.
22+
Note that because there is only one customErrors section under the
23+
<system.web> node, there is no need to use the "xdt:Locator" attribute.
24+
25+
<customErrors defaultRedirect="GenericError.htm"
26+
mode="RemoteOnly" xdt:Transform="Replace">
27+
<error statusCode="500" redirect="InternalError.htm"/>
28+
</customErrors>
29+
-->
30+
</system.web>
31+
</configuration>

‎samples/SalesforceReceiver/Web.config

+58
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
<?xml version="1.0"?>
2+
<!--
3+
For more information on how to configure your ASP.NET application, please visit
4+
http://go.microsoft.com/fwlink/?LinkId=301879
5+
-->
6+
<configuration>
7+
<appSettings>
8+
<add key="MS_WebHookReceiverSecret_SalesforceSoap" value="Your Salesforce organization ID"/>
9+
</appSettings>
10+
<!--
11+
For a description of web.config changes see http://go.microsoft.com/fwlink/?LinkId=235367.
12+
13+
The following attributes can be set on the <httpRuntime> tag.
14+
<system.Web>
15+
<httpRuntime targetFramework="4.5.1" />
16+
</system.Web>
17+
-->
18+
<system.web>
19+
<compilation debug="true" targetFramework="4.5.1"/>
20+
<httpRuntime targetFramework="4.5.1"/>
21+
</system.web>
22+
<runtime>
23+
<assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
24+
<dependentAssembly>
25+
<assemblyIdentity name="System.Web.Helpers" publicKeyToken="31bf3856ad364e35"/>
26+
<bindingRedirect oldVersion="1.0.0.0-3.0.0.0" newVersion="3.0.0.0"/>
27+
</dependentAssembly>
28+
<dependentAssembly>
29+
<assemblyIdentity name="System.Web.Mvc" publicKeyToken="31bf3856ad364e35"/>
30+
<bindingRedirect oldVersion="1.0.0.0-5.2.2.0" newVersion="5.2.2.0"/>
31+
</dependentAssembly>
32+
<dependentAssembly>
33+
<assemblyIdentity name="System.Web.WebPages" publicKeyToken="31bf3856ad364e35"/>
34+
<bindingRedirect oldVersion="1.0.0.0-3.0.0.0" newVersion="3.0.0.0"/>
35+
</dependentAssembly>
36+
<dependentAssembly>
37+
<assemblyIdentity name="System.Web.Http" publicKeyToken="31bf3856ad364e35" culture="neutral"/>
38+
<bindingRedirect oldVersion="0.0.0.0-5.2.2.0" newVersion="5.2.2.0"/>
39+
</dependentAssembly>
40+
<dependentAssembly>
41+
<assemblyIdentity name="System.Net.Http.Formatting" publicKeyToken="31bf3856ad364e35" culture="neutral"/>
42+
<bindingRedirect oldVersion="0.0.0.0-5.2.2.0" newVersion="5.2.2.0"/>
43+
</dependentAssembly>
44+
<dependentAssembly>
45+
<assemblyIdentity name="Newtonsoft.Json" publicKeyToken="30ad4fe6b2a6aeed" culture="neutral"/>
46+
<bindingRedirect oldVersion="0.0.0.0-6.0.0.0" newVersion="6.0.0.0"/>
47+
</dependentAssembly>
48+
</assemblyBinding>
49+
</runtime>
50+
<system.webServer>
51+
<handlers>
52+
<remove name="ExtensionlessUrlHandler-Integrated-4.0"/>
53+
<remove name="OPTIONSVerbHandler"/>
54+
<remove name="TRACEVerbHandler"/>
55+
<add name="ExtensionlessUrlHandler-Integrated-4.0" path="*." verb="*" type="System.Web.Handlers.TransferRequestHandler" preCondition="integratedMode,runtimeVersionv4.0"/>
56+
</handlers>
57+
</system.webServer>
58+
</configuration>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
using System.Linq;
2+
using System.Threading.Tasks;
3+
using Microsoft.AspNet.WebHooks;
4+
5+
namespace SalesforceReceiver.WebHooks
6+
{
7+
public class SalesforceWebHookHandler : WebHookHandler
8+
{
9+
public SalesforceWebHookHandler()
10+
{
11+
this.Receiver = SalesforceSoapWebHookReceiver.ReceiverName;
12+
}
13+
14+
public override Task ExecuteAsync(string receiver, WebHookHandlerContext context)
15+
{
16+
SalesforceNotifications updates = context.GetDataOrDefault<SalesforceNotifications>();
17+
string sessionId = updates.SessionId;
18+
string company = updates.Notifications.FirstOrDefault()?["Company"];
19+
return Task.FromResult(true);
20+
}
21+
}
22+
}

‎samples/SalesforceReceiver/index.html

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
<!DOCTYPE html>
2+
<html>
3+
<head>
4+
<title>Microsoft ASP.NET WebHooks Salesforce Receiver</title>
5+
<meta charset="utf-8" />
6+
</head>
7+
<body>
8+
<h1>Microsoft ASP.NET WebHooks Salesforce Receiver</h1>
9+
10+
<p>
11+
The Salesforce WebHook receiver supports Salesforce SOAP-based Outbound Messages as a WebHook even though it is
12+
indeed a SOAP message.
13+
</p>
14+
15+
<p>A sample WebHook URI is:</p>
16+
17+
<pre>https://&lt;host&gt;/api/webhooks/incoming/sfsoap/{id}</pre>
18+
19+
<p>
20+
For security reasons, the WebHook URI must be an <c>https</c> URI and the '<c>MS_WebHookReceiverSecret_SalesforceSoap</c>'
21+
application setting must be configured to the Salesforce Organization IDs. The Organizational IDs can be found at
22+
<a href="http://www.salesforce.com">Salesforce</a> under <c>Setup | Company Profile | Company Information</c>.
23+
</p>
24+
<p>
25+
Optionally you can use the {id} field to differentiate between multiple WebHooks. You can configure each endpoint by setting
26+
the <c>MS_WebHookReceiverSecret_SalesforceSoap</c> application setting to contain multiple config values, for example '
27+
<c>code0, id1=code1, id2=code2</c>'.
28+
</p>
29+
30+
<p>
31+
For more details about using ASP.NET WebHooks with Salesforce Outbound Messages, see the blog
32+
<a href="https://blogs.msdn.microsoft.com/webdev/2015/09/06/integrating-with-salesforce-using-asp-net-webhooks-preview/">Integrating with Salesforce using ASP.NET WebHooks Preview</a>.
33+
</p>
34+
</body>
35+
</html>
+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
<?xml version="1.0" encoding="utf-8"?>
2+
<packages>
3+
<package id="Microsoft.AspNet.WebApi" version="5.2.2" targetFramework="net452" />
4+
<package id="Microsoft.AspNet.WebApi.Client" version="5.2.2" targetFramework="net452" />
5+
<package id="Microsoft.AspNet.WebApi.Core" version="5.2.2" targetFramework="net452" />
6+
<package id="Microsoft.AspNet.WebApi.WebHost" version="5.2.2" targetFramework="net452" />
7+
<package id="Microsoft.CodeDom.Providers.DotNetCompilerPlatform" version="1.0.0" targetFramework="net452" />
8+
<package id="Microsoft.Net.Compilers" version="1.0.0" targetFramework="net452" developmentDependency="true" />
9+
<package id="Newtonsoft.Json" version="6.0.4" targetFramework="net452" />
10+
</packages>

‎src/Microsoft.AspNet.WebHooks.Receivers.Salesforce/Extensions/HttpConfigurationExtensions.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ public static class HttpConfigurationExtensions
1818
/// For security reasons, the WebHook URI must be an <c>https</c> URI and the '<c>MS_WebHookReceiverSecret_SalesforceSoap</c>'
1919
/// application setting must be configured to the Salesforce Organization IDs. Organizational IDs can be found at
2020
/// <c>http://www.salesforce.com</c> under <c>Setup | Company Profile | Company Information</c>.
21-
/// For details about Salesforce Outbound Messages, see <c>https://help.salesforce.com/htviewhelpdoc?id=workflow_defining_outbound_messages.htm</c>.
21+
/// For details about Salesforce Outbound Messages, see <c>https://go.microsoft.com/fwlink/?linkid=838587</c>.
2222
/// </summary>
2323
/// <param name="config">The current <see cref="HttpConfiguration"/>config.</param>
2424
public static void InitializeReceiveSalesforceWebHooks(this HttpConfiguration config)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
<?xml version="1.0" encoding="UTF-8"?>
2+
<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
3+
<soapenv:Body>
4+
<soapenv:Fault>
5+
<faultcode>soapenv:Client</faultcode>
6+
<faultstring>{0}</faultstring>
7+
<detail />
8+
</soapenv:Fault>
9+
</soapenv:Body>
10+
</soapenv:Envelope>

‎src/Microsoft.AspNet.WebHooks.Receivers.Salesforce/Microsoft.AspNet.WebHooks.Receivers.SalesForce.nuspec

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
<iconUrl>http://go.microsoft.com/fwlink/?LinkID=288859</iconUrl>
1212
<requireLicenseAcceptance>true</requireLicenseAcceptance>
1313
<description>
14-
This package provides support for receiving Salesforce SOAP-based Outbound Messages as a WebHook. For information about Salesforce WebHooks, see "https://help.salesforce.com/htviewhelpdoc?id=workflow_defining_outbound_messages.htm".
14+
This package provides support for receiving Salesforce SOAP-based Outbound Messages as a WebHook. For information about Salesforce WebHooks, see "https://go.microsoft.com/fwlink/?linkid=838587".
1515
</description>
1616
<releaseNotes></releaseNotes>
1717
<copyright>&#169; Microsoft Corporation. All rights reserved.</copyright>

‎src/Microsoft.AspNet.WebHooks.Receivers.Salesforce/Microsoft.AspNet.WebHooks.Receivers.Salesforce.csproj

+1
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,7 @@
7070
</ItemGroup>
7171
<ItemGroup>
7272
<EmbeddedResource Include="Messages\NotificationResponse.xml" />
73+
<EmbeddedResource Include="Messages\FaultResponse.xml" />
7374
<Content Include="packages.config" />
7475
</ItemGroup>
7576
<ItemGroup>

‎src/Microsoft.AspNet.WebHooks.Receivers.Salesforce/WebHooks/SalesforceNotifications.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ namespace Microsoft.AspNet.WebHooks
1111
{
1212
/// <summary>
1313
/// Describes one or more event notifications received as an Outbound Message from Salesforce.
14-
/// For details about Salesforce Outbound Messages, see <c>https://help.salesforce.com/htviewhelpdoc?id=workflow_defining_outbound_messages.htm</c>.
14+
/// For details about Salesforce Outbound Messages, see <c>https://go.microsoft.com/fwlink/?linkid=838587</c>.
1515
/// </summary>
1616
public class SalesforceNotifications
1717
{

‎src/Microsoft.AspNet.WebHooks.Receivers.Salesforce/WebHooks/SalesforceSoapWebHookReceiver.cs

+13-5
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ namespace Microsoft.AspNet.WebHooks
2121
/// For security reasons, the WebHook URI must be an <c>https</c> URI and the '<c>MS_WebHookReceiverSecret_SalesforceSoap</c>'
2222
/// application setting must be configured to the Salesforce Organization IDs. The Organizational IDs can be found at
2323
/// <c>http://www.salesforce.com</c> under <c>Setup | Company Profile | Company Information</c>.
24-
/// For details about Salesforce Outbound Messages, see <c>https://help.salesforce.com/htviewhelpdoc?id=workflow_defining_outbound_messages.htm</c>.
24+
/// For details about Salesforce Outbound Messages, see <c>https://go.microsoft.com/fwlink/?linkid=838587</c>.
2525
/// </summary>
2626
public class SalesforceSoapWebHookReceiver : WebHookReceiver
2727
{
@@ -74,7 +74,8 @@ public override async Task<HttpResponseMessage> ReceiveAsync(string id, HttpRequ
7474
{
7575
string msg = string.Format(CultureInfo.CurrentCulture, SalesforceReceiverResources.Receiver_BadValue, "OrganizationId");
7676
context.Configuration.DependencyResolver.GetLogger().Error(msg);
77-
HttpResponseMessage invalidId = request.CreateErrorResponse(HttpStatusCode.BadRequest, msg);
77+
string fault = string.Format(CultureInfo.InvariantCulture, ReadResource("Microsoft.AspNet.WebHooks.Messages.FaultResponse.xml"), msg);
78+
HttpResponseMessage invalidId = GetXmlResponse(request, HttpStatusCode.BadRequest, fault);
7879
return invalidId;
7980
}
8081

@@ -84,7 +85,8 @@ public override async Task<HttpResponseMessage> ReceiveAsync(string id, HttpRequ
8485
{
8586
string msg = string.Format(CultureInfo.CurrentCulture, SalesforceReceiverResources.Receiver_BadBody, "ActionId");
8687
context.Configuration.DependencyResolver.GetLogger().Error(msg);
87-
HttpResponseMessage badType = request.CreateErrorResponse(HttpStatusCode.BadRequest, msg);
88+
string fault = string.Format(CultureInfo.InvariantCulture, ReadResource("Microsoft.AspNet.WebHooks.Messages.FaultResponse.xml"), msg);
89+
HttpResponseMessage badType = GetXmlResponse(request, HttpStatusCode.BadRequest, fault);
8890
return badType;
8991
}
9092

@@ -94,9 +96,8 @@ public override async Task<HttpResponseMessage> ReceiveAsync(string id, HttpRequ
9496
// Add SOAP response if not already present
9597
if (response == null || response.Content == null || !response.Content.IsXml())
9698
{
97-
response = request.CreateResponse();
9899
string ack = ReadResource("Microsoft.AspNet.WebHooks.Messages.NotificationResponse.xml");
99-
response.Content = new StringContent(ack, Encoding.UTF8, "application/xml");
100+
response = GetXmlResponse(request, HttpStatusCode.OK, ack);
100101
}
101102
return response;
102103
}
@@ -115,6 +116,13 @@ internal static string GetShortOrgId(string fullOrgId)
115116
return fullOrgId;
116117
}
117118

119+
internal static HttpResponseMessage GetXmlResponse(HttpRequestMessage request, HttpStatusCode statusCode, string msg)
120+
{
121+
HttpResponseMessage response = request.CreateResponse(statusCode);
122+
response.Content = new StringContent(msg, Encoding.UTF8, "application/xml");
123+
return response;
124+
}
125+
118126
internal static string ReadResource(string name)
119127
{
120128
Assembly asm = Assembly.GetExecutingAssembly();

‎test/Microsoft.AspNet.WebHooks.Receivers.Salesforce.Test/WebHooks/SalesforceSoapWebHookReceiverTests.cs

+15-4
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
using System.Text;
99
using System.Threading.Tasks;
1010
using System.Web.Http;
11+
using System.Xml.Linq;
1112
using Moq;
1213
using Moq.Protected;
1314
using Xunit;
@@ -16,6 +17,8 @@ namespace Microsoft.AspNet.WebHooks
1617
{
1718
public class SalesforceSoapWebHookReceiverTests : WebHookReceiverTestsBase<SalesforceSoapWebHookReceiver>
1819
{
20+
private static readonly XNamespace Soap = "http://schemas.xmlsoap.org/soap/envelope/";
21+
1922
private const string TestContent = "<soapenv:Envelope xmlns:soapenv='http://schemas.xmlsoap.org/soap/envelope/'><soapenv:Body><notifications xmlns='http://soap.sforce.com/2005/09/outbound'><OrganizationId>123456789012345</OrganizationId><ActionId>abcde</ActionId></notifications></soapenv:Body></soapenv:Envelope>";
2023
private const string TestId = "";
2124
private const string TestSecret = "123456789012345";
@@ -84,8 +87,12 @@ public async Task ReceiveAsync_ReturnsError_IfPostHasInvalidToken()
8487
HttpResponseMessage actual = await ReceiverMock.Object.ReceiveAsync(TestId, RequestContext, _postRequest);
8588

8689
// Assert
87-
HttpError error = await actual.Content.ReadAsAsync<HttpError>();
88-
Assert.Equal("The 'OrganizationId' parameter provided in the HTTP request did not match the expected value.", error.Message);
90+
XElement error = await actual.Content.ReadAsAsync<XElement>();
91+
string msg = error
92+
.Element(Soap + "Body")
93+
.Element(Soap + "Fault")
94+
.Element("faultstring").Value;
95+
Assert.Equal("The 'OrganizationId' parameter provided in the HTTP request did not match the expected value.", msg);
8996
ReceiverMock.Protected()
9097
.Verify<Task<HttpResponseMessage>>("ExecuteWebHookAsync", Times.Never(), TestId, RequestContext, _postRequest, ItExpr.IsAny<IEnumerable<string>>(), ItExpr.IsAny<object>());
9198
}
@@ -101,8 +108,12 @@ public async Task ReceiveAsync_ReturnsError_IfPostHasNoAction()
101108
HttpResponseMessage actual = await ReceiverMock.Object.ReceiveAsync(TestId, RequestContext, _postRequest);
102109

103110
// Assert
104-
HttpError error = await actual.Content.ReadAsAsync<HttpError>();
105-
Assert.Equal("The HTTP request body did not contain a required 'ActionId' property.", error.Message);
111+
XElement error = await actual.Content.ReadAsAsync<XElement>();
112+
string msg = error
113+
.Element(Soap + "Body")
114+
.Element(Soap + "Fault")
115+
.Element("faultstring").Value;
116+
Assert.Equal("The HTTP request body did not contain a required 'ActionId' property.", msg);
106117
ReceiverMock.Protected()
107118
.Verify<Task<HttpResponseMessage>>("ExecuteWebHookAsync", Times.Never(), TestId, RequestContext, _postRequest, ItExpr.IsAny<IEnumerable<string>>(), ItExpr.IsAny<object>());
108119
}

0 commit comments

Comments
 (0)
Please sign in to comment.