001    /*
002     * CDDL HEADER START
003     *
004     * The contents of this file are subject to the terms of the
005     * Common Development and Distribution License, Version 1.0 only
006     * (the "License").  You may not use this file except in compliance
007     * with the License.
008     *
009     * You can obtain a copy of the license at
010     * docs/licenses/cddl.txt
011     * or http://www.opensource.org/licenses/cddl1.php.
012     * See the License for the specific language governing permissions
013     * and limitations under the License.
014     *
015     * When distributing Covered Code, include this CDDL HEADER in each
016     * file and include the License file at
017     * docs/licenses/cddl.txt.  If applicable,
018     * add the following below this CDDL HEADER, with the fields enclosed
019     * by brackets "[]" replaced with your own identifying information:
020     *      Portions Copyright [yyyy] [name of copyright owner]
021     *
022     * CDDL HEADER END
023     *
024     *
025     *      Copyright 2010-2013 UnboundID Corp.
026     */
027    package com.unboundid.directory.sdk.sync.api;
028    
029    
030    
031    import java.io.Serializable;
032    import java.sql.SQLException;
033    import java.util.Collections;
034    import java.util.Iterator;
035    import java.util.LinkedList;
036    import java.util.List;
037    import java.util.Map;
038    import java.util.concurrent.BlockingQueue;
039    import java.util.concurrent.atomic.AtomicLong;
040    
041    import com.unboundid.directory.sdk.common.internal.Configurable;
042    import com.unboundid.directory.sdk.common.internal.ExampleUsageProvider;
043    import com.unboundid.directory.sdk.common.internal.UnboundIDExtension;
044    import com.unboundid.directory.sdk.sync.config.JDBCSyncSourceConfig;
045    import com.unboundid.directory.sdk.sync.internal.SynchronizationServerExtension;
046    import com.unboundid.directory.sdk.sync.types.DatabaseChangeRecord;
047    import com.unboundid.directory.sdk.sync.types.SetStartpointOptions;
048    import com.unboundid.directory.sdk.sync.types.SyncOperation;
049    import com.unboundid.directory.sdk.sync.types.SyncServerContext;
050    import com.unboundid.directory.sdk.sync.types.TransactionContext;
051    import com.unboundid.ldap.sdk.Entry;
052    import com.unboundid.util.Extensible;
053    import com.unboundid.util.ThreadSafety;
054    import com.unboundid.util.ThreadSafetyLevel;
055    import com.unboundid.util.args.ArgumentException;
056    import com.unboundid.util.args.ArgumentParser;
057    
058    
059    
060    /**
061     * This class defines an API that must be implemented by extensions
062     * in order to synchronize data out of a relational database. Since the
063     * UnboundID Synchronization Server is LDAP-centric,
064     * this API allows you to take database content and convert it into LDAP
065     * entries which can then be processed by the Synchronization Server. The
066     * lifecycle of
067     * a sync operation is as follows:
068     * <ol>
069     * <li>Detect change at the synchronization source</li>
070     * <li>Fetch full source entry</li>
071     * <li>Perform any mappings and compute the equivalent destination entry</li>
072     * <li>Fetch full destination entry</li>
073     * <li>Diff the computed destination entry and actual (fetched) destination
074     * entry</li>
075     * <li>Apply the minimal set of changes at the destination to bring it in sync
076     * </li>
077     * </ol>
078     * This implies that the
079     * {@link #fetchEntry(TransactionContext, SyncOperation)} method will be
080     * called once for every change that is returned by
081     * {@link #getNextBatchOfChanges(TransactionContext, int, AtomicLong)}.
082     * <p>
083     * In several places a {@link TransactionContext} is provided, which allows
084     * controlled access to the target database. By default, methods in this class
085     * are always provided with a fresh connection (i.e. a new transaction), and the
086     * Synchronization Server will always commit or rollback the transaction
087     * automatically, depending on whether the method returned normally or threw an
088     * exception. Implementers may optionally perform their own transaction
089     * management within these methods if necessary.
090     * <p>
091     * Several of these methods throw {@link SQLException}, which should be used in
092     * the case of any database access error. For other types of errors, runtime
093     * exceptions may be used (IllegalStateException, NullPointerException, etc.).
094     * The Synchronization Server will automatically retry operations that fail, up
095     * to a configurable amount of attempts. The exception to this rule is if a
096     * SQLException is thrown with a SQL state string beginning with "08"; this
097     * indicates a connection error, and in this case the operation is retried
098     * indefinitely.
099     * <BR>
100     * <H2>Configuring JDBC Sync Sources</H2>
101     * In order to configure a JDBC sync source based on this API and
102     * written in Java, use a command like:
103     * <PRE>
104     *      dsconfig create-sync-source \
105     *           --source-name "<I>{source-name}</I>" \
106     *           --type third-party-jdbc \
107     *           --set "server:{server-name}" \
108     *           --set "extension-class:<I>{class-name}</I>" \
109     *           --set "extension-argument:<I>{name=value}</I>"
110     * </PRE>
111     * where "<I>{source-name}</I>" is the name to use for the JDBC sync source
112     * instance, "<I>{server-name}</I>" is the name of the JDBC external server that
113     * will be used as the sync source, "<I>{class-name}</I>" is the fully-qualified
114     * name of the Java class written using this API, and "<I>{name=value}</I>"
115     * represents name-value pairs for any arguments to provide to the JDBC sync
116     * source.  If multiple arguments should be provided to the JDBC sync source,
117     * then the "<CODE>--set extension-argument:<I>{name=value}</I></CODE>" option
118     * should be provided multiple times.
119     */
120    @Extensible()
121    @SynchronizationServerExtension(appliesToLocalContent=false,
122                                    appliesToSynchronizedContent=true)
123    public abstract class JDBCSyncSource
124           implements UnboundIDExtension,
125                      Configurable,
126                      ExampleUsageProvider
127    {
128      /**
129       * Creates a new instance of this JDBC Sync Source.  All sync
130       * source implementations must include a default constructor, but any
131       * initialization should generally be done in the
132       * {@link #initializeJDBCSyncSource} method.
133       */
134      public JDBCSyncSource()
135      {
136        // No implementation is required.
137      }
138    
139    
140    
141      /**
142       * {@inheritDoc}
143       */
144      public abstract String getExtensionName();
145    
146    
147    
148      /**
149       * {@inheritDoc}
150       */
151      public abstract String[] getExtensionDescription();
152    
153    
154    
155      /**
156       * {@inheritDoc}
157       */
158      public Map<List<String>,String> getExamplesArgumentSets()
159      {
160        return Collections.emptyMap();
161      }
162    
163    
164    
165      /**
166       * {@inheritDoc}
167       */
168      public void defineConfigArguments(final ArgumentParser parser)
169             throws ArgumentException
170      {
171        // No arguments will be allowed by default.
172      }
173    
174    
175    
176      /**
177       * This hook is called when a Sync Pipe first starts up, when the
178       * <i>resync</i> process first starts up, or when the set-startpoint
179       * subcommand is called from the <i>realtime-sync</i> command line tool.
180       * Any initialization of this sync source should be performed here. This
181       * method should generally store the {@link SyncServerContext} in a class
182       * member so that it can be used elsewhere in the implementation.
183       * <p>
184       * A {@link TransactionContext} is provided, which allows
185       * controlled access to the target database. The context will contain a fresh
186       * fresh connection (i.e. a new transaction), and the Synchronization Server
187       * will always commit or rollback the transaction automatically, depending on
188       * whether this method returns normally or throws an exception. Implementers
189       * may optionally perform their own transaction management within this method
190       * if necessary.
191       * <p>
192       * The default implementation is empty.
193       *
194       * @param ctx
195       *          a TransactionContext which provides a valid JDBC connection to the
196       *          database.
197       * @param  serverContext  A handle to the server context for the server in
198       *                        which this extension is running.
199       * @param  config         The general configuration for this sync source.
200       * @param  parser         The argument parser which has been initialized from
201       *                        the configuration for this JDBC sync source.
202       */
203      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
204      public void initializeJDBCSyncSource(final TransactionContext ctx,
205                                              final SyncServerContext serverContext,
206                                              final JDBCSyncSourceConfig config,
207                                              final ArgumentParser parser)
208      {
209        // No initialization will be performed by default.
210      }
211    
212    
213    
214      /**
215       * This hook is called when a Sync Pipe shuts down, when the <i>resync</i>
216       * process shuts down, or when the set-startpoint subcommand (from the
217       * <i>realtime-sync</i> command line tool) is finished. Any clean up of this
218       * sync source should be performed here.
219       * <p>
220       * A {@link TransactionContext} is provided, which allows
221       * controlled access to the target database. The context will contain a fresh
222       * fresh connection (i.e. a new transaction), and the Synchronization Server
223       * will always commit or rollback the transaction automatically, depending on
224       * whether this method returns normally or throws an exception. Implementers
225       * may optionally perform their own transaction management within this method
226       * if necessary.
227       * <p>
228       * The default implementation is empty.
229       *
230       * @param ctx
231       *          a TransactionContext which provides a valid JDBC connection to the
232       *          database.
233       */
234      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
235      public void finalizeJDBCSyncSource(final TransactionContext ctx)
236      {
237        //No implementation required by default.
238      }
239    
240    
241    
242      /**
243       * This method should effectively set the starting point for synchronization
244       * to the place specified by the <code>options</code> parameter. This should
245       * cause all changes previous to the specified start point to be disregarded
246       * and only changes after that point to be returned by
247       * {@link #getNextBatchOfChanges(TransactionContext, int, AtomicLong)}.
248       * <p>
249       * There are several different startpoint types (see
250       * {@link SetStartpointOptions}), and this implementation is not required to
251       * support them all. If the specified startpoint type is unsupported, this
252       * method should throw an {@link IllegalArgumentException}.
253       * <p>
254       * <b>IMPORTANT</b>: The <code>RESUME_AT_SERIALIZABLE</code> startpoint type
255       * must be supported by your implementation, because this is used when a Sync
256       * Pipe first starts up.
257       * <p>
258       * This method can be called from two different contexts:
259       * <ul>
260       * <li>When the 'set-startpoint' subcommand of the realtime-sync CLI is used
261       * (the Sync Pipe is required to be stopped in this context)</li>
262       * <li>Immediately after a Sync Pipe starts up and a connection is first
263       *     established to the source server (e.g. before the first call to
264       * {@link #getNextBatchOfChanges(TransactionContext, int, AtomicLong)})</li>
265       * </ul>
266       * <p>
267       * A {@link TransactionContext} is provided, which allows
268       * controlled access to the target database. The context will contain a fresh
269       * fresh connection (i.e. a new transaction), and the Synchronization Server
270       * will always commit or rollback the transaction automatically, depending on
271       * whether this method returns normally or throws an exception. Implementers
272       * may optionally perform their own transaction management within this method
273       * if necessary.
274       *
275       * @param ctx
276       *          a TransactionContext which provides a valid JDBC connection to the
277       *          database.
278       * @param options
279       *          an object which indicates where exactly to start synchronizing
280       *          (e.g.
281       *          the end of the changelog, specific change number, a certain time
282       *          ago, etc)
283       * @throws SQLException
284       *           if there is any error while setting the start point
285       */
286      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
287      public abstract void setStartpoint(final TransactionContext ctx,
288                                         final SetStartpointOptions options)
289                                              throws SQLException;
290    
291    
292    
293      /**
294       * Gets the current value of the startpoint for change detection. This is the
295       * "bookmark" which indicates which changes have already been processed and
296       * which have not. In most cases, a change number is used to detect changes
297       * and is managed by the Synchronization Server, in which case this
298       * implementation needs only to return the latest acknowledged
299       * change number. In other cases, the return value may correspond to a
300       * different value, such as the SYS_CHANGE_VERSION in Microsoft SQL Server.
301       * In any case, this method should return the value that is updated by
302       * {@link #acknowledgeCompletedOps(TransactionContext, LinkedList)}.
303       * <p>
304       * This method is called periodically and the return value is saved in the
305       * persistent state for the Sync Pipe that uses this script as its Sync
306       * Source.
307       * <p>
308       * <b>IMPORTANT</b>: The internal value for the startpoint should only be
309       * updated after a sync operation is acknowledged back to this script (via
310       * {@link #acknowledgeCompletedOps(TransactionContext, LinkedList)}).
311       * Otherwise it will be possible for changes to be missed when the
312       * Synchronization Server is restarted or a connection error occurs.
313       * @return a value to store in the persistent state for the Sync Pipe. This is
314       *         usually
315       *         a change number, but if a changelog table is not used to detect
316       *         changes,
317       *         this value should represent some other token to pass into
318       *         {@link #setStartpoint(TransactionContext, SetStartpointOptions)}
319       *         when the sync pipe starts up.
320       */
321      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
322      public abstract Serializable getStartpoint();
323    
324    
325    
326      /**
327       * Return the next batch of change records from the database. Change records
328       * are just hints that a change happened; they do not include the actual data
329       * of the change. In an effort to never synchronize stale data, the
330       * Synchronization Server will go back and fetch the full source entry for
331       * each change record.
332       * <p>
333       * On the first invocation, this should return changes starting from the
334       * startpoint that was set by
335       * {@link #setStartpoint(TransactionContext, SetStartpointOptions)}. This
336       * method is responsible for updating the internal state such that subsequent
337       * invocations do not return duplicate changes.
338       * <p>
339       * The resulting list should be limited by <code>maxChanges</code>. The
340       * <code>numStillPending</code> reference should be set to the estimated
341       * number of changes that haven't yet been retrieved from the changelog table
342       * when this method returns, or zero if all the current changes have been
343       * retrieved.
344       * <p>
345       * <b>IMPORTANT</b>: While this method needs to keep track of which changes
346       * have already been returned so that it does not return them again, it should
347       * <b>NOT</b> modify the official startpoint. The internal value for the
348       * startpoint should only be updated after a sync operation is acknowledged
349       * back to this script (via
350       * {@link #acknowledgeCompletedOps(TransactionContext, LinkedList)}).
351       * Otherwise it will be possible for changes to be missed when the
352       * Synchronization Server is restarted or a connection error occurs. The
353       * startpoint should not change as a result of this method.
354       * <p>
355       * A {@link TransactionContext} is provided, which allows
356       * controlled access to the target database. The context will contain a fresh
357       * fresh connection (i.e. a new transaction), and the Synchronization Server
358       * will always commit or rollback the transaction automatically, depending on
359       * whether this method returns normally or throws an exception. Implementers
360       * may optionally perform their own transaction management within this method
361       * if necessary.
362       * <p>
363       * This method <b>does not need to be thread-safe</b>. It will be invoked
364       * repeatedly by a single thread, based on the polling interval set in the
365       * Sync Pipe configuration.
366       * @param ctx
367       *          a TransactionContext which provides a valid JDBC connection to the
368       *          database.
369       * @param maxChanges
370       *          the maximum number of changes to retrieve
371       * @param numStillPending
372       *          this should be set to the number of unretrieved changes that
373       *          are still pending after this batch has been retrieved. This will
374       *          be passed in
375       *          as zero, and may be left that way if the actual value cannot be
376       *          determined.
377       * @return a list of {@link DatabaseChangeRecord} instances, each
378       *         corresponding
379       *         to a row in the changelog table (or the equivalent if some other
380       *         change
381       *         tracking mechanism is being used). If there are no new changes to
382       *         return, this
383       *         method should return an empty list.
384       * @throws SQLException
385       *           if there is any error while retrieving the next batch of changes
386       */
387      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
388      public abstract List<DatabaseChangeRecord> getNextBatchOfChanges(
389                                                  final TransactionContext ctx,
390                                                  final int maxChanges,
391                                                  final AtomicLong numStillPending)
392                                                          throws SQLException;
393    
394    
395    
396      /**
397       * Return a full source entry (in LDAP form) from the database, corresponding
398       * to the {@link DatabaseChangeRecord} that is passed in through the
399       * {@link SyncOperation}. This method should perform any queries necessary to
400       * gather the latest values for all the attributes to be synchronized.
401       * <p>
402       * A {@link TransactionContext} is provided, which allows
403       * controlled access to the target database. The context will contain a fresh
404       * fresh connection (i.e. a new transaction), and the Synchronization Server
405       * will always commit or rollback the transaction automatically, depending on
406       * whether this method returns normally or throws an exception. Implementers
407       * may optionally perform their own transaction management within this method
408       * if necessary.
409       * <p>
410       * This method <b>must be thread safe</b>, as it will be called repeatedly and
411       * concurrently by each of the Sync Pipe worker threads as they process
412       * entries.
413       * @param ctx
414       *          a TransactionContext which provides a valid JDBC connection to the
415       *          database.
416       * @param operation
417       *          the SyncOperation which identifies the database "entry" to
418       *          fetch. The DatabaseChangeRecord can be obtained by calling
419       *          <code>operation.getDatabaseChangeRecord()</code>.
420       *          This is returned by
421       *        {@link #getNextBatchOfChanges(TransactionContext, int, AtomicLong)}
422       *          or by
423       *        {@link #listAllEntries(TransactionContext, String, BlockingQueue)}
424       *          .
425       * @return a full LDAP Entry, or null if no such entry exists.
426       * @throws SQLException
427       *           if there is an error fetching the entry
428       */
429      @ThreadSafety(level = ThreadSafetyLevel.METHOD_THREADSAFE)
430      public abstract Entry fetchEntry(final TransactionContext ctx,
431                                       final SyncOperation operation)
432                                                throws SQLException;
433    
434    
435    
436      /**
437       * Provides a way for the Synchronization Server to acknowledge back to the
438       * script which sync operations it has processed. This method should update
439       * the official startpoint which was set by
440       * {@link #setStartpoint(TransactionContext, SetStartpointOptions)} and is
441       * returned by {@link #getStartpoint()}.
442       * <p>
443       * <b>IMPORTANT</b>: The internal value for the startpoint should only be
444       * updated after a sync operation is acknowledged back to this script (via
445       * this method). Otherwise it will be possible for changes to be missed when
446       * the Synchronization Server is restarted or a connection error occurs.
447       * <p>
448       * A {@link TransactionContext} is provided in case the acknowledgment needs
449       * to make it all the way back to the database itself (for example if you were
450       * using Oracle's Change Data Capture). The context will contain a fresh
451       * fresh connection (i.e. a new transaction), and the Synchronization Server
452       * will always commit or rollback the transaction automatically, depending on
453       * whether this method returns normally or throws an exception. Implementers
454       * may optionally perform their own transaction management within this method
455       * if necessary.
456       *
457       * @param ctx
458       *          a TransactionContext which provides a valid JDBC connection to the
459       *          database.
460       * @param completedOps
461       *          a list of {@link SyncOperation}s that have finished processing.
462       *          The records are listed in the order they were first detected.
463       * @throws SQLException
464       *           if there is an error acknowledging the changes back to the
465       *           database
466       */
467      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
468      public abstract void acknowledgeCompletedOps(
469                                       final TransactionContext ctx,
470                                       final LinkedList<SyncOperation> completedOps)
471                                                 throws SQLException;
472    
473    
474    
475      /**
476       * Performs a cleanup of the changelog table (if desired). There is a
477       * background thread that periodically invokes this method. It should remove
478       * any rows in the changelog table that are more than
479       * <code>maxAgeMillis</code> milliseconds old.
480       * <p>
481       * <b>NOTE:</b> If the system clock on the database server is not in sync with
482       * the system clock on the Synchronization Server, this method should query
483       * the database for its current time in order to determine the cut-off point
484       * for deleting changelog records.
485       * <p>
486       * A {@link TransactionContext} is provided, which allows
487       * controlled access to the target database. The context will contain a fresh
488       * fresh connection (i.e. a new transaction), and the Synchronization Server
489       * will always commit or rollback the transaction automatically, depending on
490       * whether this method returns normally or throws an exception. Implementers
491       * may optionally perform their own transaction management within this method
492       * if necessary.
493       * <p>
494       * If a separate mechanism will be used to manage the changelog table, this
495       * method may be implemented as a no-op and always return zero. This is how
496       * the default implementation behaves.
497       * @param ctx
498       *          a TransactionContext which provides a valid JDBC connection to the
499       *          database.
500       * @param maxAgeMillis
501       *          the period of time (in milliseconds) after which a changelog table
502       *          record should be deleted
503       * @return the number of rows that were deleted from the changelog table
504       * @throws SQLException
505       *           if there is an error purging records from the changelog table
506       */
507      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
508      public int cleanupChangelog(final TransactionContext ctx,
509                                  final long maxAgeMillis)
510                                      throws SQLException
511      {
512        //no implementation provided by default; this is an opt-in feature.
513        return 0;
514      }
515    
516    
517    
518      /**
519       * Gets a list of all the entries in the database for a given entry type. This
520       * is used by the 'resync' command line tool. The default implementation
521       * throws a {@link UnsupportedOperationException}; subclasses should override
522       * if the resync functionality is needed.
523       * <p>
524       * The <code>entryType</code> is user-defined; it will be
525       * passed in on the command line for resync. The <code>outputQueue</code>
526       * should contain {@link DatabaseChangeRecord} objects with the
527       * <code>ChangeType</code> set to <i>resync</i>.
528       * <p>
529       * This method should not return until all the entries of the given entryType
530       * have been added to the output queue. Separate threads will concurrently
531       * drain entries from the queue and process them. The queue should not
532       * actually contain full entries, but rather DatabaseChangeRecord objects
533       * which identify the full database entries. These objects are then
534       * individually passed in to
535       * {@link #fetchEntry(TransactionContext, SyncOperation)}. Therefore,
536       * it is important to make sure that the DatabaseChangeRecord instances
537       * contain enough identifiable information (e.g. primary keys) for each entry
538       * so that the entry can be found again.
539       * <p>
540       * The lifecycle of resync is similar to that of real-time sync, with a few
541       * differences:
542       * <ol>
543       * <li>Stream out a list of all IDs in the database (for a given entryType)
544       * </li>
545       * <li>Fetch full source entry for an ID</li>
546       * <li>Perform any mappings and compute the equivalent destination entry</li>
547       * <li>Fetch full destination entry</li>
548       * <li>Diff the computed destination entry and actual destination entry</li>
549       * <li>Apply the minimal set of changes at the destination to bring it in sync
550       * </li>
551       * </ol>
552       * If the total set of entries is very large, it is fine to split up the work
553       * into multiple database queries within this method. The queue will not grow
554       * out of control because it blocks when it becomes full. The queue capacity
555       * is fixed at 1000.
556       * <p>
557       * A {@link TransactionContext} is provided, which allows
558       * controlled access to the target database. The context will contain a fresh
559       * fresh connection (i.e. a new transaction), and the Synchronization Server
560       * will always commit or rollback the transaction automatically, depending on
561       * whether this method returns normally or throws an exception. Implementers
562       * may optionally perform their own transaction management within this method
563       * if necessary.
564       *
565       * @param ctx
566       *          a TransactionContext which provides a valid JDBC connection to the
567       *          database.
568       * @param entryType
569       *          the type of database entry to be fetched (this is specified
570       *          on the CLI for the resync command)
571       * @param outputQueue
572       *          a queue of DatabaseChangeRecord objects which will be individually
573       *          fetched via {@link #fetchEntry(TransactionContext, SyncOperation)}
574       * @throws SQLException
575       *           if there is an error retrieving the list of entries to resync
576       */
577      @ThreadSafety(level = ThreadSafetyLevel.METHOD_NOT_THREADSAFE)
578      public void listAllEntries(final TransactionContext ctx,
579                                 final String entryType,
580                                 final BlockingQueue<DatabaseChangeRecord>
581                                           outputQueue) throws SQLException
582      {
583        throw new UnsupportedOperationException();
584      }
585    
586    
587    
588      /**
589       * Note: This method is deprecated and may be removed in a future release.
590       * All new and existing code should be changed to use the version of this
591       * method which includes the <code>entryType</code> parameter.
592       * <p>
593       * Gets a list of all the entries in the database from a given file input.
594       * This is used by the 'resync' command line tool. The default implementation
595       * throws a {@link UnsupportedOperationException}; subclasses should override
596       * if the resync functionality is needed for specific database records, which
597       * can be specified in the input file.
598       * <p>
599       * The format for the <code>inputLines</code> (e.g. the content of the file)
600       * is user-defined; it may be key/value pairs, primary keys, or full SQL
601       * statements, for example. The use of this method is triggered via the
602       * <i>--sourceInputFile</i> argument on the resync CLI. The
603       * <code>outputQueue</code> should contain {@link DatabaseChangeRecord}
604       * objects with the <code>ChangeType</code> set to <i>resync</i>.
605       * <p>
606       * This method should not return until all the entries specified by the input
607       * file have been added to the output queue. Separate threads will
608       * concurrently drain entries from the queue and process them. The queue
609       * should not actually contain full entries, but rather DatabaseChangeRecord
610       * objects which identify the full database entries. These objects are then
611       * individually passed in to
612       * {@link #fetchEntry(TransactionContext, SyncOperation)}. Therefore,
613       * it is important to make sure that the DatabaseChangeRecord instances
614       * contain enough identifiable information (e.g. primary keys) for each entry
615       * so that the entry can be found again.
616       * <p>
617       * The lifecycle of resync is similar to that of real-time sync, with a few
618       * differences:
619       * <ol>
620       * <li>Stream out a list of all IDs in the database (using the given input
621       *  file)</li>
622       * <li>Fetch full source entry for an ID</li>
623       * <li>Perform any mappings and compute the equivalent destination entry</li>
624       * <li>Fetch full destination entry</li>
625       * <li>Diff the computed destination entry and actual destination entry</li>
626       * <li>Apply the minimal set of changes at the destination to bring it in sync
627       * </li>
628       * </ol>
629       * If the total set of entries is very large, it is fine to split up the work
630       * into multiple database queries within this method. The queue will not grow
631       * out of control because it blocks when it becomes full. The queue capacity
632       * is fixed at 1000.
633       * <p>
634       * A {@link TransactionContext} is provided, which allows
635       * controlled access to the target database. The context will contain a fresh
636       * fresh connection (i.e. a new transaction), and the Synchronization Server
637       * will always commit or rollback the transaction automatically, depending on
638       * whether this method returns normally or throws an exception. Implementers
639       * may optionally perform their own transaction management within this method
640       * if necessary.
641       *
642       * @param ctx
643       *          a TransactionContext which provides a valid JDBC connection to the
644       *          database.
645       * @param inputLines
646       *          an Iterator containing the lines from the specified input file to
647       *          resync (this is specified on the CLI for the resync command).
648       *          These lines can be any format, for example a set of primary keys,
649       *          a set of WHERE clauses, a set of full SQL queries, etc.
650       * @param outputQueue
651       *          a queue of DatabaseChangeRecord objects which will be individually
652       *          fetched via {@link #fetchEntry(TransactionContext, SyncOperation)}
653       * @throws SQLException
654       *           if there is an error retrieving the list of entries to resync
655       */
656      @Deprecated()
657      public void listAllEntries(final TransactionContext ctx,
658                                 final Iterator<String> inputLines,
659                                 final BlockingQueue<DatabaseChangeRecord>
660                                          outputQueue) throws SQLException
661      {
662        throw new UnsupportedOperationException();
663      }
664    
665    
666      /**
667       * Gets a list of all the entries in the database from a given file input.
668       * This is used by the 'resync' command line tool. The default implementation
669       * throws a {@link UnsupportedOperationException}; subclasses should override
670       * if the resync functionality is needed for specific database records, which
671       * can be specified in the input file.
672       * <p>
673       * The format for the <code>inputLines</code> (e.g. the content of the file)
674       * is user-defined; it may be key/value pairs, primary keys, or full SQL
675       * statements, for example. The use of this method is triggered via the
676       * <i>--sourceInputFile</i> argument on the resync CLI. The
677       * <code>outputQueue</code> should contain {@link DatabaseChangeRecord}
678       * objects with the <code>ChangeType</code> set to <i>resync</i>.
679       * <p>
680       * This method should not return until all the entries specified by the input
681       * file have been added to the output queue. Separate threads will
682       * concurrently drain entries from the queue and process them. The queue
683       * should not actually contain full entries, but rather DatabaseChangeRecord
684       * objects which identify the full database entries. These objects are then
685       * individually passed in to
686       * {@link #fetchEntry(TransactionContext, SyncOperation)}. Therefore,
687       * it is important to make sure that the DatabaseChangeRecord instances
688       * contain enough identifiable information (e.g. primary keys) for each entry
689       * so that the entry can be found again.
690       * <p>
691       * The lifecycle of resync is similar to that of real-time sync, with a few
692       * differences:
693       * <ol>
694       * <li>Stream out a list of all IDs in the database (using the given input
695       *  file)</li>
696       * <li>Fetch full source entry for an ID</li>
697       * <li>Perform any mappings and compute the equivalent destination entry</li>
698       * <li>Fetch full destination entry</li>
699       * <li>Diff the computed destination entry and actual destination entry</li>
700       * <li>Apply the minimal set of changes at the destination to bring it in sync
701       * </li>
702       * </ol>
703       * If the total set of entries is very large, it is fine to split up the work
704       * into multiple database queries within this method. The queue will not grow
705       * out of control because it blocks when it becomes full. The queue capacity
706       * is fixed at 1000.
707       * <p>
708       * A {@link TransactionContext} is provided, which allows
709       * controlled access to the target database. The context will contain a fresh
710       * fresh connection (i.e. a new transaction), and the Synchronization Server
711       * will always commit or rollback the transaction automatically, depending on
712       * whether this method returns normally or throws an exception. Implementers
713       * may optionally perform their own transaction management within this method
714       * if necessary.
715       *
716       * @param ctx
717       *          a TransactionContext which provides a valid JDBC connection to the
718       *          database.
719       * @param entryType
720       *          the type of database entry to be fetched (this is specified
721       *          on the CLI for the resync command)
722       * @param inputLines
723       *          an Iterator containing the lines from the specified input file to
724       *          resync (this is specified on the CLI for the resync command).
725       *          These lines can be any format, for example a set of primary keys,
726       *          a set of WHERE clauses, a set of full SQL queries, etc.
727       * @param outputQueue
728       *          a queue of DatabaseChangeRecord objects which will be individually
729       *          fetched via {@link #fetchEntry(TransactionContext, SyncOperation)}
730       * @throws SQLException
731       *           if there is an error retrieving the list of entries to resync
732       */
733      public void listAllEntries(final TransactionContext ctx,
734                                 final String entryType,
735                                 final Iterator<String> inputLines,
736                                 final BlockingQueue<DatabaseChangeRecord>
737                                          outputQueue) throws SQLException
738      {
739        throw new UnsupportedOperationException();
740      }
741    }