Dynamics 365 – Provedor de Dados Customizado/Custom Data Provider (Parte 1/3)


Olá pessoal,

Nativamente existem apenas dois tipos diferentes de Provedores de Dados (Data Providers) disponíveis para as Entidades Virtuais:

  • Solution Component Data Provider
  • OData v4 Data Provider

É claro que fazer uso deles, facilita bem uma integração. Porém muitas vezes, precisamos consultar outras fontes de dados com formatos/protocolos diferentes ou simplesmente não é viável modificar a API para um formato mais moderno como OData v4. Assim, precisamos partir para a criação de um provedor de dados personalizado. Como criá-lo, registrá-lo e testá-lo, serão os objetivos desta série de três posts!

Um provedor customizado não é nada mais do que um tipo de plug-in que implementa as mensagens de Retrieve (individual / form) e RetrieveMultiple (lista / grids). O plugin irá acessar a API externa, transformando o retorno da consulta em uma BusinessEntity para o Retrieve e uma BusinessCollection para o RetrieveMultiple (ambas as classes são tipos de objeto que o Dynamics já conhece) e deixando o Dynamics se encarregar de “montar” a camada visual para nós!

Para aqueles que ainda não viram meu post anterior, recomendo a leitura, pois existem tópicos que eu irei utilizar neste post.

Novamente irei utilizar a entidade Contatos e a entidade virtual ExternalActivity que utilizei no meu último post.

Bom, vamos ao que interessa!

API

Se você já possui uma API que segue as limitações das entidades virtuais, vá para a próxima etapa “Provedor de Dados Customizado“.

Já que estamos criando um provedor customizado, irei criar um WCF que retorna JSON, talvez seja o cenário mais provável para as integrações entre entidades virtuais e o Dynamics.

Aqui vai um printscreen to projeto no Visual Studio para ajudar a se localizar:

Abaixo minha Model (ExternalActivity.cs), veja que estou usando a mesma do post anterior, com isso não precisarei criar uma nova entidade virtual:

using System;

namespace DemoJSON
{
    public class ExternalActivity
    {
        public Guid ExternalActivityId { get; set; }

        public int ExternalActivityKey { get; set; }

        public Guid ContactId { get; set; }

        public string Subject{ get; set; }

        public string Details { get; set; }

        public string From { get; set; }

        public string To { get; set; }

    }
}

Agora a Interface de Serviço (IJSONService.cs):

using System.Collections.Generic;
using System.ServiceModel;
using System.ServiceModel.Web;

namespace DemoJSON
{
    // NOTE: You can use the "Rename" command on the "Refactor" menu to change the interface name "IService1" in both code and config file together.
    [ServiceContract]
    public interface IJSONService
    {
        [OperationContract]
        [WebInvoke(UriTemplate = "ExternalActivity/{externalActivityId}", Method = "GET", BodyStyle = WebMessageBodyStyle.WrappedRequest, RequestFormat = WebMessageFormat.Json, ResponseFormat = WebMessageFormat.Json)]
        List<ExternalActivity> ExternalActivity(string externalActivityId);

        [OperationContract]
        [WebInvoke(UriTemplate = "ExternalActivities/{contactId}", Method = "GET", BodyStyle = WebMessageBodyStyle.WrappedRequest, RequestFormat = WebMessageFormat.Json, ResponseFormat = WebMessageFormat.Json)]
        List<ExternalActivity> ExternalActivities(string contactId);
    }
}

Acima, existem dois métodos, um para ser utilizado no Retrieve (ExternalActivity, que recebe como parâmetro externalActivityId) e outro para o RetrieveMultiple (ExternalActivities, que recebe como parâmetro contactId).

Vamos a implementação da interface com sua lógica (JSONService.cs):

using System;
using System.Collections.Generic;
using System.Linq;

namespace DemoJSON
{
    public class JSONService : IJSONService
    {
        public List<Guid> listExternalActivitiesIDs = new List<Guid>();

        public JSONService()
        {
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED041"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED042"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED043"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED044"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED045"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED046"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED047"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED048"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED049"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED050"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED051"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED052"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED053"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED054"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED055"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED056"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED057"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED058"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED059"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED060"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED061"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED062"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED063"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED064"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED065"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED066"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED067"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED068"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED069"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED070"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED071"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED072"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED073"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED074"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED075"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED076"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED077"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED078"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED079"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED080"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED081"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED082"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED083"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED084"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED085"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED086"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED087"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED088"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED089"));
            listExternalActivitiesIDs.Add(new Guid("86B77A42-2EBF-4144-A570-0E51B69ED090"));
        }

