A big part of my job these days is interop between Dynamics AX and various external services/resources. A WCF service hosted in our AX environment is often a key part of that equation. With older .NET Framework applications, it’s easy to add a reference to a WCF web service. And I’ve done that so often that I could probably do it in my sleep. If I need to interface with a new AX service, I’ll generally just go through the “Add Service Reference” procedure, then copy & paste some code from a previous project and adjust it for my curent needs.
I was recently working on a new program that I decided to try to write using .NET Core instead of .NET Framework. It took me quite a while to figure out how to deal with calling an AX web service under .NET Core, so I thought I’d write it up, briefly, with a couple of sample code snippets.
First, there is a facility for adding a WCF service reference in a .NET Core 2 project in VS 2017. (I think this might have been missing in earlier versions of VS and/or earlier versions of .NET Core.) It’s pretty similar to the tool that works with .NET Framework projects, but there are a few key differences in the generated code. The biggest difference is that it doesn’t add anything to app.config/web.config, and in fact isn’t set up to read any configuration info from the config files at all. So you need to do the config in your code. (Of course, you can write your own code to read from your config file.) Anyway, it took a lot of trial and error before I figured out what I needed to do. There’s not as much documentation on this as there could be. So here’s a simple example, showing a bit of code (and config) from a .NET Framework project, and the equivalent code from a .NET Core project.
(I’m embedding it below as a Gist, since I can’t get WordPress to play nice with the XML config sample right now.)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// old way: | |
public async Task RunAsync() | |
{ | |
CallContext context = new CallContext(); | |
context.Company = "axcompany"; | |
string pingResp = string.Empty; | |
var client = new XYZPurchInfoServiceClient(); | |
var rv = await client.wsPingAsync(context); | |
pingResp = rv.response; | |
Console.WriteLine("Ping response: {0}", pingResp); | |
} | |
/* app.config: | |
<system.serviceModel> | |
<bindings> | |
<netTcpBinding> | |
<binding name="NetTcpBinding_XYZPurchInfoService" /> | |
</netTcpBinding> | |
</bindings> | |
<client> | |
<endpoint address="net.tcp://myserver:8201/DynamicsAx/Services/XYZPurchInfoServices" | |
binding="netTcpBinding" bindingConfiguration="NetTcpBinding_XYZPurchInfoService" | |
contract="XYZPurchInfoSvcRef.XYZPurchInfoService" name="NetTcpBinding_XYZPurchInfoService"> | |
<identity> | |
<userPrincipalName value="myservice@corp.local" /> | |
</identity> | |
</endpoint> | |
</client> | |
</system.serviceModel> | |
*/ | |
// new way: | |
CallContext context = new CallContext(); | |
context.Company = "axcompany"; | |
string pingResp = string.Empty; | |
var client = new XYZPurchInfoServiceClient(GetBinding(), GetEndpointAddr()); | |
var rv = await client.wsPingAsync(context); | |
pingResp = rv.response; | |
Console.WriteLine("Ping response: {0}", pingResp); | |
private NetTcpBinding GetBinding() | |
{ | |
var netTcpBinding = new NetTcpBinding(); | |
netTcpBinding.Name = "NetTcpBinding_XYZPurchInfoService"; | |
netTcpBinding.MaxBufferSize = int.MaxValue; | |
netTcpBinding.MaxReceivedMessageSize = int.MaxValue; | |
return netTcpBinding; | |
} | |
private EndpointAddress GetEndpointAddr() | |
{ | |
string url = "net.tcp://myserver:8201/DynamicsAx/Services/XYZPurchInfoServices"; | |
string user = "myservice@corp.local"; | |
var uri = new Uri(url); | |
var epid = new UpnEndpointIdentity(user); | |
var addrHdrs = new AddressHeader[0]; | |
var endpointAddr = new EndpointAddress(uri, epid, addrHdrs); | |
return endpointAddr; | |
} |
This example obviously isn’t applicable in all use cases. But I think it could point you in the right direction, if you’re trying to do this and you’re as befuddled as I was when I started this. I should also mention that reading the auto-generated code produced by the tool is somewhat useful, though the code is about as messy as most auto-generated code tends to be.
Some useful resources:
- The Github project for .NET Core WCF
- A doc page on how to Use the WCF Web Service Reference Provider Tool for .NET Core
- How to: Add, update, or remove a WCF data service reference in .NET Framework
- A reasonably good book on AX web services: Microsoft Dynamics AX 2012 R2 Services
Your article saved me a lot of time. I was down a very long rabbit hole implementing some other much more complicated solution. This worked, although I was a little unsure on what the first step should have been. I think it would be helpful if, before your code snippet, you included a note to use the built in Add Service Reference tool in VS first. Maybe also mention that the app.config snippet is there for reference only.
What’s nice about your solution is it can easily be modified to pass the address, and user, to the GetEndpointAddr method, thus allowing configuration per deployment environment.
Your solution is brilliant. I tried to do that without success for sometime now. The only doubt that I have is where you set the user
string user = “myservice@corp.local”;
Here you are not passing the user pwd, so I think that this is not used at all. I tried also leaving
string user = “”;
and it still works the same, I think with my logged in user.
Do you have any idea how to call the web service using custom credentials?
Add the following for credentials.
client.ClientCredentials.Windows.ClientCredential.Domain = “YourDomain”;
client.ClientCredentials.Windows.ClientCredential.UserName = “YourUsername”;
client.ClientCredentials.Windows.ClientCredential.Password = “YourPassword”;