UPDATE: After the initial version of this blog post, we introduced some changes in Sitefinity, which required somewhat different code. Some people noticed this in the comments below. The blog post has been updated with code working for Sitefinity 4.1. You can always find a working code sample illustrating the topic in the latest release of the Products module in Sitefinity SDK.
As you know, Sitefinity relies heavily on the OpenAccess ORM for its data layer. We’ve provided an extensible model, in which you can plug a custom implementation for your storage mechanism, but all our modules use OpenAccess by default. That’s why we are excited to announce that we’re shipping a new version of OpenAccess with the 4.1 release.
There will be tons of benefits in this new version that we’ll announce later. For now, we want to bring your attention to a particular change we’re introducing with this upgrade - Fluent mappings for persistent classes. We’ve updated all of our built-in modules to use the new mappings, and all custom modules that use OpenAccess will have to do the same.
This means anyone who has created a custom Sitefinity 4.x module that relies on OpenAccess will need to make changes to their module before it is compatible with 4.1.
In this blog post we’ll look at the following topics:- The advantages of Fluent Mappings in OpenAccess
- How to modify your custom module to make it compatible with Sitefinity 4.1
Let's get started...
Understanding the value of fluent mappings
To illustrate the value of fluent mapping support, let's look at how persistent classes were mapped until now. OpenAccess needs metadata to know how to generate tables from your persistent classes. In the current Sitefinity projects, there are two places where it gets the metadata from:
- The app.config file in your model assembly
- Attributes you put in the C# code for the persistent classes and their properties
This is cumbersome for two reasons. First, you need to maintain both the code of the persistent class and the app.config file. Second, the only way to provide different mappings for a single class is to duplicate the whole mapping.
Let’s take the sample Products module from the Sitefinity SDK as an example. Its current mappings look like this:
<
mappings
current
=
"mssqlMapping"
>
<
mapping
id
=
"mssqlMapping"
>
<
namespace
name
=
"ProductCatalogSample.Model"
>
<
class
name
=
"ProductItem"
>
<
extension
key
=
"db-table-name"
value
=
"sfex_product_item"
/>
<
field
name
=
"permissions"
>
<
collection
>
<
extension
key
=
"db-link-table"
/>
</
collection
>
</
field
>
<
field
name
=
"urls"
>
<
collection
>
<
extension
key
=
"inverse"
value
=
"parent"
/>
<
extension
key
=
"managed"
value
=
"true"
/>
</
collection
>
</
field
>
<
field
name
=
"permissionChildren"
>
<
collection
>
<
extension
key
=
"db-link-table"
/>
</
collection
>
</
field
>
</
class
>
<
class
name
=
"ProductItemUrlData"
>
<!--<extension key="db-table-name" value="sfex_product_urls" />-->
<
field
name
=
"parent"
>
<
extension
key
=
"db-ref"
value
=
"Telerik.Sitefinity.GenericContent.Model.Content.contentId"
>
<
extension
key
=
"db-column"
>
<
extension
key
=
"db-column-name"
value
=
"content_id"
/>
</
extension
>
</
extension
>
</
field
>
</
class
>
</
namespace
>
</
mapping
>
</
mappings
>
Although the above configuration doesn’t seem like a lot of work, it creates maintenance problems:
- Every property or setting not explicitly specified in the XML is not present anywhere in your project. It gets the default persistence setting, but if the default changes, your data layer may break.
- Apart from the XML, we need to put attributes for our properties in the model class itself. This means that there’s no single place where you can get a complete picture of how your class is transformed into a database table.
- If you want to tweak a mapping for a particular database (say Oracle), you have to duplicate the whole mapping, doubling the XML configuration in size.
- The mappings of all your persistent classes in the assembly are in the same configuration file. Having more persistent classes dramatically increases the size of this file, making it harder to maintain.
Fluent mappings address all of the above challenges. They provide a single place where all persistent classes are configured. Joshua Holt has created a series of videos introducing Fluent mappings in the OpenAccess ORM. I highly encourage you to watch those before you start changing your modules. Below, we’ll put the same concepts into the Context of a Sitefinity module.
How to create new fluent mappings for your module
With the newly introduced fluent mappings, all of the above is replaced with code. Instead of relying on a particular XML file, the framework calls a method that you write. You can then supply the mapping and include your own logic depending on the database (e.g. map something differently for Oracle). The steps you need to follow are the following:
- Implement the new Context property of the IOpenAccessProvider interface
- Implement the GetMetadataSource method of the IOpenAccessMetadataProvider interface
- Implement your fluent mapping class and inherit from OpenAccessFluentMappingBase
- Implement your specific metadata source and return it in the provider
Implement the new Context property of the IOpenAccessProvider interface
This property provides runtime state information used by Sitefinity. You just need to have it – nothing fancy in the implementation, except an automatic getter and setter.
public
OpenAccessProviderContext Context
{
get
;
set
;
}
Implement the GetMetadataSource method of the IOpenAccessMetadataProvider interface
This method is called automatically and returns a MetadataSource object, containing all information about the mappings in the specific provider. The object will be of the type you implement in the last step. Here just create an instance by passing the context parameter.
public
MetadataSource GetMetaDataSource(IDatabaseMappingContext context)
{
return
new
ProductsFluentMetadataSource(context);
}
Implement your fluent mapping class and inherit from OpenAccessFluentMappingBase
public
class
ProductsFluentMapping : OpenAccessFluentMappingBase
{
public
ProductsFluentMapping(IDatabaseMappingContext context)
:
base
(context)
{ }
public
override
IList<MappingConfiguration> GetMapping()
{
var mappings =
new
List<MappingConfiguration>();
MapItem(mappings);
MapUrlData(mappings);
return
mappings;
}
private
void
MapItem(IList<MappingConfiguration> mappings)
{
var itemMapping =
new
MappingConfiguration<ProductItem>();
itemMapping.HasProperty(p => p.Id).IsIdentity();
itemMapping.MapType(p =>
new
{ }).ToTable(
"custom_products"
);
itemMapping.HasProperty(p => p.Price);
itemMapping.HasProperty(p => p.QuantityInStock);
itemMapping.HasAssociation<Telerik.Sitefinity.Security.Model.Permission>(p => p.Permissions);
itemMapping.HasProperty(p => p.InheritsPermissions);
itemMapping.HasAssociation<Telerik.Sitefinity.Security.Model.PermissionsInheritanceMap>(p => p.PermissionChildren);
itemMapping.HasProperty(p => p.CanInheritPermissions);
itemMapping.HasAssociation(p => p.Urls).WithOppositeMember(
"parent"
,
"Parent"
).ToColumn(
"content_id"
).IsDependent().IsManaged();
itemMapping.HasAssociation<Telerik.Sitefinity.Workflow.Model.Tracking.ApprovalTrackingRecordMap>(p => p.ApprovalTrackingRecordMap);
mappings.Add(itemMapping);
}
private
void
MapUrlData(IList<MappingConfiguration> mappings)
{
var urlDataMapping =
new
MappingConfiguration<ProductItemUrlData>();
urlDataMapping.MapType(p =>
new
{ }).Inheritance(InheritanceStrategy.Flat).ToTable(
"sf_url_data"
);
mappings.Add(urlDataMapping);
}
}
This is the implementation of the mapping itself. All mappings inherit OpenAccessFluentMappingBase and override the GetMapping() method. In the sample Products module, we have separated its work in two calls, one for each persistent class. Everything you need to specify in a mapping has a corresponding method in the MappingConfiguration class. You can specify the name of the table, properties, any associations to other classes, inheritance configurations, etc. When developing a content-based module, they are going to be very similar to the mappings for Products in the above sample.
Implement your specific metadata source and return it in the provider
public
class
ProductsFluentMetadataSource : ContentBaseMetadataSource
{
public
ProductsFluentMetadataSource():
base
(null)
{ }
public
ProductsFluentMetadataSource(IDatabaseMappingContext context)
:
base
(context)
{ }
protected
override
IList<IOpenAccessFluentMapping> BuildCustomMappings()
{
var sitefinityMappings =
base
.BuildCustomMappings();
sitefinityMappings.Add(
new
ProductsFluentMapping(this.Context));
return
sitefinityMappings;
}
}
The metadata source is an object that you return in the GetMetadataSource method you implemented in the second step above. If you are building a custom module from scratch, you can directly create an instance of any class that derives from MetadataSource. However, when building content-based modules, most of the work has been done for you anyway. We’ve provided a way for you to reuse it – just create your own metadata source and inherit from ContentBaseMetadataSource. Then your only job is to override BuildCustomMappings() and add an instance of the mapping class you created in the previous step to the ones created in the base implementation (Generic Content).
Samples and additional help
Please note that all modules working with OpenAccess need to edit their project files and supply the correct path to the OpenAccess assemblies. For more information on this, refer to our documentation.
The source code for the new Products module mappings is available in the 4.1 SDK. Once you follow the steps above, your module should be ready for Sitefinity 4.1. Have these changes in mind before upgrading your projects to the release coming up next week. We’ll be available to help you if you have questions about the new OpenAccess version. We’re looking forward to seeing how you extend Sitefinity. Happy coding.