/uqframework

Allows using LINQ to query any data source

Primary LanguageC#

UQframework

UQFramework allows linq to any data store in a similiar way as Linq2SQL and Entity Framework.

The library is based on .Net Standard 2.0 which means it can be used in .Net Framework and .Net Core projects.

Getting Started

To use UQFramework just follow these simple steps:

  1. Create Model(s). Entities which you serialize/deserialize your data into/from.
  2. Crate DataAccessObject for the model. A component which reads/writes the entities
  3. Create a context. Contains collections of the entities and enables querying using LINQ and CRUD operations.

Creating Model

Any model with string identifier can be used in UQFramework. The identifier property must be marked with [Key] or [Identifier] attribute.

    public class Entity
    {
        [Key]
        public string Identifier { get; set; }

        public string Name { get; set; }

        public string Property1 { get; set; }
    }

Creating Data Access Components (DAO)

There are following interfaces available for implementation in Data Access Components:

  • IDataSourceReader<> - supports getting entity by identifier;
  • IDataBulkReader<> - supports getting a set of entities by a set of identifiers in one go;
  • IDataSourceWriter<> - supports CRUD operations;
  • IDataSourceBulkWriter<> - supports updating, creation and deletion of entities in one go.
  • INeedDataSourceProperties - allows passing parameters to datastore (such as connection strings)
  • IDataSourceEnumerator - supports enumeration of the entities. Must be implemented if caching is used (see below).

if DAO is intended to be used for fetching data it must implement either IDataSourceReader<> or IDataBulkReader<>. if in addition to that data changes are required then it must implement either IDataSourceWriter<> or IDataSourceBulkWriter<>

INeedDataSourceProperties is used when DAO needs some configuration to access the data. In the example below DAO operates over files in a folder and gets the path to this folder from the parameters supplied through INeedDataSourceProperties interface.

Example

    internal class DaoFile : IDataSourceReader<Entity>, IDataSourceEnumerator<Entity>, IDataSourceWriter<Entity>, INeedDataSourceProperties
    {
        private string _folder;

        public void SetProperties(IReadOnlyDictionary<string, object> properties)
        {
            _folder = (string)properties["folder"];
        }

        public Entity GetEntity(string identifier)
        {
            var file = Path.Combine(_folder, $"{identifier}.json");
            if (!File.Exists(file))
                return null;
            return JsonConvert.DeserializeObject<Entity>(File.ReadAllText(file));
        }

        public void AddEntity(Entity entity)
        {
            if (string.IsNullOrEmpty(entity.Identifier))
                entity.Identifier = (GetAllEntitiesIdentifiers().Max(int.Parse) + 1).ToString();

            var file = Path.Combine(_folder, $"{entity.Identifier}.json");
            File.WriteAllText(file, JsonConvert.SerializeObject(entity));
        }

        public void UpdateEntity(Entity entity)
        {
            var file = Path.Combine(_folder, $"{entity.Identifier}.json");
            File.WriteAllText(file, JsonConvert.SerializeObject(entity));
        }

        public void DeleteEntity(Entity entity)
        {
            var file = Path.Combine(_folder, $"{entity.Identifier}.json");
            File.Delete(file);
        }

        public IEnumerable<string> GetAllEntitiesIdentifiers()
        {
            return Directory.EnumerateFiles(_folder, "*.json").Select(Path.GetFileNameWithoutExtension);
        }

        public int Count()
        {
            return Directory.EnumerateFiles(_folder, "*.json").Count();
        }
    }

Creating Context

To access entities in collections and being able to query them using LINQ a context class must be created, according to the following rules:

  • It must iherit from UQContext
  • For each entity a property of type IUQCollection<> must be created and it must have a setter (preferably a private one).
  • The DAO matching the property should be specified in DataAccessObject attribute

Example

    internal class DataStoreContext : UQContext
    {
        public DataStoreContext() : base("db1", new Dictionary<string, object>
        {
            ["folder"] = @"C:\UQFramework.Demo\Files"
        })
        { }

        [DataAccessObject(typeof(DaoFile))]
        public IUQCollection<Entity> Entities { get; set; }
    }

In the example above the base constructor take dataStoreId and a dictionary of paramters which will be internally passed to all data access components implementing the INeedDataSourceProperties interface. dataStoreId - is logical name for the data storage which also allows switching between different datastores if the application supports many.

Querying the Entities

After all the previous steps are done data can be filtered and projected using LINQ:

    var context = new DataStoreContext();
    // search by substring
    var subString = "aSubstring";
    var list = context.Entities
        .Where(x => x.Name.IndexOf(subString, StringComparison.InvariantCultureIgnoreCase) >= 0)
        .ToList();

See UQFramework.Demo project for more details.

CRUD Operations

For CRUD operations use the corresponding methods of the context and then call SaveChanges():

var context = new DataStoreContext();
var item1 = context.Entities.FirstOrDefault(x => x.Identifier == "1");
var item2 = context.Entities.FirstOrDefault(x => x.Identifier == "2");

