Entity Framework Datacontract Serialization

  1. Is there a series of MSDN articles which describe how to utilize WCF Serialization for Entity Framework Entities? I would like to expose the entities which are part of my Entity Data Model over WCF Services.
  2. The attributes of DataContract and DataMember are needed for the Entity Framework to use these objects. Notice on the product object I have a virtual list of sale objects, making this virtual allows for entity framework to use lazy loading.

Serialization of Entity Framework objects with One to Many Relationship. Consider using a DataContractResolver or add any types not known statically to the list of known types - for example, by using the KnownTypeAttribute attribute or by adding them to the list of known types passed to DataContractSerializer.

7 Dec 2016CPOL
Convert a class library of POCO classes to datacontracts for use with WCF without having to write DataContract and DataMember attributes directly to the POCO classes.

Introduction

In a previous article titled Use EF Power Tool to generate EF POCO classes, a class library containing POCO classes was autogenerated from an existing SQL server database using the EF Power Tool. The purpose of this article is to demonstrate how this class library containing POCO classes can be converted into DataContracts for use in a WCF service application, without the need to modify the autogenerated POCO classes manually. For completeness, there is also a brief section on how to create DataContracts from a class library of POCOs not produced using the EF Power Tool.

Background

When using the EF Power Tool to generate POCO classes, mappings, and DBContext from an existing database to write an application using the Code First approach, there may a situation when some or all of the POCO classes need to be used in a WCF service. In order to do this, it would be necessary to make changes to the POCO classes manually, and add DataContract and DataMember attributes. However, the changes would be overwritten the next time the EF Power tool was executed to refresh any changes made to the database schema. Some way of producing DataContracts for a WCF Service, from existing POCO classes, would be useful without having to change the POCO classes manually. The article explains how this is possible.

The process is detailed in the following two sections:

Generate DataContracts from POCO Class Library Produced Using the EF Power Tool

In order to produce the DataContracts from a class library of POCO classes created by the EF power tool, the following 3 steps are involved:

  1. Modify the T4 Template for the EF Power tool and add additional attributes
  2. Create an XSD file from the POCO class library using xsd.exe tool
  3. Use SVCUtil.exe to convert the xsd file into datacontracts.

The diagram below provides a summary of these steps:

The first zip file contains a sample VS 2013 solution which will be used to walk though this process.

The second zip file contains the same VS 2013 solution with the suggested changes already applied in order to produce the datacontracts. It is attached for the purpose of reference only.

Let's open the solution contained in the first zip file in Visual Studio. The solution consists of three projects, i.e.:

  • DataContracts - Autogenerated datacontracts file will be placed here
  • The WCF Service Library - uses the DataContracts project and is only a scaffold used for this article.
  • Models - the existing models created using the EF Power Tool to reverse engineer an existing database.

The above screenshot shows the Models project, the DataContracts project, and a WCF Service library project which references the DataContracts project. The Models project is actually not referenced by the Service Library in this example. In a more realistic application, it would be setup as follows:

Here, the Models project is referenced by the Repositories project for performing the database operations requested by the WCF Service Library. In this article, we will keep things simple. The focus is the Models project, and how this can be converted into the DataContracts project, rather than delving into the Repositories and CRUD operations using the POCO library in the Models project.

1. Modify the T4 Template for the EF Power Tool and Add Additional Attributes

EF Power Tool uses T4 Templates to autogenerate the POCO classes, DBContext, and mappings. The default autogenerated code in the Models class library cannot be converted into an xsd file without producing serialization errors for properties of type ICollection and DBSet. The ICollection or a List would normally be declared in a servicecontract. The DBSet type would not need to be served to a client. Therefore, both data types will be excluded from the resulting XSD file. In order to do this, we will directly modify the T4 templates as shown below and regenerate the POCO class library which was created in Use EF Power Tool to generate EF POCO classes.

1.1 Right-click on the Models project and select -> Entity Framework -> Customize Reverse Engineer Templates.

If the menu option fails to appear, then EF Power Tool has not been installed in Visual Studio. This can be installed from the following location:

The Models project will now appear as below:

A new folder named CodeTemplates has been created. This exposes the three T4 templates used by the EF Power tool to create the DBContext, POCO classes, and Mappings.

1.2 Modify the Entity.tt file and add the following at line 61.

1.3 Modify theContext.ttfile and add the following at line 30:

The text will now appear as:

1.4 In order to apply the changes in the T4 templates to the POCO classes, it is necessary to generate them again. Right click on the Models project and select Reverse Engineer Code First as shown below.

Now key in the login details for SQL Server.

Click the Advanced button.

Set the highlighted property to True and click the OK button to close the Advanced Properties window. Click the OK button in the Connection properties window. This will regenerate the files in the Models folder with the changes applied to the T4 Templates.

Once the process is completed, the User.cs file will have the following change:

The UsersContext.csalso has the XmlIgnore attribute applied to all the DbSet properties.

The [System.Xml.Serialization.XmlIgnore] prevents a property from being serialized. The changes to the T4 templates has targeted specific type of properties from being serialized such as ICollection and Dbset. The DataContractSerializer of the xsd tool is unable to serialize an interface, therefore the ICollection would need to be changed to a List in order to avoid an error. In this example, I have chosen to prevent the ICollection from being serialized. DBSet would also fail to serialize as it inherits from the IEnumerable interface.

By modifying the T4 template directly to add the [System.Xml.Serialization.XmlIgnore] attribute, the change to the templates would only need to be applied once. When the class library is generated again, the [System.Xml.Serialization.XmlIgnore] attribute will be applied automatically to the properties which are not to be serialized.

Let's build the Models class library project. During the build, you may see the following error:

This error can be ignored because when the project builds, it does not use the T4 templates. The templates are only used when the EF power is run to generate the POCO classes.

