Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/sebcam/CrossSync
Provides Mobile / Server synchronization with offline support using .NetCore and EntityFramework.Core
https://github.com/sebcam/CrossSync
entity-framework-core mvvm netcore netcore-webapi offline-sync sqlite sqlserver synchronization xamarin xamarin-forms
Last synced: 28 days ago
JSON representation
Provides Mobile / Server synchronization with offline support using .NetCore and EntityFramework.Core
- Host: GitHub
- URL: https://github.com/sebcam/CrossSync
- Owner: sebcam
- License: mit
- Created: 2018-08-14T15:06:44.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2023-01-06T20:20:13.000Z (almost 2 years ago)
- Last Synced: 2024-12-01T10:27:51.201Z (about 1 month ago)
- Topics: entity-framework-core, mvvm, netcore, netcore-webapi, offline-sync, sqlite, sqlserver, synchronization, xamarin, xamarin-forms
- Language: C#
- Size: 291 KB
- Stars: 17
- Watchers: 4
- Forks: 4
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# CrossSync
![.NetCore](https://img.shields.io/badge/.NetCore-2.1-blue.svg) ![.NetStandard](https://img.shields.io/badge/.NetStandard-2.0-blue.svg)CrossSync provides cross-plaform librairies for Xamarin and AspNetCore for easy data synchronization with offline data support based on Entity Framework Core.
### Why ?
Entity Framework Core now offers a way to share the business data code between the mobile and server applications.
For now, Azure Mobile App is the most popular framework to do this. But it's working with .NetFramework and bring it to .NetCore seems to be not announced yet.
So to work with Azure Mobile App, your data layer cannot be shared across these different platforms.This libs can be usefull for developpers which are looking for a way to synchronize offline datas on your mobile application using Entity Framework Core.
High level features are :
- EntityFramework use for both mobile (SQLite) and API (SQL Server)
- Offline data support
- Repository & Unit of work patterns
- ...## Top level Packages
| Name | Description | Nuget |
| ---- | ----------- | ----- |
| `CrossSync.AspNetCore` | Provides aspnet core sync tools | ![Version](https://img.shields.io/badge/-0.1.0-blue.svg) |
| `CrossSync.Xamarin`| Provides all implementations for mobile data offline and sync | ![Version](https://img.shields.io/badge/-0.1.0-blue.svg) |## Get Started
### Server side
On your AspNetCore API Project :
nuget install CrossSync.AspNetCore
##### Entity framework configuration
If you are not aware with EF, see [EF Core docs](https://docs.microsoft.com/en-us/ef/core/)Your **DbContext** have to inherit from **ServerContext** class.
All of your entities which need to be synchronized with mobile have to extend **VersionableEntity** class.
1. Your entities have to inherit from **VersionableEntity** class or implement **IVersionableEntity**.
2. Your EF Context have to inherit from **ServerContext** class.##### Controllers configuration
Controllers have to inherit from **SyncController** which is a crud controller.
When you are done, juste add sync registration into your startup.cs :
public void ConfigureServices(IServiceCollection services)
{
/* ... */
// Sync services registration
services.AddSync();
}
### Client Side using Xamarin.Forms
#### Installation
On your Core assembly :nuget install CrossSync.Xamarin
#### Configuration
This library needs to know about the connectivity status.
Create a class which implements IConnectivityService.
For example, the following uses the [Plugin Connectivity](https://github.com/jamesmontemagno/ConnectivityPlugin) from James Montemagno.public class ConnectivityService : IConnectivityService
{
private readonly IConnectivity connectivity;public ConnectivityService(IConnectivity connectivity)
{
this.connectivity = connectivity;
}public bool IsConnected => connectivity.IsConnected;
public Task IsRemoteReachable(string url)
{
return connectivity.IsRemoteReachable(url);
}
}It also need a service to show errors during synchornization by implementing IErrorService.
For example, the following uses [Acr UserDialogs](https://github.com/aritchie/userdialogs) :public class ErrorService : IErrorService
{
private readonly IUserDialogs dialogs;public ErrorService(IUserDialogs dialogs)
{
this.dialogs = dialogs;
}
public void ShowError(string error)
{
using (dialogs.Toast(new ToastConfig(error) { Duration = new TimeSpan(0, 0, 2), Position = ToastPosition.Top })) { }
}
}The last step is to register services to the IoC container :
// using autofac
builder.RegisterType().AsImplementedInterfaces().SingleInstance();
builder.RegisterType().AsImplementedInterfaces().SingleInstance();
builder.RegisterType().AsSelf().InstancePerLifetimeScope();#### How to use
##### Business Data Services implementation
public class TodoListService : SyncService
{
private readonly IUnitOfWork unitOfWork;public TodoListService(IUnitOfWork unitOfWork, IConnectivityService connectivityService, Lazy errorService, SyncConfiguration config) : base(unitOfWork, connectivityService, errorService, config)
{
this.unitOfWork = unitOfWork;
}public override string ApiUri => "api/todolist";
// Order in which sync is done between services depending on data relations
public override int Order => 100;
}##### Start a synchronization
Synchronization, for now, have to be manually launched like this :
// Resolve the registrated sync service
var synchronizationService = IoC.Resolve();
// Starts the synchronization
await synchronizationService.SyncAsync();Soon, the synchronization will be done within an async background task.
##### Handle conflicts
In order to handler version conflicts between offline and server datas, you need to implement IConflictHandler<> on your data service
public class TodoListService : SyncService, IConflictHandler
{
/* ... */public async Task HandleConflict(Entities.Shared.TodoList clientValue, Entities.Shared.TodoList serverValue)
{
// Ask for user to choose between client and server version
var selectedResult = await IoC.Resolve().ActionSheetAsync("Conflict resolution", "Keep local version", null, buttons: new[] { "Take Server version" });
return selectedResult == "Keep local version" ? clientValue : serverValue;
}
}