Skip to main content

Notifications

Announcements

No record found.

Microsoft Dynamics CRM (Archived)

System.Runtime.Serialization.SerializationException

Posted on by 75

using Microsoft.Xrm.Sdk;
using Mono.Web;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using System.Xml.Linq;
using System.ServiceModel;
using Microsoft.Xrm.Sdk.Query;

namespace MyPugin
{ [Serializable]
public class PluginPresta : IPlugin
{
public void Execute(IServiceProvider serviceProvider)
{
// Extract the tracing service for use in debugging sandboxed plug-ins.
// If you are not registering the plug-in in the sandbox, then you do
// not have to add any tracing service related code.
ITracingService tracingService =
(ITracingService)serviceProvider.GetService(typeof(ITracingService));

// Obtain the execution context from the service provider.
IPluginExecutionContext context = (IPluginExecutionContext)
serviceProvider.GetService(typeof(IPluginExecutionContext));

// The InputParameters collection contains all the data passed in the message request.
if (context.InputParameters.Contains("Target") &&
context.InputParameters["Target"] is Entity)
{
// Obtain the target entity from the input parameters.
Entity entity = (Entity)context.InputParameters["Target"];

// Verify that the target entity represents an entity type you are expecting.
// For example, an account. If not, the plug-in was not registered correctly.
if (entity.LogicalName != "contact")
return;

// Obtain the organization service reference which you will need for
// web service calls.
IOrganizationServiceFactory serviceFactory =
(IOrganizationServiceFactory)serviceProvider.GetService(typeof(IOrganizationServiceFactory));
IOrganizationService service = serviceFactory.CreateOrganizationService(context.UserId);

try
{
new ConditionExpression("statecode", ConditionOperator.Equal, 0);


string apiUrl = @"ps.local/.../api";
string apiKey = "LNMSHTZ6X732849VBT7NVMT86SI3QBYA";
PrestaShopWebService webService = new PrestaShopWebService(apiUrl, apiKey);
Task<XElement> customer = webService.Get("customers", 88);
var doc = new XDocument(customer.Result);
doc.Element("prestashop").Element("customer").Element("id").Remove();
Task<XElement> task1 = webService.AddWithUrl("ps.local/.../customers", doc.Root);

}

catch (FaultException<OrganizationServiceFault> ex)
{
throw new InvalidPluginExecutionException("An error occurred in Web Service plugin.", ex);
}

catch (Exception ex)
{
tracingService.Trace("WEB SERVICE PLUGIN : {0}", ex.ToString());
throw;
}
}


}
public class Helpers
{
/// <summary>
/// Returns a canonicalized, escaped string of key=value pairs from a Dictionary of parameters
/// </summary>
/// <param name="options">A Dictionary of options ('filter', 'display' etc)</param>
/// <returns>A canonicalized escaped string of the parameters</returns>
public static string Canonicalize(Dictionary<string, string> options)
{
var builder = new StringBuilder();

foreach (var option in options)
{
if (builder.Length > 0)
{
builder.Append("&");
}

builder.AppendFormat("{0}={1}", option.Key, HttpUtility.UrlEncode(option.Value));
}

return builder.ToString();
}

/// <summary>
/// Returns a valid link rewrite
/// </summary>
/// <param name="text">A string that will be turned into a valid link rewrite</param>
/// <returns>A valid link rewrite</returns>
public static string BuildLinkRewrite(string text)
{
var funnyChars = new string[]
{
",", ".", ":", ";", "!", "´", "%", "$", "£", "€", "@", "&", "?", "/", @"\", "\"", "\'", "#", "<", ">", "(", ")", "[", "]", "«", "»", "®", "™"
};

string link = text.ToLower();

foreach (var fc in funnyChars)
{
link = link.Replace(fc, "");
}

link = link.Replace(' ', '-');
link = link.Replace("æ", "ae");
link = link.Replace("ø", "oe");
link = link.Replace("å", "aa");
link = link.Replace("ü", "u");
link = link.Replace("ö", "o");
link = link.Replace('+', '_');

return link.Trim();
}
}

public interface IPrestaShopWebService
{
Version CurrentVersion { get; }

Task<XElement> Add(string resource, XElement xml);
Task<XElement> AddWithUrl(string url, XElement xml);

Task<XElement> Get(string resource);
Task<XElement> Get(string resource, int id);
Task<XElement> Get(string resource, Dictionary<string, string> options);
Task<XElement> Get(string resource, int? id, Dictionary<string, string> options);
Task<XElement> GetWithUrl(string url);
Task<string> Head(string resource);
Task<string> Head(string resource, int id);
Task<string> Head(string resource, int? id, Dictionary<string, string> options);
Task<string> HeadWithUrl(string url);
Task<XElement> Edit(string resource, int id, XElement xml);
Task<XElement> EditWithUrl(string url, XElement xml);
Task Delete(string resource, int id);
Task Delete(string resource, int[] ids);
Task DeleteWithUrl(string url);
}
public class PrestaShopWebserviceException : Exception
{
public PrestaShopWebserviceException() { }
public PrestaShopWebserviceException(string message) : base(message) { }
public PrestaShopWebserviceException(string message, Exception innerException) : base(message, innerException) { }
}

public class RequestResponse
{
public RequestResponse(int code, string header, string data)
{
this.Code = code;
this.Header = header;
this.Data = data;
}

public int Code { get; set; }
public string Header { get; set; }
public string Data { get; set; }
}

/// <summary>
/// Instantiate the PrestaShopWebService to start executing operations against the PrestaShop Web Service
/// </summary>
[Serializable]
public class PrestaShopWebService : IPrestaShopWebService
{
private readonly string apiUrl;
private readonly string apiKey;
private readonly bool debug;

// Versions of the PrestaShop Web Service supported by this client library
private readonly Version MIN_COMPATIBLE_VERSION;
private readonly Version MAX_COMPATIBLE_VERSION;

// For URL encoding
private readonly Encoding ENCODING;

// Output type (XML or JSON, default: XML)
// Only supported in PS 1.6.0.9 and greater
//private readonly IOFormat IO_FORMAT;

public Version CurrentVersion { get; private set; }

/// <summary>
/// Create instance of PrestaShopWebService
/// </summary>
/// <param name="apiUrl"></param>
/// <param name="apiKey"></param>
/// <param name="ioFormat">Only supported in PS 1.6.0.9 and greater</param>
/// <param name="debug"></param>
public PrestaShopWebService(string apiUrl, string apiKey, bool debug = true)
{
this.MIN_COMPATIBLE_VERSION = new Version("1.4.0.17");
//this.MAX_COMPATIBLE_VERSION = new Version("1.6.0.9");
this.MAX_COMPATIBLE_VERSION = new Version("1.7.5.1");


this.ENCODING = Encoding.UTF8;

this.apiUrl = MakeValidApiUrl(apiUrl);
this.apiKey = apiKey;
this.debug = debug;
}

/// <summary>
/// Add a slash to the url if it does not have it
/// </summary>
/// <param name="url"></param>
/// <returns>Url with slash at the end</returns>
private string MakeValidApiUrl(string url)
{
if (url[url.Length - 1] != '/')
{
url += '/';
}

return url;
}

/// <summary>
/// Take the status code and throw an exception if the server didn't return 200 or 201 code
/// </summary>
/// <param name="statusCode">Status code of an HTTP return</param>
private int CheckStatusCode(HttpStatusCode statusCode)
{
string errorLabel = "This call to PrestaShop Web Services failed and returned an HTTP status of {0}. That means: {1}.";

switch (statusCode)
{
case HttpStatusCode.OK:
return 200;
case HttpStatusCode.Created:
return 201;
case HttpStatusCode.NoContent:
throw new PrestaShopWebserviceException(String.Format(errorLabel, 204, "No content"));
case HttpStatusCode.BadRequest:
throw new PrestaShopWebserviceException(String.Format(errorLabel, 400, "Bad Request"));
case HttpStatusCode.Unauthorized:
throw new PrestaShopWebserviceException(String.Format(errorLabel, 401, "Unauthorized"));
case HttpStatusCode.NotFound:
throw new PrestaShopWebserviceException(String.Format(errorLabel, 404, "Not Found"));
case HttpStatusCode.MethodNotAllowed:
throw new PrestaShopWebserviceException(String.Format(errorLabel, 405, "Method Not Allowed"));
case HttpStatusCode.InternalServerError:
throw new PrestaShopWebserviceException(String.Format(errorLabel, 500, "Internal Server Error"));
default:
throw new PrestaShopWebserviceException(String.Format("This call to PrestaShop Web Services returned an unexpected HTTP status of: {0}", statusCode));
}
}

/// <summary>
/// Take the version and throw an exception if it does not conform to compatible version
/// </summary>
/// <param name="version">Version from HTTP header</param>
private void CheckVersion(Version version)
{
if (version.CompareTo(MIN_COMPATIBLE_VERSION) < 0 || version.CompareTo(MAX_COMPATIBLE_VERSION) > 0)
{
throw new PrestaShopWebserviceException("This library is not compatible with this version of PrestaShop. Please upgrade/downgrade this library");
}

// Set CurrentVersion based on response header from Execute
this.CurrentVersion = version;
}

/// <summary>
/// Execute a request on the PrestaShop Webservice
/// </summary>
/// <param name="url">Full url to call</param>
/// <param name="method">GET, POST, PUT, DELETE, HEAD</param>
/// <param name="document">For PUT (edit) and POST (add) only, the xml sent to PrestaShop</param>
/// <returns>RequestResponse with statuscode, header and data</returns>
private async Task<RequestResponse> Execute(string url, string method, XDocument document = null)
{
int statusCode = 0;
string header = String.Empty;
string data = String.Empty;

//string mediaType = (IOFormat.JSON == this.IO_FORMAT) ? "text/json" : "text/xml";
string mediaType = "text/xml";

using (var handler = new HttpClientHandler { Credentials = new NetworkCredential(this.apiKey, "") })
using (var client = new HttpClient(handler))
{
HttpResponseMessage response;
HttpContent content;

try
{
switch (method.ToUpper())
{
case "GET":
response = await client.GetAsync(url);
break;
case "POST":
response = await client.PostAsync(url, new StringContent(document.ToString(), ENCODING, mediaType));
break;
case "PUT":
response = await client.PutAsync(url, new StringContent(document.ToString(), ENCODING, mediaType));
break;
case "DELETE":
response = await client.DeleteAsync(url);
break;
case "HEAD":
response = await client.SendAsync(new HttpRequestMessage(HttpMethod.Head, url));
break;
default:
throw new PrestaShopWebserviceException("Invalid Http Method provided. GET, POST, PUT, DELETE, HEAD are valid");
}
}
catch (HttpRequestException)
{
throw new PrestaShopWebserviceException("An error occured while sending the request");
}

statusCode = CheckStatusCode(response.StatusCode);
header = response.Headers.ToString() + "\n";

if (response != null)
{
content = response.Content;
data = await content.ReadAsStringAsync();
}

List<string> versionHeaders = response.Headers.GetValues("PSWS-Version").ToList();
if (versionHeaders.Count == 1)
{
Version version = Version.Parse(versionHeaders[0]);
CheckVersion(version);
}

response.Dispose();
}

return new RequestResponse(statusCode, header, data);
}

/// <summary>
/// Loads an XML into an Elem from a String
/// Throws an exception if there is no XML or it won't validate
/// </summary>
/// <param name="xml">The XML string to parse</param>
/// <returns>The parsed XML in an XElement ready to work with</returns>
public XElement Parse(string xml)
{
XDocument xdoc;

if (String.IsNullOrEmpty(xml))
{
throw new PrestaShopWebserviceException("HTTP XML response was empty");
}

try
{
xdoc = XDocument.Parse(xml);
}
catch (Exception e)
{
throw new PrestaShopWebserviceException("HTTP XML response is not parsable: " + e.Message);
}

return xdoc.Descendants("prestashop").Single();
}

/// <summary>
/// Validates that the parameters are all either 'filter', 'display', 'sort', 'limit' or 'schema'
/// Strictly schema isn't a permitted param for HEAD (only GET) but let's leave it
/// Throws a PrestaShopWebServiceException if not
/// </summary>
/// <param name="options">Options to validate</param>
/// <returns>The original parameters if everything is okay</returns>
private Dictionary<string, string> Validate(Dictionary<string, string> options)
{
string[] validOptions = { "display", "sort", "limit", "schema" };

foreach (var option in options.Keys)
{
if (!((IList<string>)validOptions).Contains(option) && option.Substring(0, 6) != "filter")
{
throw new PrestaShopWebserviceException(String.Format("Parameter {0} is not supported", option));
}
}

return options;
}

/// <summary>
/// Executes an empty HEAD request. This is only used to set CurrentVersion.
/// </summary>
/// <returns></returns>
public async Task SetCurrentVersion()
{
await Execute(apiUrl, "HEAD");
}

/// <summary>
/// Add (POST) a resource, self-assembly version
/// </summary>
/// <param name="resource">Type of resource to add</param>
/// <param name="xml">Full XML of new resource</param>
/// <returns>XML response from Web Service</returns>
public async Task<XElement> Add(string resource, XElement xml)
{
string url = this.apiUrl + resource;
return await AddWithUrl(url, xml);
}

/// <summary>
/// Add (POST) a resource, URL version
/// </summary>
/// <param name="url">Full URL for a POST request to the Web Service</param>
/// <param name="xml">Full XML of new resource</param>
/// <returns>XML response from Web Service</returns>
public async Task<XElement> AddWithUrl(string url, XElement xml)
{
RequestResponse response = await Execute(url, "POST", new XDocument(xml));
return Parse(response.Data);
}

/// <summary>
/// Retrieve (GET) a resource, self-assembly version with parameters
/// </summary>
/// <param name="resource">Type of resource to retrieve</param>
/// <returns>XML response from Web Service</returns>
public async Task<XElement> Get(string resource)
{
return await Get(resource, null, null);
}

/// <summary>
/// Retrieve (GET) a resource, self-assembly version without parameters
/// </summary>
/// <param name="resource">Type of resource to retrieve</param>
/// <param name="id">Resource ID to retrieve</param>
/// <returns>XML response from Web Service</returns>
public async Task<XElement> Get(string resource, int id)
{
return await Get(resource, id, null);
}

/// <summary>
/// Retrieve (GET) a resource, self-assembly version with parameters
/// </summary>
/// <param name="resource">Type of resource to retrieve</param>
/// <param name="options">Dictionary of options (one or more of 'filter', 'display', 'sort', 'limit')</param>
/// <returns>XML response from Web Service</returns>
public async Task<XElement> Get(string resource, Dictionary<string, string> options)
{
XElement returnValue = await Get(resource, null, options);

return returnValue;
}

/// <summary>
/// Retrieve (GET) a resource, self-assembly version with parameters
/// </summary>
/// <param name="resource">Type of resource to retrieve</param>
/// <param name="id">Resource ID to retrieve</param>
/// <param name="options">Dictionary of options (one or more of 'filter', 'display', 'sort', 'limit')</param>
/// <returns>XML response from Web Service</returns>
public async Task<XElement> Get(string resource, int? id, Dictionary<string, string> options)
{
string url = this.apiUrl + resource;

if (id != null)
{
url += "/" + id.ToString();
}

if (options != null)
{
url += String.Format("?{0}", Helpers.Canonicalize(Validate(options)));
}

XElement returnValue = await GetWithUrl(url);

return returnValue;
}

/// <summary>
/// Retrieve (GET) a resource, URL version
/// </summary>
/// <param name="url">A URL which explicitly sets the resource type and ID to retrieve</param>
/// <returns>XML response from the Web Service</returns>
public async Task<XElement> GetWithUrl(string url)
{
RequestResponse response = await Execute(url, "GET");
return Parse(response.Data);
}

/// <summary>
/// Head (HEAD) all resources of a type, self-assembly version
/// </summary>
/// <param name="resource">Type of resource to head</param>
/// <returns>Header from Web Service's response</returns>
public async Task<string> Head(string resource)
{
return await Head(resource, null, null);
}

/// <summary>
/// Head (HEAD) an individual resource, self-assembly version
/// </summary>
/// <param name="resource">Type of resource to head</param>
/// <param name="id">Resource ID to head (if not provided, head all resources of this type)</param>
/// <returns>Header from Web Service's response</returns>
public async Task<string> Head(string resource, int id)
{
return await Head(resource, id, null);
}

/// <summary>
/// Head (HEAD) an individual resource or all resources of a type with possible parameters
/// </summary>
/// <param name="resource">Type of resource to head</param>
/// <param name="id">Optional resource ID to head (if not provided, head all resources of this type)</param>
/// <param name="options">Optional Dictionary of parameters (one or more of 'filter', 'display', 'sort', 'limit')</param>
/// <returns>Header from Web Service's response</returns>
public async Task<string> Head(string resource, int? id, Dictionary<string, string> options)
{
string url = this.apiUrl + resource;

if (id != null)
{
url += "/" + id.ToString();
}

if (options != null)
{
url += String.Format("?{0}", Helpers.Canonicalize(Validate(options)));
}

return await HeadWithUrl(url);
}

/// <summary>
/// Head (HEAD) an individual resource or all resources of a type, URL version
/// </summary>
/// <param name="url">Full URL for the HEAD request to the Web Service</param>
/// <returns>Header from Web Service's response</returns>
public async Task<string> HeadWithUrl(string url)
{
RequestResponse response = await Execute(url, "HEAD");
return response.Header;
}

/// <summary>
/// Edit (PUT) a resource, self-assembly version
/// </summary>
/// <param name="resource">Type of resource to update</param>
/// <param name="id">Resource ID to update</param>
/// <param name="xml">Modified XML of the resource</param>
public async Task<XElement> Edit(string resource, int id, XElement xml)
{
string url = this.apiUrl + resource + String.Format("/{0}", id);
return await EditWithUrl(url, xml);
}

/// <summary>
/// Edit (PUT) a resource, URL version
/// </summary>
/// <param name="url">A URL which explicitly sets the resource type and ID to edit</param>
/// <param name="xml">Modified XML of the resource</param>
public async Task<XElement> EditWithUrl(string url, XElement xml)
{
RequestResponse response = await Execute(url, "PUT", xml.Document);
return Parse(response.Data);
}

/// <summary>
/// Delete (DELETE) a resource, self-assembly version supporting one ID
/// This version takes a resource type and a single ID to delete
/// </summary>
/// <param name="resource">The type of resource to delete (e.g. "orders")</param>
/// <param name="id">An ID of this resource type, to delete</param>
public async Task Delete(string resource, int id)
{
string url = this.apiUrl + resource + String.Format("/{0}", id);
await DeleteWithUrl(url);
}

/// <summary>
/// Delete (DELETE) a resource, self-assembly version supporting multiple IDs
/// This version takes a resource type and an array of IDs to delete
/// </summary>
/// <param name="resource">The type of resource to delete (e.g. "orders")</param>
/// <param name="ids">An array of IDs of this resource type, to delete</param>
public async Task Delete(string resource, int[] ids)
{
string url = this.apiUrl + resource + String.Format("/?id=[{0}]", String.Join(",", ids));
await DeleteWithUrl(url);
}

/// <summary>
/// Delete (DELETE) a resource, URL version
/// </summary>
/// <param name="url">A URL which explicitly sets resource type and resource ID</param>
public async Task DeleteWithUrl(string url)
{
await Execute(url, "DELETE");
}
}
}
public static class TypeTester
{
#region "Specific value types"
public static bool IsBirthDate(string value)
{
return Regex.IsMatch(value, "^([0-9]{4})-((0?[1-9])|(1[0-2]))-((0?[1-9])|([1-2][0-9])|(3[01]))( [0-9]{2}:[0-9]{2}:[0-9]{2})?$");
}

public static bool IsColor(string value)
{
return Regex.IsMatch(value, "^(#[0-9a-fA-F]{6}|[a-zA-Z0-9-]*)$");
}

public static bool IsEmail(string value)
{
return Regex.IsMatch(value, @"^[a-z0-9!#$%&\'*+\/=?^`{}|~_-]+[.a-z0-9!#$%&\'*+\/=?^`{}|~_-]*@[a-z0-9]+[._a-z0-9-]*\.[a-z0-9]+$");
}

public static bool IsImageSize(string value)
{
return Regex.IsMatch(value, "^[0-9]{1,4}$");
}

public static bool IsLanguageCode(string value)
{
return Regex.IsMatch(value, "^[a-zA-Z]{2}(-[a-zA-Z]{2})?$");
}

public static bool IsLanguageIsoCode(string value)
{
return Regex.IsMatch(value, "^[a-zA-Z]{2,3}$");
}

public static bool IsMd5(string value)
{
return Regex.IsMatch(value, "^[a-f0-9A-F]{32}$");
}

public static bool IsNumericIsoCode(string value)
{
return Regex.IsMatch(value, "^[0-9]{2,3}$");
}

public static bool IsPasswd(string value)
{
return Regex.IsMatch(value, @"^[.a-zA-Z_0-9-!@#$%\^&*()]{5,32}$");
}

public static bool IsPasswdAdmin(string value)
{
return Regex.IsMatch(value, @"^[.a-zA-Z_0-9-!@#$%\^&*()]{8,32}$");
}

public static bool IsPhpDateFormat(string value)
{
return Regex.IsMatch(value, "^[^<>]+$");
}

public static bool IsReference(string value)
{
return Regex.IsMatch(value, "^[^<>;={}]*$");
}

public static bool IsUrl(string value)
{
return Regex.IsMatch(value, @"^[~:#,%&_=\(\)\.\? \+\-@\/a-zA-Z0-9]+$");
}
#endregion

#region "Names"
public static bool IsCatalogName(string value)
{
return IsGenericName(value);
}

public static bool IsCarrierName(string value)
{
return IsGenericName(value);
}

public static bool IsConfigName(string value)
{
return Regex.IsMatch(value, "^[a-zA-Z_0-9-]+$");
}

public static bool IsGenericName(string value)
{
return Regex.IsMatch(value, "^[^<>;=#{}]*$");
}

public static bool IsImageTypeName(string value)
{
return Regex.IsMatch(value, "^[a-zA-Z0-9_ -]+$");
}

public static bool IsName(string value)
{
return Regex.IsMatch(value, "^[^0-9!<>,;?=+()@#\"°{}_$%:]*$");
}

public static bool IsTplName(string value)
{
return Regex.IsMatch(value, "^[a-zA-Z0-9_-]+$");
}
#endregion

#region "Location"
public static bool IsAddress(string value)
{
return Regex.IsMatch(value, "^[^!<>?=+@{}_$%]*$");
}

public static bool IsCityName(string value)
{
return Regex.IsMatch(value, "^[^!<>;?=+@#\"°{}_$%]*$");
}

public static bool IsCoordinate(string value)
{
return Regex.IsMatch(value, @"/^\-?[0-9]{1,8}\.[0-9]{1,8}$");
}

public static bool IsMessage(string value)
{
return Regex.IsMatch(value, "[<>{}]");
}

public static bool IsPhoneNumber(string value)
{
return Regex.IsMatch(value, "/^[+0-9. ()-]*$");
}

public static bool IsPostCode(string value)
{
return Regex.IsMatch(value, "^[a-zA-Z 0-9-]+$");
}

public static bool IsStateIsoCode(string value)
{
return Regex.IsMatch(value, "^[a-zA-Z0-9]{2,3}((-)[a-zA-Z0-9]{1,3})?$");
}

public static bool IsZipCodeFormat(string value)
{
return Regex.IsMatch(value, "^[NLCnlc -]+$");
}
#endregion

#region "Products"
public static bool IsAbsoluteUrl(string value)
{
return Regex.IsMatch(value, @"^https?:\/\/[:#%&_=\(\)\.\? \+\-@\/a-zA-Z0-9]+$");
}

public static bool IsDniLite(string value)
{
return Regex.IsMatch(value, "^[0-9A-Za-z-.]{1,16}$");
}

public static bool IsEan13(string value)
{
return Regex.IsMatch(value, "^[0-9]{0,13}$");
}

public static bool IsLinkRewrite(string value)
{
return Regex.IsMatch(value, "^[_a-zA-Z0-9-]+$");
}

public static bool IsUpc(string value)
{
return Regex.IsMatch(value, "^[0-9]{0,12}$");
}
#endregion
}
}

*This post is locked for comments

Under review

Thank you for your reply! To ensure a great experience for everyone, your content is awaiting approval by our Community Managers. Please check back later.

Helpful resources

Quick Links

December Spotlight Star - Muhammad Affan

Congratulations to a top community star!

Top 10 leaders for November!

Congratulations to our November super stars!

Tips for Writing Effective Suggested Answers

Best practices for providing successful forum answers ✍️

Leaderboard

#1
André Arnaud de Calavon Profile Picture

André Arnaud de Cal... 291,269 Super User 2024 Season 2

#2
Martin Dráb Profile Picture

Martin Dráb 230,198 Most Valuable Professional

#3
nmaenpaa Profile Picture

nmaenpaa 101,156

Leaderboard

Featured topics

Product updates

Dynamics 365 release plans