Five New Optimizely Certifications are Here! Validate your expertise and advance your career with our latest certification exams. Click here to find out more
Five New Optimizely Certifications are Here! Validate your expertise and advance your career with our latest certification exams. Click here to find out more
The Dynamic Data Store (DDS) is a component offering an API and infrastructure for the saving, loading and searching of both compile time data types (.NET object instances) and runtime data types (property bags). The component is shipped as part of the EPiServer Framework package.
Note that this document refers to a DDS sample project which can be downloaded from EPiServer World. The code samples are for EPiServer CMS 6 R2, but are applicable also for EPiServer 7.
The Dynamic Data Store is a new component offering an API and infrastructure for the saving, loading and searching of both compile time data types (.NET object instances) and runtime data types (property bags).
Alternative technologies include Microsoft’s Entity Framework 2.0 and NHibernate for .NET. However, the Dynamic Data Store has been specifically designed with EPiServer CMS and its flexible user-driven data in mind.
The EPiServer.Data assembly contains the following namespaces:
Use the DynamicDataStoreFactory class to create, obtain and delete stores. The class has a single instance which can be obtained from the static Instance property. Alternatively, stores can be automatically created for .NET classes by decorating them with the EPiServerDataStoreAttribute and setting the AutomaticallyCreateStore property to true.
See the UsingStores class in the DDS sample project for examples on creating, obtaining and deleting stores.
Data can be saved and loaded using compile time data types (.NET classes) and runtime data types via the EPiServer.Data.Dynamic.PropertyBag class. The Dynamic Data Store is divided into logical stores which are identified by name. Stores are not polymorphic which means only one property set may be saved in a store although it is possible to re-map stores and achieve a level of polymorphism though the use of interfaces and template types.
See the LoadSaveType and LoadSavePropertyBag classes in the the DDS sample project for examples of loading and saving data.
You can search data in the Dynamic Data Store in the following ways:
See the UsingLinq and UsingFind classes in the DDS sample project for examples of searching for data.
The Dynamic Data Store is essentially an Object-Relational mapper. When used with compile time data types (.NET classes), all properties that have a public “getter” and a “setter” (setter does not need to be public) are mapped to a column in a database table. For runtime data types, each property added to a PropertyBag is also mapped in the same way.
The Dynamic Data Store uses the “big table” approach to storing data. That is by default, all data types are stored in one database table. This table contains many columns, several of each data type that the Dynamic Data Store supports.
When a data structure is saved, the .NET CLR type of each property is mapped against an internal list of supported types. The following types of mapping supported:
Inline mapping is where a property of a class or PropertyBag can be mapped directly against one of the supported “big table” database columns. The following types can be mapped inline:
A property is mapped as a collection if it implements the System.IEnumerable interface. In this case all elements of the collection (both keys and values in the case of System.IDictionary) are stored in a special reference table.
Even though the EPiServer.Data.Dynamic.PropertyBag implements System.IEnumerable is will actually be treated as a reference type (see below).
All properties that cannot be mapped inline or as a collection (plus the EPiServer.Data.Dynamic.PropertyBag type) are mapped as references. This means that their properties are mapped in-turn as a sub-type, and a link row is added in the reference table to link the parent data structure with the child data structure. This allows for complex trees of data structures (object graphs) to be saved in the Dynamic Data Store.
The default Dynamic Data Store “big table” is called tblBigTable, which contains the following fixed columns (meaning mandatory columns):
The default “big table” also contains the following optional columns:
The columns whose name starts with “Indexed” have database indexes created on them.
Perhaps you want to add and remove columns in this table to suit the type of data you are saving. This may be particularly useful if you know you are going to store a data type with more than, for example, 10 strings. By default, the 11th to 20th strings would be stored in a 2nd row for the type which means a join has to be done at runtime when reading the data. By adding String11, String12 etc to the “big table”, you limit the chance of a row overspill and therefore increase performance. If you require more indexes then add columns with names starting with “Indexed” and ensure an index is created on them.
You can also add your own “big table” if you want. This may be particularly useful if you know you will be storing a type that only contains strings for example. Along with the mandatory columns (pkId, Row, StoreName, ItemType) you can add about 20 StringXX columns.
The following tables lists the database columns types in the default “big table” and the .NET CLR “inline” types they are mapped to:
Database Column Type | .NET CLR “Inline” Types |
---|---|
varbinary(max) varbinary(900) |
System.Byte[] |
int | System.Byte, System.Int16, System.Int32, System.Enum |
bigint | System.Int64 |
float | System.Single, System.Double |
datetime | System.DateTime |
uniqueidentifier | System.Guid |
nvarchar(max) nvarchar(450) |
System.String, System.Char, System.Char[], EpiServer.Data.Identity |
bit | System.Boolean |
Each store is actually represented in the database by a view. The views can be used as normal including cross joining with other tables and views in the database.
Each data structure that is saved in the Dynamic Data Store is given an identity. This identity is represented by the EPiServer.Data.Identity class, which contains the following parts:
The implementer of a .NET class that is to be stored in the Dynamic Data Store can choose to explicitly manage the ID the objects get when stored. This can be done in the following ways:
See the UsingManagedIdentity class in the DDS sample project for examples of specific identity management.
The Dynamic Data Store supports POCO objects. When POCO objects are stored in the Dynamic Data Store, special care needs to be taken when saving (updating) existing objects. Because the Dynamic Data Store does not have an ID it can use to determine if an object is new or existing, it relies on state information held for objects that have been previously loaded through the same instance of the Dynamic Data Store when saving them back.
See the UsingImplicitidentity class in the DDS sample project for examples of implicit identity management.
Dynamic Data Store has support for storing objects implementing IEntity, which can be used in the entity system (previously shipped with EPiServer Community/Relate). Classes implementing IEntity can use the same functionality provided to classes that implement IDynamicData (but the interfaces cannot be used at the same time).
An object, stored in the Dynamic Data Store with an IEntity property, will be stored in two different ways in depending if the IEntity has been registered to a provider or not.
If the IEntity has not been registered against a provider, the IEntity object will be stored in the same way as any other reference type used as a property.
If the IEntity has been registered to a provider, the property will be stored as a line in the reference table with property type, provider type, and the Identity (ID) value of the IEntity instance. When saving an object the IEntityProvider will not be called automatically for the property. Storing the property in its provider should therefore be done before calling the Save method on the main object.
When loading an object, the provider registered with the IEntity type will be called to load the property from its provider during load on the main object.
IEntity properties stored through a provider will not support LINQ. This is because the provider could store the objects in any possible way. It will still be possible to query on any other property in the class, but when querying on the IEntity property, a NotSupportedException will be thrown.
The Dynamic Data Store has extensive support for Microsoft’s Language Integrated Query (LINQ). The LINQ support is the same for both typed stores and for property bags.
“Where” is supported on inline types, independent if the inline types are directly on the queried object or nestled inside another object.
var query = (from person in _personStore.Items<Person>() where
person.Address.City == "Stockholm" select person);
“Order by” and “Then by” are supported for inline types, independent if the inline types are directly on the queried object or nestled inside another object.
var query = (from person in _personStore.Items<Person>() orderby
person.Address.City select person);
To receive the whole object in the store, the “select object” can be used.
var query = (from person in _personStore.Items<Person>() select person);
It is also possible to receive an anonym type from the store by using the “new” keyword.
var query = (from person in _personStore.Items<Person>() select new {
person.FirstName, person.LastName });
If an object contains an enumeration of an inline type, some operations on the enumeration are supported. “Max()”, “Min()”, and “Average()” are supported without predicates, “Count()” are supported both with and without predicates, and “Contains()” are supported with predicate. The predicates are only supported when the predicate queries an inline type.
var queryMax = (from person in _personStore.Items<Person>() select
person.List.Max());
var queryMin = (from person in _personStore.Items<Person>() where
person.List.Min() < 10 select person);
var queryAverage = (from person in _personStore.Items<Person>() select
person.List.Average());
var queryCount = (from person in _personStore.Items<Person>() where
person.List.Count == 2 select person);
var queryCountWithPredicate = (from person in _personStore.Items<Person>()
select person.Address.Count(p => p.Street == "testar"));
var queryContains = (from person in _personStore.Items<Person>() select
person.List.Countains(p => p == "testar"));
It also supports “Contains()” for the opposite scenario, querying an inline property against a .NET enumeration, that is not stored in the DDS. This can be used to get all people with the lastname “Smith”, “Anderson”, or “Svensson”.
var lastNames = new List<string>();
lastNames.Add("Smith");
lastNames.Add("Anderson");
lastNames.Add("Svensson");
var query = _personStore.Items<Person>().where(p =>
lastNames.Contains(p.LastName).ToList();
Group by is supported for inline types. If the query has been grouped, some operations are supported for the grouped data. “Sum()”, “Max()”, “Min()”, and “Average()” are supported with predicates, and “Count()” are supported without predicate. The predicates are only supported when the predicate queries an inline type.
var query = _personStore.Items<Person>().GroupBy(p => p.Age).Select(m =>
new { Count = m.Count(), Sum = m.Sum(s => s.Friends.ShoeSize), Max = m.Max(s =>
s.Friends.ShoeSize), Min = m.Min(s => s.Friends.ShoeSize), Average =
m.Avergage(s => s.Friends.ShoeSize) });
Multiple groupings are also supported.
var query = _personStore.Items<Person>().GroupBy(p => new { FirstName =
p.FirstName, Age = p.Age }).Select(m => new { m.Key.FirstName, m.Key.Age });
Skip(x), Take(y) and Reverse() are also supported methods. Those can be helpful when developing paging.
query.Reverse();
query.Skip(10).Take(20);
The following string operations are supported:
var startsWith = (from person in _personStore.Items<Person>() where
person.Address.City.StartsWith("St") select person);
var contains = (from person in _personStore.Items<Person>() where
person.Address.City.Contains("St") select person);
var EndsWith = (from person in _personStore.Items<Person>() where
person.Address.City.EndsWith("holm") select person);
var SubString = (from person in _personStore.Items<Person>() where
person.Address.City.SubString(2) == "ockholm" select person);
var trim = (from person in _personStore.Items<Person>() where
person.Address.City.Trim() == "Stockholm" select person);
var isNullOrEmpty = (from person in _personStore.Items<Person>() where
string.IsNullOrEmpty(person.Address.City) select person);
var toUpper = (from person in _personStore.Items<Person>() where
person.Address.City.ToUpper() == "STOCKHOLM" select person);
var toLower = (from person in _personStore.Items<Person>() where
person.Address.City.ToLower() == "stockholm" select person);
The following DateTime operations are supported:
var addYears = (from person in _personStore.Items<Person>() where
person.DateOfBirth.AddYears(4) < DateTime.Now select person);
var addMonths = (from person in _personStore.Items<Person>() where
person.DateOfBirth.AddMonth(4) < DateTime.Now select person);
var addDays = (from person in _personStore.Items<Person>() where
person.DateOfBirth.AddDays(4) < DateTime.Now select person);
var addMinutes = (from person in _personStore.Items<Person>() where
person.DateOfBirth.AddMinutes(4) < DateTime.Now select person);
var addSeconds = (from person in _personStore.Items<Person>() where
person.DateOfBirth.AddSeconds(4) < DateTime.Now select person);
var addMilliseconds = (from person in _personStore.Items<Person>() where
person.DateOfBirth.AddMilliseconds(4) < DateTime.Now select person);
var add = (from person in _personStore.Items<Person>() where
person.DateOfBirth.Add(new TimeSpan(1,2,3,4,5) < DateTime.Now select person);
var subtract = (from person in _personStore.Items<Person>() where
person.DateOfBirth.Subtract (DateTime.Now) < new TimeSpan(1,2,3,4,5);
The following string properties are supported:
var year = (from person in _personStore.Items<Person>() select new {
person.DateOfBirth.Year });
var year = (from person in _personStore.Items<Person>() select new {
person.DateOfBirth.Month });
var year = (from person in _personStore.Items<Person>() select new {
person.DateOfBirth.Day });
var year = (from person in _personStore.Items<Person>() select new {
person.DateOfBirth.Hour });
var year = (from person in _personStore.Items<Person>() select new {
person.DateOfBirth.Minute });
var year = (from person in _personStore.Items<Person>() select new {
person.DateOfBirth.Second });
var year = (from person in _personStore.Items<Person>() select new {
person.DateOfBirth.DayOfYear });
The queries are not executed until an deferrer (execution method) gets called. This means that it is possible to work with the query until an execution method gets called without the overhead of going to the database all the time.
The following deferred methods are supported:
List<People> peopleList = _personStore.Items<Person>().ToList();
Dictionary<Guid, Person> peopleDictionary =
_personStore.Items<Person>().ToDictionary(p => p.GuidId);
ILookup<Guid, Person> peopleLookups = _personStore.Items<Person>().ToLookup(p => p.GuidId);
int count = _personStore.Items<Person>().Count();
int countWithPredicate = _personStore.Items<Person>().Count(p => p.LastName == "Svensson");
People firstPerson = _personStore.Items<Person>().OrderBy(p => p.FirstName).First();
People firstPersonWithPredicate = _personStore.Items<Person>().OrderBy(p => p.FirstName).First(p => p.FirstName == "Svensson");
People singlePerson = _personStore.Items<Person>().where(p => p.FirstName == "Svensson").OrderBy(p => p.FirstName).Single();
People singlePersonWithPredicate = _personStore.Items<Person>().OrderBy(p => p.FirstName).Single(p => p.FirstName == "Svensson");
People lastPerson = _personStore.Items<Person>().OrderBy(p => p.FirstName).Last();
People lastPersonWithPredicate = _personStore.Items<Person>().OrderBy(p => p.FirstName).Last(p => p.FirstName == "Svensson");
Sometimes it is necessary to have a condition in the code that should render the query in different ways depending on the condition. The following example shows how to work with a query:
var query = (from person in _personStore.Items<Person>() select person);
if (myCondition)
{
query = query.where(person.LastName.StartsWith("a");
}
var result = query.ToList();
See the UsingLinq class in the DDS sample project for examples of LINQ support.
When instances of a compile time data type (.NET classes excluding EPiServer.Data.Dynamic.PropertyBag and classes implementing System.IEnumerable) are saved in the Dynamic Data Store, their “inline” properties are mapped to columns in the “big table”. This is known logically as a store.
The default algorithm for mapping .NET classes (excluding EPiServer.Data.Dynamic.PropertyBag and classes implementing System.IEnumerable) to as store is as follows:
It is possible to override the default mapping behavior. This is useful if you do not want certain public properties to be mapped or do want certain non-public properties to be mapped.
To use custom mapping you need to add the System.Runtime.Serialization.DataContactAttribute to your class definition. In this case, only properties marked with the System.Runtime.Serialization.DataMemberAttribute will be mapped and saved in the Dynamic Data Store regardless of the accessibility status. They must still however have both a getter and setter.
See the MappingWithDataContract class in the DDS sample project for examples.
Perhaps you want to save an object of an existing class that has already been marked with DataContactAttribute and its member properties with DataMemberAttribute. One problem might be that the use of these properties does not match the desired behavior that you want when an object instance is saved in the Dynamic Data Store. In these cases you can also add the EPiServerDataContractAttribute to the class definition and EPiServerDataMemberAttribute to the properties to be saved. The Dynamic Data Store will use these attributes in preference to the Microsoft ones to resolve the conflict.
See the MappingWithEPiServerDataContract class in the DDS sample project for examples.
Some classes in the .NET Framework do not have properties that the Dynamic Data Store could use to extract the value and save to the database and then re-inflate an instance with the value from the database. You may want to use such a type as a property on a class that is to be saved to the Dynamic Data Store and therefore in this case you will need to register a Type Handler for it.
A good example of this is the System.Uri class. This class only has read-only properties and therefore saving an instance to the Dynamic Data Store without a Type Handler is meaningless as no data will be stored for it.
See the MappingWithTypeHandler class in the DDS sample project for examples.
Properties that are saved using PropertyBags are mapped as if the properties were public members properties on a normal .NET class. PropertyBags can be mapped in the following ways:
See the ImplicitDynamicMapping and ExplicitDynamicMapping classes in the DDS sample project for examples.
From time to time you may need to change the structure of your data. This may mean adding, removing or changing properties on your .NET classes or PropertyBags.
The Dynamic Data Store is quite flexible when it comes to accepting changes to types that have been saved in a store.
You can remap .NET classes to stores in the following ways:
A .NET class whose instances are to be saved in the Dynamic Data Store can be optionally decorated with the EPiServerDataStoreAttribute. In these cases set the AutomaticallyRemapStore property to true. When the EPiServer application starts it scans for all classes with this attribute and will automatically do a remap of the .NET class to the store if one is needed. Any properties that have been renamed MUST be marked with the EPiServerDataPropertyRenameAttribute attribute otherwise the remap treats them as if one property was removed (with the old name) and one added (with the new name).
To re-map a store, obtain the store definition of a store either via the StoreDefinition property of a DynamicDataStore instance or via the StoreDefinition.Get method. You can then call the Rename and Remap methods to update the store's mappings. Note that you should call Rename before Remap otherwise properties that have been renamed in the data type will be treated as one property removed and one added. Finally call the CommitChanges method of StoreDefinition to update the store's meta information in the database. If a DynamicDataStore instance reference is held then its Refresh method should be called to align its in-memory copy of the store definition with the one committed to disk.
The following rules are followed when remapping stores:
See the StoreReMapping class for examples of store re-mapping and the PropertyReName class for how to update mappings when a property has been renamed on a type, both in the Dynamic Data Store SDK.
It may be convenient to be able to always save instances of a Type in the same store, regardless of where those instances are in an object graph. You have the following options:
Example:
Result: The top level Person will be saved in the “MyPeople” store but all other instances of the Person Type in the object graph will be saved in the “People” store (because of the global mapping).
In order to adhere to the global Type to Store mappings you should create/obtain your top level store by calling DynamicDataStoreFactory.Create or GetStore with just the type of your type and not a store name.
See the UsingGlobalTypeToStoreMapping and UsingLocalTypeToStoreMapping classes in the DDS sample project for more details.
In the same way as it may be convenient to map a Type to a Store, it may also prove convenient to map a Store to a custom Big Table. See the Big Table section for more details.
The UsingGlobalStoreToTableMapping class in the DDS sample project demonstrates this technique.
Properties saved in store can be indexed for faster searching. When an index is set on a property it will automatically be mapped to an indexed column in the big table.
The UsingIndexes class in the DDS sample project demonstrates this technique.
You have the following options to configure through the application’s configuration file:
Last updated: Feb 23, 2015