<rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/" xmlns:copyright="http://blogs.law.harvard.edu/tech/rss" xmlns:image="http://purl.org/rss/1.0/modules/image/">
    <channel>
        <title>Sync Framework</title>
        <link>http://geekswithblogs.net/argot/category/10803.aspx</link>
        <description>MS Sync Framework</description>
        <language>en-AU</language>
        <copyright>argot</copyright>
        <managingEditor>bruceworld@gmail.com</managingEditor>
        <generator>Subtext Version 0.0.0.0</generator>
        <item>
            <title>Sync Framework - Inefficient ApplyChanges</title>
            <link>http://geekswithblogs.net/argot/archive/2009/10/27/sync-framework---inefficient-applychanges.aspx</link>
            <description>&lt;p&gt;for the ado database sync, the method &lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;public virtual SyncContext ApplyChanges(SyncGroupMetadata groupMetadata, DataSet dataSet,SyncSession syncSession)&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;on server side is not efficient, as it receive the changed data from client side, but it again sent it back to client within the SyncContext. &lt;/p&gt;
&lt;p&gt;In the returned SyncContext object, I found DataSet and &lt;font face=""&gt;GroupProgress.Changes is almost the same as input dataset. by verifying the code inside sync framework, I found:&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;
&lt;table style="MARGIN-BOTTOM: 0px" cellspacing="0" cellpadding="0" width="100%"&gt;
    &lt;tbody&gt;
        &lt;tr&gt;
            &lt;td style="PADDING-BOTTOM: 4px; PADDING-LEFT: 5px; PADDING-RIGHT: 5px; PADDING-TOP: 4px" valign="top" colspan="2"&gt;
            &lt;pre&gt;&lt;font color="#1000a0"&gt;internal&lt;/font&gt; &lt;a title="Microsoft.Synchronization.Data.SyncGroupMetadata" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.SyncGroupMetadata"&gt;SyncGroupMetadata&lt;/a&gt; &lt;strong&gt;&lt;a class="bold" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/ResetProivderState(Microsoft.Synchronization.Data.SyncGroupMetadata,System.Data.DataSet):Microsoft.Synchronization.Data.SyncGroupMetadata"&gt;ResetProivderState&lt;/a&gt;&lt;/strong&gt;(&lt;a title="Microsoft.Synchronization.Data.SyncGroupMetadata" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.SyncGroupMetadata"&gt;SyncGroupMetadata&lt;/a&gt; groupMetadata, &lt;a title="System.Data.DataSet" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://System.Data:2.0.0.0:b77a5c561934e089/System.Data.DataSet"&gt;DataSet&lt;/a&gt; dataSet)
{
    &lt;font color="#1000a0"&gt;this&lt;/font&gt;.&lt;a title="object Microsoft.Synchronization.Data.Server.DbServerSyncProvider._rawOldAnchor;" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/_rawOldAnchor:Object"&gt;_rawOldAnchor&lt;/a&gt; = &lt;font color="#800000"&gt;null&lt;/font&gt;;
    &lt;font color="#1000a0"&gt;this&lt;/font&gt;.&lt;a title="object Microsoft.Synchronization.Data.Server.DbServerSyncProvider._rawNewAnchor;" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/_rawNewAnchor:Object"&gt;_rawNewAnchor&lt;/a&gt; = &lt;font color="#800000"&gt;null&lt;/font&gt;;
    &lt;font color="#1000a0"&gt;this&lt;/font&gt;.&lt;a title="object Microsoft.Synchronization.Data.Server.DbServerSyncProvider._rawMaxAnchor;" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/_rawMaxAnchor:Object"&gt;_rawMaxAnchor&lt;/a&gt; = &lt;font color="#800000"&gt;null&lt;/font&gt;;
    &lt;font color="#1000a0"&gt;this&lt;/font&gt;.&lt;a title="SyncContext Microsoft.Synchronization.Data.Server.DbServerSyncProvider._syncContext;" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/_syncContext:Microsoft.Synchronization.Data.SyncContext"&gt;_syncContext&lt;/a&gt; = &lt;font color="#1000a0"&gt;new&lt;/font&gt; &lt;a title="Microsoft.Synchronization.Data.SyncContext.SyncContext();" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.SyncContext/.ctor()"&gt;SyncContext&lt;/a&gt;();
    &lt;font color="#1000a0"&gt;this&lt;/font&gt;.&lt;a title="SyncContext Microsoft.Synchronization.Data.Server.DbServerSyncProvider._syncContext;" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/_syncContext:Microsoft.Synchronization.Data.SyncContext"&gt;_syncContext&lt;/a&gt;.&lt;a title="DataSet Microsoft.Synchronization.Data.SyncContext.DataSet { ... }" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.SyncContext/property:DataSet:System.Data.DataSet"&gt;DataSet&lt;/a&gt; = &lt;a title="DataSet dataSet; // Parameter"&gt;dataSet&lt;/a&gt;;
    &lt;a title="Microsoft.Synchronization.Data.SyncGroupMetadata" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.SyncGroupMetadata"&gt;SyncGroupMetadata&lt;/a&gt; &lt;strong&gt;metadata&lt;/strong&gt; = &lt;font color="#1000a0"&gt;this&lt;/font&gt;.&lt;a title="SyncGroupMetadata Microsoft.Synchronization.Data.Server.DbServerSyncProvider.GenerateOrderedGroupMetadata(SyncGroupMetadata groupMetadata);" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/GenerateOrderedGroupMetadata(Microsoft.Synchronization.Data.SyncGroupMetadata):Microsoft.Synchronization.Data.SyncGroupMetadata"&gt;GenerateOrderedGroupMetadata&lt;/a&gt;(&lt;a title="SyncGroupMetadata groupMetadata; // Parameter"&gt;groupMetadata&lt;/a&gt;);
    &lt;font color="#1000a0"&gt;this&lt;/font&gt;.&lt;a title="SyncContext Microsoft.Synchronization.Data.Server.DbServerSyncProvider._syncContext;" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data.Server:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.Server.DbServerSyncProvider/_syncContext:Microsoft.Synchronization.Data.SyncContext"&gt;_syncContext&lt;/a&gt;.&lt;a title="SyncGroupProgress Microsoft.Synchronization.Data.SyncContext.GroupProgress { ... }" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.SyncContext/property:GroupProgress:Microsoft.Synchronization.Data.SyncGroupProgress"&gt;GroupProgress&lt;/a&gt; = &lt;font color="#1000a0"&gt;new&lt;/font&gt; &lt;a title="Microsoft.Synchronization.Data.SyncGroupProgress.SyncGroupProgress(SyncGroupMetadata, DataSet);" href="http://www.aisto.com/roeder/dotnet/Default.aspx?Target=code://Microsoft.Synchronization.Data:3.0.0.0:89845dcd8080cc91/Microsoft.Synchronization.Data.SyncGroupProgress/.ctor(Microsoft.Synchronization.Data.SyncGroupMetadata,System.Data.DataSet)"&gt;SyncGroupProgress&lt;/a&gt;(&lt;a title="SyncGroupMetadata metadata // Local Variable"&gt;metadata&lt;/a&gt;, &lt;a title="DataSet dataSet; // Parameter"&gt;dataSet&lt;/a&gt;);
    &lt;font color="#1000a0"&gt;return&lt;/font&gt; &lt;a title="SyncGroupMetadata metadata // Local Variable"&gt;metadata&lt;/a&gt;;
}
&lt;/pre&gt;
            &lt;/td&gt;
        &lt;/tr&gt;
    &lt;/tbody&gt;