The models.dll file is created in the bindebugfolder, and will be used in the next step for serialization.

2. Create an XSD file from the POCO class library using xsd.exe tool

An xsd file of the POCO classes will be created using the xsd.exe tool from VS command prompt. The .xsd file will then be added to the DataContracts project, and the SVCUtil.exe tool will be used to produce the .cs file containing the datacontracts.

2.1 Go to theVS command prompt by loading the following window from the Visual Studio 2013 tools folder:

Entity Framework Versions

Select the highlighted option above. Note: The procedure may be different depending on the version of VS you have installed.

Change directory to the directory containing the models.dll file. This file is contained in the bindebug folder as shown above.

Enter the following command:

The file namedSchema0.xsd is generated in the bindebug folder.

Copy and add the Schema0.xsd file into the DataContracts project as highlighted below.

3. Use SVCUtil.exe to convert the xsd file into datacontracts.

Open theVisual Studio Command prompt and change directory to the application folder of the DataContracts project.

Key in the following command:

Versions

The command will generate a file named DataContracts.cs. This file can be included in the DataContractLayer project from Visual Studio and used by a servicecontract to serve data to the client.

The schema0.xsd can also be given to a client which consumes the service and allow the client developer to generate the datacontracts using the SVCUtil.exe, or the DataContracts.cs file can be sent to the developer who is writing the client application.

The DataContracts.cs file contains the datacontracts for all the class files in the Models projects. When the file is opened, the first datacontract shown is for the Group model as seen below. Notice that the ICollection property has been excluded from the datacontract as it was marked with the [System.Xml.Serialization.XmlIgnore] attribute in the model.

Also, if you were to scroll down in the file and locate the UsersContext datacontract, this will show an empty class because we are only interested in producing a datacontract for simple POCO classes. As mentioned earlier, the DBContext will not be served to a Client.

Entity Framework Datacontract Serialization

The actual model for the above datacontract is shown below with the [System.Xml.Serialization.XmlIgnore] attribute applied to the DBSet return type properties.When this is serialized by xsd.exe and passed through svcutil.exe, the properties with the XmlIgnore attribute are excluded from the resulting datacontract.

This concludes the section on how to convert a class library project of POCO's generated from the EF Power tool to datacontracts. The above approach can also be used to generate datacontracts for POCOs which were created from the EF designer, by modifying the T4 templates as suggested above and creating a .xsd file which can be passed through the SVCUtil.exe tool in order to create the datacontracts file.

Convert a Class Library Project Containing POCO Classes Not Created using the EF Power Tool

The first step of changing the T4 template to generate POCOs to make them suitable for conversion to datacontracts can be skipped as the EF Power tool was not used.

Here is a summary of the revised steps involved in producing datacontracts for a class library:

  • Modify the POCO classes directly and apply the [System.Xml.Serialization.XmlIgnore] attribute to those properties returning data types which cannot be serialized such as ICollection and DBSet. Alternatively, use a different return type, i.e., change ICollection to List.
  • Build the class library to refresh the project DLL in theBinDebug folder.
  • Pass the project DLL through the xsd.exe tool as shown above either serializing the whole assembly or specific classes only.
  • Pass the resulting .xsd file through SVCUtil.exe in order to produce a file containing all the datacontracts.
Entity Framework Datacontract Serialization

Points of Interest

  1. When running the XSD tool, an error similar to the one shown below may appear.

    It appears because xsd.exe is unable to serialize interfaces. This was resolved by modifying the T4 templates and adding the [System.Xml.Serialization.XmlIgnore] over the property causing the error. It is also possible to resolve this error by changing the ICollection to a List.

  2. The datacontracts.cs file contains empty classes or classes which should not be served as datacontracts.

    It is possible to generate an xsd file containing specific classes to be converted to datacontracts, and leaving out the ones which are not to be exposed to a consumer.

    In the BinDebug folder of the Models project, enter the following command:

    In the above example, only the User, UserGroup, Group, Permission, GroupPermission classes are to be serialized to the Schema0.xsd file. Other classes in the models.dll such as UsersContext.cs are excluded.

Conclusion

The article demonstrated how it is possible to convert a class library of POCO classes to DataContracts by converting the .dll for the class library project to a .xsd file, then using the SVCUtil.exe tool to create a DataContracts file. The focus of the article has been to demonstrate how this can be achieved for POCO classes produced from using the EF Power tool by directly manipulating the T4 template which produces the classes to make the classes more suitable for serialization. The article also briefly explained how to create datacontracts for a class library of POCOs which were not produced using the EF Power tool.

History

  • v 1.0 - 12:25GMT 2016-02-26

Entity Framework Connection String

Re: DataAnnotations with EntityFramework (Database First) method

Aug 24, 2011 04:29 PM|wsyeager36|LINK

Alan, I've already viewed that article. It is not set up for the Database First method. It is set up for Code First. The way I have my Model class set up is supposed to be the way to set it up using EntityFramewowrk with the DataBase First methodology. Take a look at the previous post to see how the class is set up.

Additionally, the content in there is outdated as far as the DLLs go. Check the DLLs I'm using in the above post.

The bottom line is this....

My data model for the Email field specifies the following:
[Required]
[StringLength(50)]
[DataType(DataType.EmailAddress)]

The ModelState.Count parameter is correct and the ModelState.Keys are correct along with their respective values that I am checking during debugging in the collection. It's not setting the ModelState.IsValid property to FALSE based on the above directives.

Datacontract Serialization

The Model directives above are not being triggered on the front end for validation.

It's amazing how I can't get a simple direct answer of how to use DataAnnotations using the DataBase First methodology...

Entity Framework Data Contract Serialization

How can this be resolved?