        public List<ExternalActivity> ExternalActivity(string paramExternalActivityId)
        {
            List<ExternalActivity> ExternalActivities = new List<ExternalActivity>();

            for (int i = 0; i < 1000; i++)
            {
                Guid contactId = Guid.NewGuid();
                Guid externalActivityId = Guid.NewGuid();

                if (i < 50)
                {
                    contactId = new Guid("C4BA7D70-347B-E811-A961-000D3AE05F14");
                    externalActivityId = listExternalActivitiesIDs[i];
                }

                ExternalActivities.AddRange(new List<ExternalActivity>
                {
                    new ExternalActivity()
                    {
                        ExternalActivityId = externalActivityId,
                        ExternalActivityKey = i,
                        ContactId = contactId,
                        From = RandomString(10),
                        To = RandomString(10),
                        Subject= RandomString(20),
                        Details = RandomString(100),
                    }
                });
            }

            if (!string.IsNullOrEmpty(paramExternalActivityId))
            {
                return ExternalActivities.Where(m => m.ExternalActivityId.Equals(new Guid(paramExternalActivityId))).ToList();
            }
            else
            {
                return ExternalActivities;
            }
        }

        public List<ExternalActivity> ExternalActivities(string paramContactId)
        {
            List<ExternalActivity> ExternalActivities = new List<ExternalActivity>();

            for (int i = 0; i < 1000; i++)
            {
                Guid contactId = Guid.NewGuid();
                Guid externalActivityId = Guid.NewGuid();

                if (i < 50)
                {
                    contactId = new Guid("C4BA7D70-347B-E811-A961-000D3AE05F14");
                    externalActivityId = listExternalActivitiesIDs[i];
                }

                ExternalActivities.AddRange(new List<ExternalActivity>
                {
                    new ExternalActivity()
                    {
                        ExternalActivityId = externalActivityId,
                        ExternalActivityKey = i,
                        ContactId = contactId,
                        From = RandomString(10),
                        To = RandomString(10),
                        Subject= RandomString(20),
                        Details = RandomString(100),
                    }
                });
            }

            if (!string.IsNullOrEmpty(paramContactId))
            {
                return ExternalActivities.Where(m => m.ContactId.Equals(new Guid(paramContactId))).ToList();
            }
            else
            {
                return ExternalActivities;
            }
        }

        static Random random = new Random();

        public static string RandomString(int length)
        {
            const string chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
            return new string(Enumerable.Repeat(chars, length).Select(s => s[random.Next(s.Length)]).ToArray());
        }
    }
}

Linhas destacadas (9 à 63, 77 e 117): vejam que criei uma lista fixa (hardcoded) do objeto ExternalActivity, para garantir que os métodos irão possuir os mesmos IDs, na “vida real”, você já deve possuir os IDs únicos!

O método ExternalActivity recebe um id de ExternalActivity e retorna uma lista de ExternalActivity, embora o método retorne apenas uma lista com um item, mantive desta forma para podermos reaproveitar parte do código quando estivermos criando o provedor de dados para o Retrieve e RetrieveMultiple.

Já o método ExternalActivities recebe um id de um Contato e retorna uma lista de ExternalActivity.

Bom fim, a configuração do Web.config:

<?xml version="1.0"?>
<configuration>

  <appSettings>
    <add key="aspnet:UseTaskFriendlySynchronizationContext" value="true" />
  </appSettings>
  <system.web>
    <compilation debug="true" targetFramework="4.5.2" />
    <httpRuntime targetFramework="4.5.2"/>
  </system.web>
  <system.serviceModel>
    <behaviors>
      <serviceBehaviors>
        <behavior name="ServiceBehavior">
          <serviceMetadata httpGetEnabled="true"/>
          <serviceDebug includeExceptionDetailInFaults="true"/>
        </behavior>
      </serviceBehaviors>
      <endpointBehaviors>
        <behavior name="ServiceAspNetAjaxBehavior">
          <webHttp />
        </behavior>
      </endpointBehaviors>
    </behaviors>
    <serviceHostingEnvironment aspNetCompatibilityEnabled="true"/>
    <services>
      <service behaviorConfiguration="ServiceBehavior" name="DemoJSON.JSONService">
        <endpoint address="" behaviorConfiguration="ServiceAspNetAjaxBehavior" binding="webHttpBinding" contract="DemoJSON.IJSONService">
          <identity>
            <dns value="localhost" />
          </identity>
        </endpoint>
        <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
      </service>
    </services>
  </system.serviceModel>


  <system.webServer>
    <modules runAllManagedModulesForAllRequests="true"/>
    <!-- To browse web app root directory during debugging, set the value below to true. Set to false before deployment to avoid disclosing web app folder information. -->
    <directoryBrowse enabled="true"/>
  </system.webServer>

</configuration>

API em JSON criada, vamos para o provedor de dados customizado! Este será meu próximo post, não percam!

Para maiores detalhes, seguem os documentos oficiais:

Custom virtual entity data providers

Sample: Generic virtual entity data provider plug-in

[]’s,

Tiago

Deixe um comentário

Preencha os seus dados abaixo ou clique em um ícone para log in:

Logo do WordPress.com

Você está comentando utilizando sua conta WordPress.com. Sair /  Alterar )

Foto do Facebook

Você está comentando utilizando sua conta Facebook. Sair /  Alterar )

Conectando a %s

Este site utiliza o Akismet para reduzir spam. Saiba como seus dados em comentários são processados.