branch | package | AppVeyor |
---|---|---|
master |
||
develop |
- |
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.
- 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
InfoCarrier.Core is bringing together the following open source projects
- Entity Framework Core by Microsoft.
- Remote.Linq and aqua-core by Christof Senn.
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.
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) { ... }
}
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();
}
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);
}