Skip to content

EntityFrameworkCore provider for multitier applications.

License

Notifications You must be signed in to change notification settings

azabluda/InfoCarrier.Core

Repository files navigation

InfoCarrier.Core

branch package AppVeyor
master NuGet Badge Build status
develop - Build status

Description:

InfoCarrier.Core is a framework developed by on/off it-solutions gmbh for building multi-tier applications in .NET. This repository contains the key data access component of the framework which essentially is a non-relational provider for Entity Framework Core which can be deployed on the client-side of your 3-tier application allowing you to use the full power of EF.Core right in your client application (e.g. WPF, WinForms, Xamarin, UWP, Blazor Wasm, etc). The main idea is that instead of querying the relational database the commands are translated into requests to your application server where they are executed against the real database.

It is important to note that InfoCarrier.Core dictates neither a communication platform nor a serialization framework. We had positive experience with WCF and ASP.NET Core, both using Json.NET for serialization, but you are free to choose other frameworks and libraries. InfoCarrier.Core is only responsible for translating client commands into serializable objects, leaving it up to you how to deliver them to the server for actual execution. The same is valid for the execution results.

Features:

  • All features of Entity Framework Core in your client application
    • LINQ
    • Change Tracking
    • Identity Map
    • Navigation Property Fix-up
    • Eager/Lazy/Explicit Loading of Navigation Properties
    • etc.
  • DbContext and entity classes shared between client and server, no need to duplicate this code
  • Concise client-side interface IInfoCarrierClient
  • Easy to use server-side service IInfoCarrierServer
  • You decide what communication platform to use

Credits:

InfoCarrier.Core is bringing together the following open source projects

Sample

The complete WCF sample is located in the /sample folder. There you also find simple client/server applications based on ASP.NET Core Web API and ServiceStack with basic support of transactions.

Entities and DbContext

This code/assembly can be shared between client and server

public class Blog { ... }

public class Post { ... }

public class Author { ... }

public class BloggingContext : DbContext
{
    public BloggingContext(DbContextOptions options)
        : base(options)
    { }

    public DbSet<Blog> Blogs { get; set; }

    public DbSet<Author> Authors { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder) { ... }
}

Client

Implement IInfoCarrierClient interface, e.g. using Windows Communication Foundation

public class WcfInfoCarrierClient : IInfoCarrierClient
{
    private readonly ChannelFactory<IWcfService> channelFactory
        = new ChannelFactory<IWcfService>(...);

    // Gets the remote server address. Used for logging.
    public string ServerUrl
        => this.channelFactory.Endpoint.Address.ToString();

    public QueryDataResult QueryData(QueryDataRequest request, DbContext dbContext)
    {
        IWcfService channel = this.channelFactory.CreateChannel();
        using ((IDisposable)channel)
        {
            return channel.ProcessQueryDataRequest(request);
        }
    }

    public SaveChangesResult SaveChanges(SaveChangesRequest request)
    {
        IWcfService channel = this.channelFactory.CreateChannel();
        using ((IDisposable)channel)
        {
            return channel.ProcessSaveChangesRequest(request);
        }
    }

    // Let other methods of IInfoCarrierClient just throw NotSupportedException for now.
    ...
}

Configure and use Entity Framework Core

var optionsBuilder = new DbContextOptionsBuilder()
    .UseInfoCarrierClient(new WcfInfoCarrierClient());

using (var context = new BloggingContext(optionsBuilder.Options))
{
    Post myBlogPost = (
        from blog in context.Blogs
        from post in blog.Posts
        join author in context.Authors on blog.AuthorId equals author.Id
        where author.login == "hi-its-me"
        where post.Title == "my-blog-post"
        select post).Single();

    myBlogPost.Date = DateTime.Now;

    context.SaveChanges();
}

Server

Implement your back-end service with the help of IInfoCarrierServer. In the simple case when no transaction support is required it may look like the following:

[ServiceContract]
public interface IWcfService
{
    [OperationContract]
    QueryDataResult ProcessQueryDataRequest(QueryDataRequest request);

    [OperationContract]
    SaveChangesResult ProcessSaveChangesRequest(SaveChangesRequest request);
}

[ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)]
public class InfoCarrierService : IWcfService
{
    private readonly ServiceProvider serviceProvider;
    private readonly DbContextOptions dbContextOptions;
    private readonly IInfoCarrierServer infoCarrierServer;

    public InfoCarrierService()
    {
        this.serviceProvider = new ServiceCollection()
            .AddEntityFrameworkSqlServer()
            .AddInfoCarrierServer() // add IInfoCarrierServer service
            .BuildServiceProvider();

        var optionsBuilder = new DbContextOptionsBuilder()
            .UseInternalServiceProvider(this.serviceProvider)
            .UseSqlServer(connectionString);
        this.dbContextOptions = optionsBuilder.Options;

        this.infoCarrierServer = this.serviceProvider.GetRequiredService<IInfoCarrierServer>();
    }

    private DbContext CreateDbContext()
        => new BloggingContext(dbContextOptions);

    public QueryDataResult ProcessQueryDataRequest(QueryDataRequest request)
        => this.infoCarrierServer.QueryData(this.CreateDbContext, request);

    public SaveChangesResult ProcessSaveChangesRequest(SaveChangesRequest request)
        => this.infoCarrierServer.SaveChanges(this.CreateDbContext, request);
}