var newItem = new Entity { Name = "New Name" };
item1.Name = "Changed name for item 1";

context.Entities.Add(newItem);
context.Entities.Update(item1);
context.Entities.Delete(item2);

context.SaveChanges();

Caching

Filtering operations (such as Where clause and so on) can take a lot of time in case of large data stores for each item needs to be loaded and checked for meeting the filter criteria. UQ Framework allows to improve performance of such operations significantly by caching of a subset of entity properties. Cached properties are kept in memory along with the item identifier so that when only cached properties are used in an expression UQ framework is able to execute filter over the cache. Then when identifiers of the items is obtained only they are loaded from the datastore so the resulting number of data requests is significantly reduced. Furthermore, if only cached properties are retrieved (lets say an aggregation operation is used or Select projects on another type) then UQFrameworks executes the whole query entirely in memory without any call to data access components.

UQ keeps the cache in sync by updating on each CRUD operation (only added, changed or deleted items are updated).

To enable caching in UQ Framework the following needs to be done

  • Cache storage must be configured in App.config
  • DAO must implement IDataSourceEnumerator<> or IDataSourceEnumeratorEx<> interface
  • Each cached property must be decorated with Cached attribute.

Cache configuration

To configure cache storage add h-cache section to your app.config file. Then you must specify type and set enabled to true. Out of the box UQFramework supports storing the cache in text files (type "UQFramework.Cache.Providers.TextFileCacheProvider1, UQFramework"*) or in SQL Server database (type *"UQFramework.Cache.Providers.SqlCacheProvider1, UQFramework"). Alternatively, you can implement your own cache storage by inheriting from PersistentCacheProviderBase.

Example configuration to store cache in text files

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <configSections>
    <section name="uqframework" type="UQFramework.Configuration.UQConfiguration, UQFramework"/>
  </configSections>
  <startup>
    <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.7.2" />
  </startup>

  <uqframework>
    <h-cache enabled ="true" type="UQFramework.Cache.Providers.TextFileCacheProvider`1, UQFramework">
      <add name="folder" value ="C:\UQFramework.Demo\_Cache"/>
    </h-cache>
  </uqframework>
</configuration>

Example configuration to store cache in relational db

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <configSections>
    <section name="uqframework" type="UQFramework.Configuration.UQConfiguration, UQFramework"/>
  </configSections>
  <startup>
    <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.7.2" />
  </startup>

  <uqframework>
    <h-cache enabled ="true" type="UQFramework.Cache.Providers.SqlCacheProvider`1, UQFramework">
      <add name="connectionstring" value ="Data Source=.;Integrated Security=True"/>
    </h-cache>
  </uqframework>
</configuration>

Data Access Component changes

When cache is enabled or a new data store is connected UQFramework needs to build cache. This is mostly a one-time operation, although there might be occasions when cache re-build is required, so it can be called manually by calling NotifyCacheExpired or NotifyCacheItemsExpired methods of UQContext.

In order to build/(full rebuild) cache UQFramework retrieves all the items from datastore which means that the data access component must provide a list of all the items in the storage. It can be achived by implementing IDataSourceEnumerator<> or IDataSourceEnumeratorEx<>

Example

public IEnumerable<string> GetAllEntitiesIdentifiers()
{
    return Directory.EnumerateFiles(_folder, "*.json").Select(Path.GetFileNameWithoutExtension);
}

public int Count()
{
    return Directory.EnumerateFiles(_folder, "*.json").Count();
}

DaoVersion attribute

The DaoVersion attribute allows to tell UQFramework that data provided by the data access component is changed significantly and the current cache is no more relevant. So this attribute could be usefull when a new application version is deployed and existing cache needs to be rebuild.

[DaoVersion("1.1.0.0")]
internal class DaoFile : IDataSourceReader<Entity>, IDataSourceEnumerator<Entity>, IDataSourceWriter<Entity>, INeedDataSourceProperties

At the first usage UQFramework checks if data access component version stored in cache matches the value of the attribute and rebuild the cache if not. If the attribute is not specified its value considered equal to 1.0.0.0

See also Cache Rebuild Behaviour

Cached property

To put a property in cache decorate it with Cached attribute. Only properties which satisfy the following rules can be put in cache.

  • The property must have public setter and getter
  • The property must be of a value type or string or IEnumerable of a value type or string.

Example

public class Entity
{
    [Key]
    public string Identifier { get; set; }

    [Cached]
    public string Name { get; set; }

    public string Property1 { get; set; }

    [Cached]
    public IEnumerable<string> Collection { get; set; }
}

Cache rebuild behaviour

When an application version is upgraded or downgraded UQFramework attempts to use existing cache to avoid potentially costly operation of cache rebuild. However, to avoid the data discrepancy UQFramework automatically rebuilds the whole cache for an entity in the following cases:

  • Cached properties are changed;
  • Data Access Component for the entity collection is changed;
  • Data Access Component is still the same but its DaoVersion attribute has a different value;