&lt;/table&gt;
&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;so I use the code below:&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;SyncContext context = _serverSyncProvider.ApplyChanges(groupMetadata, dataSet, syncSession);&lt;br /&gt;
context.DataSet = null;&lt;br /&gt;
&lt;/font&gt;&lt;font face=""&gt;context.GroupProgress.Changes.Clear();&lt;br /&gt;
&lt;/font&gt;&lt;font face=""&gt;return context; &lt;/font&gt;&lt;/p&gt;
&lt;p&gt;to make the size smaller.&lt;/p&gt; &lt;img src="http://geekswithblogs.net/argot/aggbug/135732.aspx" width="1" height="1" /&gt;</description>
            <dc:creator>argot</dc:creator>
            <guid>http://geekswithblogs.net/argot/archive/2009/10/27/sync-framework---inefficient-applychanges.aspx</guid>
            <pubDate>Tue, 27 Oct 2009 05:54:08 GMT</pubDate>
            <wfw:comment>http://geekswithblogs.net/argot/comments/135732.aspx</wfw:comment>
            <comments>http://geekswithblogs.net/argot/archive/2009/10/27/sync-framework---inefficient-applychanges.aspx#feedback</comments>
            <wfw:commentRss>http://geekswithblogs.net/argot/comments/commentRss/135732.aspx</wfw:commentRss>
        </item>
        <item>
            <title>Sync Framework Common Practise of ADO.NET</title>
            <link>http://geekswithblogs.net/argot/archive/2009/10/18/sync-framework-common-practise-of-ado.net.aspx</link>
            <description>&lt;p&gt;&lt;font face=""&gt;I have been working on database synchronization for a couple of weeks, and seems it at the final testing stage, I am using &lt;a href="http://msdn.microsoft.com/en-us/sync/default.aspx"&gt;Microsoft Sync Framework 2.0&lt;/a&gt;, the DB server is SQL server 2008 Standard, and clients are SQL Server 2008 Express as I use Change Tracking to track DB changes instead of TombStone Tables and Guid tracking columns, I use WCF+IIS+SSL to host server side service, the binding uses Gzip binary encoding.&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;The reason we are not using SQL Server Replication is because our tables need dynamic filters and some of them logic is quite complicated (we need to download the certain data to client according to the client name and setting), so when Replication try to generate the snapshot, it takes quite long time and CPU usage is quite high.&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;Coding didn't take me so long to finish (main part is write customized filter queries) but I realize some of the sync tables using bigint as primary key instead of Guid, so I have to assign a new seed to each client to avoid confliction. This new seed number will download and apply when clients reinitialize after schema download and applied.&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;Also, I found after "Applychanges" has been called, some tables identities will be changed, because in the "InsertCommand" of the Adaptor, using "SET IDENTITY_INSERT [tableName] ON", this will cause "If the value inserted is larger than the current identity value for the table, SQL Server automatically uses the new inserted value as the current identity value.", so we use a new table to hold the current identity for tables before applychanges and set it back after applychanges.&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;I have found when uploading changes to server side, the dataset contains all changed data, and after &lt;a href="http://msdn.microsoft.com/en-us/library/microsoft.synchronization.data.server.dbserversyncprovider.applychanges.aspx"&gt;ApplyChanges&lt;/a&gt; called, the return object &lt;a href="http://msdn.microsoft.com/en-us/library/microsoft.synchronization.data.synccontext(SQL.105).aspx"&gt;SyncContext&lt;/a&gt; still contains the dataset which sent to server, this is very inefficient, also when some changes apply failed, the return size grows very big because the &lt;a href="http://msdn.microsoft.com/en-us/library/microsoft.synchronization.data.synccontext.groupprogress(SQL.105).aspx"&gt;GroupProgress&lt;/a&gt; inside &lt;a href="http://msdn.microsoft.com/en-us/library/microsoft.synchronization.data.synccontext(SQL.105).aspx"&gt;SyncContext&lt;/a&gt; contains all dataset info plus failed table and row info, this makes memory growing very fast on server side as well as transferring.&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;Server side need to maintain a session as the the clients call is each client based, so the WCF interface is like this:&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;[ServiceContract(SessionMode = SessionMode.Required)]&lt;br /&gt;
    public interface IServiceForSync&lt;br /&gt;
    {&lt;br /&gt;
        [OperationContract(IsInitiating = true)]&lt;br /&gt;
        bool InitialiseSetting();&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        SyncContext ApplyChanges(SyncGroupMetadata groupMetadata, DataSet dataSet, SyncSession syncSession);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        SyncContext GetChanges(SyncGroupMetadata groupMetadata, SyncSession syncSession);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        SyncSchema GetSchema(Collection&amp;lt;string&amp;gt; tableNames, SyncSession syncSession);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        SyncServerInfo GetServerInfo(SyncSession syncSession);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        SyncInfo GetSyncInfo(string SyncHostName, long rpuid);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        List&amp;lt;String&amp;gt; GetInitSchema();&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        List&amp;lt;String&amp;gt; GetUpdateSchema(string SyncHostName);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsInitiating = false)]&lt;br /&gt;
        bool ResetStatus(string SyncHostName);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;        [OperationContract(IsTerminating = true)]&lt;br /&gt;
        void EndSession(); &lt;br /&gt;
    }&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;The Service side binding in the web.config is like this:&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;   &amp;lt;customBinding&amp;gt;&lt;br /&gt;
    &amp;lt;binding name="ISyncServer" receiveTimeout="00:30:00" sendTimeout="00:30:00"&amp;gt;&lt;br /&gt;
     &amp;lt;gzipMessageEncoding innerMessageEncoding="binaryMessageEncoding"/&amp;gt;&lt;br /&gt;
     &amp;lt;reliableSession ordered="true" inactivityTimeout="00:10:00" maxPendingChannels="128"/&amp;gt;&lt;br /&gt;
     &amp;lt;httpsTransport hostNameComparisonMode="StrongWildcard" manualAddressing="False" maxReceivedMessageSize="2000000000" authenticationScheme="Anonymous" bypassProxyOnLocal="False" realm="" useDefaultWebProxy="True"/&amp;gt;&lt;br /&gt;
    &amp;lt;/binding&amp;gt;&lt;br /&gt;
   &amp;lt;/customBinding&amp;gt;&lt;/font&gt;&lt;/p&gt;
