Offline data for occasionally connected applications
Introduction
Whenever I go into a fast food centre to order food, I get a couple of choices: ‘to-go’ (take-away) or ‘have it here’ (eat-in). Because I usually need to get back to work quickly, I mostly choose to-go (take-away) and take the meal back to my home or work place. Most of the smart client applications you develop nowadays are to-go. You take the application with you on a laptop or other mobile device that is sometimes disconnected from the corporate network where your data resides. The requirement to access data anytime anywhere, immaterial of whether you are connected or disconnected is increasingly important. This article talks about Synchronization Services for ADO.NET. This allows you to take your data ‘to-go’ along with your application. This article describes the version of Synchronization Services for ADO.NET that ships along with Visual Studio.NET 2008 SP1.
Synchronization Services for ADO.NET
There are a few technologies that are already available for managing a local database along with your application. Microsoft has Remote Data Access and Merge Replication for the purpose of maintaining a local database and keeping it synchronized with a server database. Merge Replication is an important server-based synchronization technology that supports many advanced features. Remote Data Access is a rapid application development technology that can be used to quickly create synchronization to a database. Synchronization Services for ADO.NET brings in most of the rapid application development features of Remote Data Access and at the same time supporting many advanced scenarios for synchronizing data.
Synchronization Services for ADO.NET is part of the Microsoft Synchronization Framework. Microsoft Synchronization Framework is a broader platform that is useful for more than just database access. It provides synchronization capabilities for many other services: For example using the Synchronization Framework you can programmatically synchronize folders, outlook contacts, feeds etc. For the purpose of this article we will be looking at Synchronization Services for ADO.NET.
Create a Local Database Cache
We’ll work with the a simple example of a database table named Customer that stores customer details and is available in a SQL Server 2008 database in the server.
The smart client application that we create will not always have access to the customer data in the server. So we need to provide a way for the data to be available not only for offline usage to the smart client application, but also to sync the data with the server database when online. To accommodate this, we can make our smart client application rely on an intermediate source to retrieve and update data instead of the server database. So we’ll go ahead and add a Local Database Cache, named LocalDataCache.sync, to our smart client application. When you add a Local Database Cache to your application, the Sync Designer would be prompted to enable you to configure the settings for the data synchronization.
In the above window we have configured the server database connection to the SQL Server 2008 database. We have also configured the client database connection that indicates a new SQL Server Compact Edition database will be created to store data locally. We have also indicated that we want to use the SQL Server change-tracking feature to track changes on the data. SQL Server change-tracking is a new feature in SQL Server 2008 that allows us to track changes in the database tables without altering our server database schema so as to keep track of inserts, updates and deletes. We can also opt to use our own columns in the tables rather than to use SQL Server change-tracking. Next we can add tables from the server database that we want to make available offline
Since we opted to use SQL Server change-tracking, the options to configure change-tracking for the Customer table is disabled since it would be internally handled by SQL Server. If we do not use SQL Server change-tracking, we need to create columns in the server database to identify updates and inserts and we need to create a table within the database to move all deleted items to.
The Local Database Cache creates a SQL Server Compact Edition database to store data for offline use and generates the following code to synchronize the local database with the server database.
C# Code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 |
namespace SyncServices.SmartClient { public partial class LocalDataCacheClientSyncProvider : Microsoft.Synchronization.Data.SqlServerCe.SqlCeClientSyncProvider { ... } public partial class LocalDataCacheSyncAgent : Microsoft.Synchronization.SyncAgent { private CustomerSyncTable _customerSyncTable; ... public partial class CustomerSyncTable : Microsoft.Synchronization.Data.SyncTable { ... } } } namespace SyncServices.SmartClient { public partial class CustomerSyncAdapter : Microsoft.Synchronization.Data.Server.SyncAdapter { ... } public partial class LocalDataCacheServerSyncProvider : Microsoft.Synchronization.Data.Server.DbServerSyncProvider { private CustomerSyncAdapter _customerSyncAdapter; ... } } |
The generated code includes a sync provider for the SQL Server CE database (LocalDataCacheClientSyncProvider) and a sync agent (LocalDataCacheSyncAgent) for the client end. It also includes a sync provider for the SQL Server database (LocalDataCacheServerSyncProvider) and sync adapters for each table (CustomerSyncAdapter) we made available offline at the server end. The designer also generates SQL scripts for the changes made on the server database, for example, in this instance, enabling change-tracking for the Customer table.
To synchronize data we can use the LocalDataCacheSyncAgent as shown below
C# Code:
1 2 |
LocalDataCacheSyncAgent syncAgent = new LocalDataCacheSyncAgent(); SyncStatistics syncStats = syncAgent.Synchronize(); |
The above code will synchronize the data and provide statistics on the synchronization providing information such as the start time, completion time and the changes applied. By default the synchronization downloads the changes that have happened on the server database from that database to the client. You can set the SyncDirection for the Customer table to be bidirectional to indicate that the sync process should download and upload changes to and from the local SQL Server CE database.
C# Code
1 2 3 |
LocalDataCacheSyncAgent syncAgent = new LocalDataCacheSyncAgent(); syncAgent.Customer.SyncDirection = SyncDirection.Bidirectional; SyncStatistics syncStats = syncAgent.Synchronize(); |
The LocalDataCacheSyncAgent performs the synchronization by using an instance of a local provider (the generated LocalDataCacheClientSyncProvider) and an instance of a remote provider (the generated LocalDataCacheServerSyncProvider). The LocalDataCacheSyncAgent collects the changes from both databases and applies the changes using the providers at either end.
We can get a reference to these providers to intercept the data sent for synchronization or to monitor the progress of the synchronization process or to resolve conflicts during synchronization. For example the following code gets a reference to the providers at the client and server ends
C# Code:
1 2 3 4 5 6 |
LocalDataCacheSyncAgent syncAgent = new LocalDataCacheSyncAgent(); syncAgent.Customer.SyncDirection = SyncDirection.Bidirectional; LocalDataCacheServerSyncProvider serverProvider = syncAgent.RemoteProvider as LocalDataCacheServerSyncProvider; LocalDataCacheClientSyncProvider clientProvider = syncAgent.LocalProvider as LocalDataCacheClientSyncProvider; |
When we invoke the Synchronize method of the LocalDataCacheSyncAgent, the incremental changes from both the providers would be downloaded and applied at either ends. The providers also expose events that can be used to add custom event handling code during synchronization. For example the following code hooks an event handler to the ApplyChangesFailed event handler of the provider at the server end to log any conflicts during synchronization
C# code
1 2 3 4 5 6 7 8 9 |
LocalDataCacheSyncAgent syncAgent = new LocalDataCacheSyncAgent(); syncAgent.Customer.SyncDirection = SyncDirection.Bidirectional; LocalDataCacheServerSyncProvider serverProvider = syncAgent.RemoteProvider as LocalDataCacheServerSyncProvider; serverProvider.ApplyChangeFailed += delegate(object sender,ApplyChangeFailedEventArgs e) { //Log conflicts }; |
Create a SOA Friendly Data Cache
One big advantage of the ADO.NET sync services is that it is service-oriented architecture (SOA) friendly and can be easily altered to support different types of application. Let’s, for example, assume that the smart client application we created will not have access to the server database directly and we need to perform the synchronization through the HTTP protocol when the smart client has connectivity. We can easily create an additional layer between the smart client application and the server database to support this scenario. So let us add a WCF Service application as the server application and alter our smart client application to sync the data using the WCF service instead of syncing directly with the server database.
After adding the WCF service to the same solution as the smart client application, let us open the sync designer for the LocalDataCache.sync in our smart client application. When you expand the advanced section in the designer you will notice we can designate separate applications for the client and the server portions of the sync components. We’ll configure it, as shown below, to include both the server components in the WCF service application and the client components in the smart client application.
This will create the following service contract and service implementation in our WCF service to handle the synchronization
C# Code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 |
[ServiceContractAttribute()] public interface ILocalDataCacheSyncContract { [OperationContract()] SyncContext ApplyChanges(SyncGroupMetadata groupMetadata, DataSet dataSet, SyncSession syncSession); [OperationContract()] SyncContext GetChanges(SyncGroupMetadata groupMetadata, SyncSession syncSession); [OperationContract()] SyncSchema GetSchema(Collection<string> tableNames, SyncSession syncSession); [OperationContract()] SyncServerInfo GetServerInfo(SyncSession syncSession); } public partial class LocalDataCacheSyncService : object, ILocalDataCacheSyncContract { private LocalDataCacheServerSyncProvider _serverSyncProvider; public LocalDataCacheSyncService() { this._serverSyncProvider = new LocalDataCacheServerSyncProvider(); } public virtual SyncContext ApplyChanges(SyncGroupMetadata groupMetadata, DataSet dataSet, SyncSession syncSession) { return this._serverSyncProvider.ApplyChanges(groupMetadata, dataSet, syncSession); } public virtual SyncContext GetChanges(SyncGroupMetadata groupMetadata, SyncSession syncSession) { return this._serverSyncProvider.GetChanges(groupMetadata, syncSession); } public virtual SyncSchema GetSchema(Collection<string> tableNames, SyncSession syncSession) { return this._serverSyncProvider.GetSchema(tableNames, syncSession); } public virtual SyncServerInfo GetServerInfo(SyncSession syncSession) { return this._serverSyncProvider.GetServerInfo(syncSession); } } |
The designer will also move the CustomerSyncAdapter and the LocalDataCacheServerSyncProvider from the smart client application to the WCF service. The generated code for the WCF service implementation wraps the LocalDataCacheServerSyncProvider and provides its functionalities through the service contract. This would mean that the RemoteProvider property of the LocalDataCacheSyncAgent in the client application that had a reference to the LocalDataCacheServerSyncProvider earlier would have a null reference now since the LocalDataCacheServerSyncProvider was moved to the WCF service.
To enable synchronization from the client, we need to set the remote provider to invoke the WCF proxy from our smart client to communicate with the service for synchronization. Once we add the reference to the WCF service, we can alter the synchronization code as follows:
C# Code:
1 2 3 4 5 |
LocalDataCacheSyncAgent syncAgent = new LocalDataCacheSyncAgent(); syncAgent.Customer.SyncDirection = SyncDirection.Bidirectional; syncAgent.RemoteProvider = new ServerSyncProviderProxy( new ServiceReference.LocalDataCacheSyncContractClient()); SyncStatistics syncStats = syncAgent.Synchronize(); |
In the above code we are setting the remote provider of the LocalDataCacheSyncAgent to use the LocalDataCacheSyncContractClient client proxy as the provider for synchronization. When we call the Synchronize method, the WCF service would be invoked to retrieve changes from and apply changes to the server database.
Summary
Smart client applications these days require offline capabilities so as to continue to work even when disconnected. An ADO.NET synchronization service provides an extensible framework to easily create local databases and keep them synchronized with server databases. It also makes it easier to change the synchronization architecture to support different types of synchronization such as synchronizing through a WCF service.
Load comments