dotConnect for Dynamics 365 Documentation
In This Topic
    Entity Framework Tutorial
    In This Topic

    This tutorial guides you through the process of creating a simple application powered by ADO.NET Entity Framework. In less than 5 minutes you will have a ready-to-use data access layer for your business objects.

    Please note that this tutorial is not applicable for Entity Framework Core. It is intended for previous Entity Framework versions.

    In this walkthrough:

    Introducing the ADO.NET Entity Framework

    ADO.NET Entity Framework is an object-relational mapping (ORM) framework for the .NET Framework. It is designed to enable developers to create data access applications by programming against a conceptual application model instead of programming directly against a relational storage schema. The goal is to decrease the amount of code and maintenance required for data-oriented applications.

    Requirements

    In order to connect to Dynamics 365 server you need the corresponding Dynamics 365 connection parameters, dotConnect for Dynamics 365 installed and IDE running.

    For Entity Framework 6, you will also need a NuGet Visual Studio extension installed since it is used for adding EntityFramework NuGet package. Alternatively you may create model for Entity Framework v4, which don't require NuGet, in this tutorial.

    In this sample we will create a simple console application. It could be any other project type as well, but for simplicity's sake we'll use console project throughout the tutorial. Start Visual Studio and create a new console application.

    Entity Framework v6

    The following actions are required if you want to create an Entity Framework v6 model.

    Open the Package Manager Console window and execute the following command in it.

    install-package EntityFramework

    After this add the following line:

    <provider invariantName="Devart.Data.Dynamics" type="Devart.Data.Dynamics.Entity.DynamicsEntityProviderServices, 
    Devart.Data.Dynamics.Entity.EF6, Version=1.0.0.0, Culture=neutral, PublicKeyToken=09af7300eec23701" />
    

    to the entityFramework -> providers section.

    <entityFramework>
        <providers>
          <provider invariantName="Devart.Data.Dynamics" type="Devart.Data.Dynamics.Entity.DynamicsEntityProviderServices, 
          Devart.Data.Dynamics.Entity.EF6, Version=1.0.0.0, Culture=neutral, PublicKeyToken=09af7300eec23701" />
        </providers>
    </entityFramework>
    

    Note: replace 1.0.0.0 with the actual assembly version.

    After this you need to rebuild the project before running the EDM wizard.

    Generating Model from Database

    1. In the Solution Explorer right-click on the project and choose Add | New Item.
    2. In the dialog select ADO.NET Entity Data Model, click Add. This launches Entity Data Model Wizard.
    3. In the wizard select Generate from database, click Next.
    4. Pick an existing dotConnect for Dynamics 365 connection string or create a new one. When creating a new connection select Dynamics 365 Data Source in the Data Source list, then dotConnect for Dynamics 365 in the Data provider list. See the screenshot below.
      EDM wizard - connection
    5. Agree to include the sensitive data in the connection string.
    6. In the Save entity connection settings... box type DynamicsEntities. This will be the name of the main data access class. Click Next.
    7. Select database objects that will be used in the model. Enter DynamicsModel in the Model Namespace box.
      EDM wizard - selecting objects
    8. Click Finish. The model will be generated and opened in EDM Designer.
      Entity data model

    The model you've just generated is ready to use. Its name is Model1.edmx, unless you changed it in the step 2. You can inspect it visually in the designer or take a look behind the scenes with XML Editor.

    The wizard creates classes for all selected tables that represent entities. It also creates a descendant of System.Data.Objects.DbContext class, which controls the connection to the database, and the whole data flow. This class includes properties and methods named after your database objects. You will use these members to retrieve and modify data in the context. The code is contained in an autogenerated file Model1.Designer.cs (Model1.Designer.vb).

    Note that if you have an association between two properties of non-coinciding numeric types, you can manually change both conceptual and storage types to the type that will be wide enough to include data for each property.

    Querying Data

    All Entity Framework operations are executed through a DbContext descendant (default since Visual Studio 2012) or through a ObjectContext descendant. In our tutorial, it's a DbContext descendant, which is named DynamicsEntities. To retrieve data you have to first create an instance of the context, then prepare a query with LINQ to Entities or EntitySQL or their mix, and then access the object returned by the query, which may be a collection of objects or a single object.

    Let's try performing a LINQ to Entities query against our model. Add the following block of code to the method Main:

    
        DynamicsEntities context = new DynamicsEntities();
        var query = from it in context.campaign
                    orderby it.name
                    select it;
    
        foreach (campaign a in query)
            Console.WriteLine("{0} | {1} | {2}", a.name, a.actualstart, a.actualend);
    
        Console.ReadLine();
    
    
        Dim context As New DynamicsEntities
        Dim query = From it In context.campaign
            Order By it.name
            Select it
    
        Dim a As campaign
        For Each a In query
            Console.WriteLine("{0} | {1} | {2}", a.name, a.actualstart, a.actualend)
        Next
    
        Console.ReadLine()
    
    
    

    As simple as that. You prepare a query and then iterate through it as you would do with a usual collection of objects. The database interaction is performed by Entity Framework in the background. Now let's see who is who in this code sample.

    Here is the project's output in the console:

    Output

    Note that the LINQ to Entities query code just describes the query. It does not execute it. This approach is known as deferred execution.

    Now let's query data from two objects united with a relation. Replace the old code with this:

       // Add references to the following assemblies in your project:
       // Devart.Data.dll,  Devart.Data.SqlShim.dll, Devart.Data.Dynamics.dll, Devart.Data.Dynamics.Entity.EF6.dll
       var monitor = new Devart.Data.Dynamics.DynamicsMonitor() { IsActive = true };
    
        DynamicsEntities context = new DynamicsEntities();
        var query = from it in context.campaign.Include("campaignactivity")
                    orderby it.name
                    select it;
    
        foreach (campaign account in query)
            Console.WriteLine("{0} | {1} | {2}",
                account.name, account.description,
                account.campaignactivity.FirstOrDefault() == null ? "null" : account.campaignactivity.FirstOrDefault().subject
                );
    
        Console.ReadLine();
    
    
       ' Add the references to the following assemblies in your project:
       ' Devart.Data.dll, Devart.Data.SqlShim.dll, Devart.Data.Dynamics.dll, Devart.Data.Dynamics.Entity.EF6.dll
       Dim monitor As New Devart.Data.Dynamics.DynamicsMonitor()
       monitor.IsActive = True
    
        Dim context As New DynamicsEntities
        Dim query = From it In context.campaign.Include("campaignactivity")
            Order By it.name
            Select it
    
        For Each account As campaign In query
            Console.WriteLine("{0} | {1} | {2}", account.name, account.description,
                                If(account.campaignactivity.FirstOrDefault() Is Nothing, "null", account.campaignactivity.FirstOrDefault().subject))
        Next
    
        Console.ReadLine()
    
    
    

    This sample is much like the previous one, with exception that it adds the Include method that instructs the query to retrieve data from one more object.

    Inserting New Data

    What earlier was adding rows to tables, now is just adding new objects to context collections. When you are ready to send the changes to the database, call the SaveChanges() method of the context. Before doing this, you must first set all properties that do not support null (Nothing) values. The SaveChanges() method generates and executes commands that perform the equivalent INSERT, UPDATE, or DELETE statements against the data source.

    By Microsoft design, DbContext.SaveChanges() should work within a distributed transaction, but it is not supported by API of Dynamics 365. That's why SaveChanges() will submit all DML operations (till the failed one, if any). This code snippet includes enabled DynamicsMonitor for tracing the SQL statements sent to data source.

    In this tutorial we will create new campaign and campaignactivity objects. For the sake of this tutorial, we need to set the StoreGeneratedPattern property to Computed for the following entity properties in the model in order to allow Dynamics 365 to generate values for these fields:

    Let's add new objects to the data source. Replace the old code with this:

        // Add references to the following assemblies in your project:
        // Devart.Data.dll,  Devart.Data.SqlShim.dll, Devart.Data.Dynamics.dll, Devart.Data.Dynamics.EF6.dll
    
        DynamicsEntities context = new DynamicsEntities();
    
        // Create a new campaignactivity
        var a = new campaignactivity();
        a.subject = "new subject";
        a.prioritycode = "Normal";
        a.typecode = "Research";
        context.campaignactivity.Add(a);
    
        // Create a new campaign
        var c = new campaign();
        c.name = "new campaign";
        c.typecode = "Advertisement";
    
        // Associate the new campaignactivity with the new campaign
        c.campaignactivity.Add(a);
        context.campaign.Add(c);
    
        // Send the changes to the database.
        // Until you do it, the changes are cached on the client side.
    
        context.SaveChanges();
    
        // Request the new campaign from the database
        var query = from it in context.campaign.Include("campaignactivity")
                    where it.name == "new campaign"
                    select it;
    
        // Since we query for a single object instead of a collection, we can use the method First()
        campaign myCampaign = query.First();
        Console.WriteLine("{0} | {1} ",
            myCampaign.campaignactivity.FirstOrDefault().subject, myCampaign.name);
        Console.ReadLine();
    
    
        ' Add the references to the following assemblies in your project:
        ' Devart.Data.dll, Devart.Data.SqlShim.dll, Devart.Data.Dynamics.dll, Devart.Data.Dynamics.EF6.dll
        Dim config = Devart.Data.Dynamics.Entity.Configuration.DynamicsEntityProviderConfig.Instance
        config.DmlOptions.InsertNullBehaviour = Devart.Data.Dynamics.Entity.Configuration.InsertNullBehaviour.Omit
    
        Dim context As New DynamicsEntities
    
        'Create a new campaignactivity
        Dim a = New campaignactivity()
        a.subject = "new subject"
        a.prioritycode = "Normal"
        a.typecode = "Research"
        context.campaignactivity.Add(a)
    
        'Create a new campaign
        Dim c = New campaign()
        c.name = "new campaign"
        c.typecode = "Advertisement"
    
        'Associate the new campaignactivity with the new campaign
        c.campaignactivity.Add(a)
        context.campaign.Add(c)
    
        'Send the changes to the database.
        'Until you do it, the changes are cached on the client side.
    
        context.SaveChanges()
    
        'Request the new campaign from the database
        Dim query = From it In context.campaign.Include("campaignactivity")
                    Where it.name = "new campaign"
                    Select it
    
        ' Since we query for a single object instead of a collection, we can use the method First()
        Dim myCampaign = query.First()
        Console.WriteLine("{0} | {1} ",
            myCampaign.campaignactivity.FirstOrDefault().subject, myCampaign.name)
        Console.ReadLine()
    
    
    

    The Add methods and others are automatically generated in the context. Such methods exist for every entity in your model.

    Updating Data

    Entity instances are modified as usual. The only thing to remember is that you have to invoke the SaveChanges() method to send the data to the database.

    Append the following block to the existing code and launch the project:

        myCampaign.name = "edited name";
        context.SaveChanges();
    
    
        myCampaign.name = "edited name"
        context.SaveChanges()
    
    
    

    Deleting Data

    To extract an instance from a context use the DeleteObject method of the context. The object is removed from the collection of its type, but not destroyed. To delete the object's data from the database invoke the SaveChanges() method.

    You can do this with a block of code like the following:

        context.campaign.Remove(myCampaign);
        context.campaignactivity.Remove(myCampaignactivity);
        context.SaveChanges();
    
    
        context.campaign.Remove(myCampaign)
        context.campaignactivity.Remove(myCampaignactivity)
        context.SaveChanges()
    
    
    

    Additional Information

    Now that you can perform the basic data manipulation with Entity Framework, you can move on to some advanced topics.

    We recommend you to use Entity Developer (Devart Entity Model, *.edml) instead of EDM Designer (ADO.NET Entity Data Model, *.edmx) because it is adjusted for working with Dynamics 365 and offers advanced functionality. Additionally, Entity Developer registers Entity Framework v6 providers in app.config automatically and offers advanced visual designer and support for Database First / Model First approaches for EF Core.

    Here are some useful links to MSDN:

    For hands-on experience download the separate Entity Framework Query Samples (EF1/EF4/EF5/EF6) package or use samples shipped with dotConnect for Dynamics 365. You can access the samples from the Start menu.

    To understand deeper the works of Entity Framework engine you can watch the generated SQL statements in dbMonitor.

    See Also

    Entity Framework section  | Entity Framework Support Overview