&lt;p&gt; &lt;/p&gt;
&lt;p&gt;On client side the remoteServer of the syncAgent will be the proxy, and I use one SyncGroup as any exception happens, it will reverse back to original.&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;public class SqlExpressClientSyncProvider : ClientSyncProvider&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;and in the ApplyChanges of the ClientSyncProvider, we need to &lt;font face=""&gt;Map SyncDirection from client point of view to our internal server point of view, so part of code is like:&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt; foreach (var tableMetadata in groupMetadata.TablesMetadata)&lt;br /&gt;
            {&lt;br /&gt;
                switch (tableMetadata.SyncDirection)&lt;br /&gt;
                {&lt;br /&gt;
                    case SyncDirection.Snapshot:&lt;br /&gt;
                    case SyncDirection.DownloadOnly:&lt;br /&gt;
                        tableMetadata.SyncDirection = SyncDirection.UploadOnly;&lt;br /&gt;
                        break;&lt;br /&gt;
                    case SyncDirection.UploadOnly:&lt;br /&gt;
                        tableMetadata.SyncDirection = SyncDirection.DownloadOnly;&lt;br /&gt;
                        break;&lt;br /&gt;
                }&lt;br /&gt;
            }&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;in GetChanges, we need to swap anchors:&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;public override SyncContext GetChanges(SyncGroupMetadata groupMetadata, SyncSession syncSession)&lt;br /&gt;
        {&lt;br /&gt;
            // neet to set the LastReceivedAnchor as the LastSentAnchor since &lt;br /&gt;
            // DbServerSyncProvider operates from the server's perspective, so&lt;br /&gt;
            // we swap the two fields temporarily. &lt;br /&gt;
            foreach (var metaTable in groupMetadata.TablesMetadata)&lt;br /&gt;
            {&lt;br /&gt;
                var temp = metaTable.LastReceivedAnchor;&lt;br /&gt;
                metaTable.LastReceivedAnchor = metaTable.LastSentAnchor;&lt;br /&gt;
                metaTable.LastSentAnchor = temp;&lt;br /&gt;
            }&lt;br /&gt;
           &lt;br /&gt;
            var context = _dbSyncProvider.GetChanges(groupMetadata, syncSession);&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=""&gt;&lt;br /&gt;
            if (_bEnableTimeAdjusting.HasValue &amp;amp;&amp;amp; _bEnableTimeAdjusting.Value)&lt;br /&gt;
                AdjustDSTimeZone(context.DataSet, "CLIENT"); //-TN - adjust all datetime columns before sync.&lt;br /&gt;
            //swap them back for consistency&lt;br /&gt;
            foreach (var metaTable in groupMetadata.TablesMetadata)&lt;br /&gt;
            {&lt;br /&gt;
                var temp = metaTable.LastReceivedAnchor;&lt;br /&gt;
                metaTable.LastReceivedAnchor = metaTable.LastSentAnchor;&lt;br /&gt;
                metaTable.LastSentAnchor = temp;&lt;br /&gt;
            }&lt;br /&gt;
            return context;&lt;br /&gt;
        }&lt;/font&gt;&lt;/p&gt;
&lt;p&gt; &lt;/p&gt;
&lt;p&gt; &lt;/p&gt;
&lt;p&gt; &lt;/p&gt; &lt;img src="http://geekswithblogs.net/argot/aggbug/135541.aspx" width="1" height="1" /&gt;</description>
            <dc:creator>argot</dc:creator>
            <guid>http://geekswithblogs.net/argot/archive/2009/10/18/sync-framework-common-practise-of-ado.net.aspx</guid>
            <pubDate>Sun, 18 Oct 2009 13:30:37 GMT</pubDate>
            <wfw:comment>http://geekswithblogs.net/argot/comments/135541.aspx</wfw:comment>
            <comments>http://geekswithblogs.net/argot/archive/2009/10/18/sync-framework-common-practise-of-ado.net.aspx#feedback</comments>
            <slash:comments>9</slash:comments>
            <wfw:commentRss>http://geekswithblogs.net/argot/comments/commentRss/135541.aspx</wfw:commentRss>
        </item>
    </channel>
</rss>
