1 /*
2  * Copyright (C) 2013 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.content;
18 
19 import android.accounts.Account;
20 import android.annotation.NonNull;
21 import android.compat.annotation.UnsupportedAppUsage;
22 import android.os.Build;
23 import android.os.Bundle;
24 import android.os.Parcel;
25 import android.os.Parcelable;
26 
27 /**
28  * Convenience class to construct sync requests. See {@link android.content.SyncRequest.Builder}
29  * for an explanation of the various functions. The resulting object is passed through to the
30  * framework via {@link android.content.ContentResolver#requestSync(SyncRequest)}.
31  */
32 public class SyncRequest implements Parcelable {
33     private static final String TAG = "SyncRequest";
34     /** Account to pass to the sync adapter. Can be null. */
35     @UnsupportedAppUsage
36     private final Account mAccountToSync;
37     /** Authority string that corresponds to a ContentProvider. */
38     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
39     private final String mAuthority;
40     /** Bundle containing user info as well as sync settings. */
41     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
42     private final Bundle mExtras;
43     /** Don't allow this sync request on metered networks. */
44     private final boolean mDisallowMetered;
45     /**
46      * Amount of time before {@link #mSyncRunTimeSecs} from which the sync may optionally be
47      * started.
48      */
49     private final long mSyncFlexTimeSecs;
50     /**
51      * Specifies a point in the future at which the sync must have been scheduled to run.
52      */
53     @UnsupportedAppUsage
54     private final long mSyncRunTimeSecs;
55     /** Periodic versus one-off. */
56     @UnsupportedAppUsage
57     private final boolean mIsPeriodic;
58     /** Service versus provider. */
59     private final boolean mIsAuthority;
60     /** Sync should be run in lieu of other syncs. */
61     private final boolean mIsExpedited;
62     /** Sync sound be ran as an expedited job. */
63     private final boolean mIsScheduledAsExpeditedJob;
64 
65     /**
66      * {@hide}
67      * @return whether this sync is periodic or one-time. A Sync Request must be
68      *         either one of these or an InvalidStateException will be thrown in
69      *         Builder.build().
70      */
isPeriodic()71     public boolean isPeriodic() {
72         return mIsPeriodic;
73     }
74 
75     /**
76      * {@hide}
77      * @return whether this sync is expedited.
78      */
isExpedited()79     public boolean isExpedited() {
80         return mIsExpedited;
81     }
82 
83     /**
84      * {@hide}
85      * @return whether this sync is scheduled as an expedited job.
86      */
isScheduledAsExpeditedJob()87     public boolean isScheduledAsExpeditedJob() {
88         return mIsScheduledAsExpeditedJob;
89     }
90 
91     /**
92      * {@hide}
93      *
94      * @return account object for this sync.
95      * @throws IllegalArgumentException if this function is called for a request that targets a
96      * sync service.
97      */
getAccount()98     public Account getAccount() {
99         return mAccountToSync;
100     }
101 
102     /**
103      * {@hide}
104      *
105      * @return provider for this sync.
106      * @throws IllegalArgumentException if this function is called for a request that targets a
107      * sync service.
108      */
getProvider()109     public String getProvider() {
110         return mAuthority;
111     }
112 
113     /**
114      * {@hide}
115      * Retrieve bundle for this SyncRequest. Will not be null.
116      */
getBundle()117     public Bundle getBundle() {
118         return mExtras;
119     }
120 
121     /**
122      * {@hide}
123      * @return the earliest point in time that this sync can be scheduled.
124      */
getSyncFlexTime()125     public long getSyncFlexTime() {
126         return mSyncFlexTimeSecs;
127     }
128     /**
129      * {@hide}
130      * @return the last point in time at which this sync must scheduled.
131      */
getSyncRunTime()132     public long getSyncRunTime() {
133         return mSyncRunTimeSecs;
134     }
135 
136     public static final @android.annotation.NonNull Creator<SyncRequest> CREATOR = new Creator<SyncRequest>() {
137 
138         @Override
139         public SyncRequest createFromParcel(Parcel in) {
140             return new SyncRequest(in);
141         }
142 
143         @Override
144         public SyncRequest[] newArray(int size) {
145             return new SyncRequest[size];
146         }
147     };
148 
149     @Override
describeContents()150     public int describeContents() {
151         return 0;
152     }
153 
154     @Override
writeToParcel(Parcel parcel, int flags)155     public void writeToParcel(Parcel parcel, int flags) {
156         parcel.writeBundle(mExtras);
157         parcel.writeLong(mSyncFlexTimeSecs);
158         parcel.writeLong(mSyncRunTimeSecs);
159         parcel.writeInt((mIsPeriodic ? 1 : 0));
160         parcel.writeInt((mDisallowMetered ? 1 : 0));
161         parcel.writeInt((mIsAuthority ? 1 : 0));
162         parcel.writeInt((mIsExpedited? 1 : 0));
163         parcel.writeInt(mIsScheduledAsExpeditedJob ? 1 : 0);
164         parcel.writeParcelable(mAccountToSync, flags);
165         parcel.writeString(mAuthority);
166     }
167 
SyncRequest(Parcel in)168     private SyncRequest(Parcel in) {
169         mExtras = Bundle.setDefusable(in.readBundle(), true);
170         mSyncFlexTimeSecs = in.readLong();
171         mSyncRunTimeSecs = in.readLong();
172         mIsPeriodic = (in.readInt() != 0);
173         mDisallowMetered = (in.readInt() != 0);
174         mIsAuthority = (in.readInt() != 0);
175         mIsExpedited = (in.readInt() != 0);
176         mIsScheduledAsExpeditedJob = (in.readInt() != 0);
177         mAccountToSync = in.readParcelable(null, android.accounts.Account.class);
178         mAuthority = in.readString();
179     }
180 
181     /** {@hide} Protected ctor to instantiate anonymous SyncRequest. */
SyncRequest(SyncRequest.Builder b)182     protected SyncRequest(SyncRequest.Builder b) {
183         mSyncFlexTimeSecs = b.mSyncFlexTimeSecs;
184         mSyncRunTimeSecs = b.mSyncRunTimeSecs;
185         mAccountToSync = b.mAccount;
186         mAuthority = b.mAuthority;
187         mIsPeriodic = (b.mSyncType == Builder.SYNC_TYPE_PERIODIC);
188         mIsAuthority = (b.mSyncTarget == Builder.SYNC_TARGET_ADAPTER);
189         mIsExpedited = b.mExpedited;
190         mIsScheduledAsExpeditedJob = b.mScheduleAsExpeditedJob;
191         mExtras = new Bundle(b.mCustomExtras);
192         // For now we merge the sync config extras & the custom extras into one bundle.
193         // TODO: pass the configuration extras through separately.
194         mExtras.putAll(b.mSyncConfigExtras);
195         mDisallowMetered = b.mDisallowMetered;
196     }
197 
198     /**
199      * Builder class for a {@link SyncRequest}. As you build your SyncRequest this class will also
200      * perform validation.
201      */
202     public static class Builder {
203         /** Unknown sync type. */
204         private static final int SYNC_TYPE_UNKNOWN = 0;
205         /** Specify that this is a periodic sync. */
206         private static final int SYNC_TYPE_PERIODIC = 1;
207         /** Specify that this is a one-time sync. */
208         private static final int SYNC_TYPE_ONCE = 2;
209         /** Unknown sync target. */
210         private static final int SYNC_TARGET_UNKNOWN = 0;
211         /** Specify that this is a sync with a provider. */
212         private static final int SYNC_TARGET_ADAPTER = 2;
213         /**
214          * Earliest point of displacement into the future at which this sync can
215          * occur.
216          */
217         private long mSyncFlexTimeSecs;
218         /** Displacement into the future at which this sync must occur. */
219         private long mSyncRunTimeSecs;
220         /**
221          * Sync configuration information - custom user data explicitly provided by the developer.
222          * This data is handed over to the sync operation.
223          */
224         private Bundle mCustomExtras;
225         /**
226          * Sync system configuration -  used to store system sync configuration. Corresponds to
227          * ContentResolver.SYNC_EXTRAS_* flags.
228          * TODO: Use this instead of dumping into one bundle. Need to decide if these flags should
229          * discriminate between equivalent syncs.
230          */
231         private Bundle mSyncConfigExtras;
232         /** Whether or not this sync can occur on metered networks. Default false. */
233         private boolean mDisallowMetered;
234         /**
235          * Whether this builder is building a periodic sync, or a one-time sync.
236          */
237         private int mSyncType = SYNC_TYPE_UNKNOWN;
238         /** Whether this will go to a sync adapter. */
239         private int mSyncTarget = SYNC_TARGET_UNKNOWN;
240         /** Whether this is a user-activated sync. */
241         private boolean mIsManual;
242         /**
243          * Whether to retry this one-time sync if the sync fails. Not valid for
244          * periodic syncs. See {@link ContentResolver#SYNC_EXTRAS_DO_NOT_RETRY}.
245          */
246         private boolean mNoRetry;
247         /**
248          * Whether to respect back-off for this one-time sync. Not valid for
249          * periodic syncs. See
250          * {@link ContentResolver#SYNC_EXTRAS_IGNORE_BACKOFF};
251          */
252         private boolean mIgnoreBackoff;
253 
254         /** Ignore sync system settings and perform sync anyway. */
255         private boolean mIgnoreSettings;
256 
257         /** This sync will run in preference to other non-expedited syncs. */
258         private boolean mExpedited;
259 
260         /**
261          * The Account object that together with an Authority name define the SyncAdapter (if
262          * this sync is bound to a provider), otherwise null.
263          */
264         private Account mAccount;
265         /**
266          * The Authority name that together with an Account define the SyncAdapter (if
267          * this sync is bound to a provider), otherwise null.
268          */
269         private String mAuthority;
270         /**
271          * Whether the sync requires the phone to be plugged in.
272          */
273         private boolean mRequiresCharging;
274 
275         /**
276          * Whether the sync should be scheduled as an expedited job.
277          */
278         private boolean mScheduleAsExpeditedJob;
279 
Builder()280         public Builder() {
281         }
282 
283         /**
284          * Request that a sync occur immediately.
285          *
286          * Example
287          * <pre>
288          *     SyncRequest.Builder builder = (new SyncRequest.Builder()).syncOnce();
289          * </pre>
290          */
syncOnce()291         public Builder syncOnce() {
292             if (mSyncType != SYNC_TYPE_UNKNOWN) {
293                 throw new IllegalArgumentException("Sync type has already been defined.");
294             }
295             mSyncType = SYNC_TYPE_ONCE;
296             setupInterval(0, 0);
297             return this;
298         }
299 
300         /**
301          * Build a periodic sync. Either this or syncOnce() <b>must</b> be called for this builder.
302          * Syncs are identified by target {@link android.provider} and by the
303          * contents of the extras bundle.
304          * You cannot reuse the same builder for one-time syncs after having specified a periodic
305          * sync (by calling this function). If you do, an <code>IllegalArgumentException</code>
306          * will be thrown.
307          * <p>The bundle for a periodic sync can be queried by applications with the correct
308          * permissions using
309          * {@link ContentResolver#getPeriodicSyncs(Account account, String provider)}, so no
310          * sensitive data should be transferred here.
311          *
312          * Example usage.
313          *
314          * <pre>
315          *     Request a periodic sync every 5 hours with 20 minutes of flex.
316          *     SyncRequest.Builder builder =
317          *         (new SyncRequest.Builder()).syncPeriodic(5 * HOUR_IN_SECS, 20 * MIN_IN_SECS);
318          *
319          *     Schedule a periodic sync every hour at any point in time during that hour.
320          *     SyncRequest.Builder builder =
321          *         (new SyncRequest.Builder()).syncPeriodic(1 * HOUR_IN_SECS, 1 * HOUR_IN_SECS);
322          * </pre>
323          *
324          * N.B.: Periodic syncs are not allowed to have any of
325          * {@link ContentResolver#SYNC_EXTRAS_DO_NOT_RETRY},
326          * {@link ContentResolver#SYNC_EXTRAS_IGNORE_BACKOFF},
327          * {@link ContentResolver#SYNC_EXTRAS_IGNORE_SETTINGS},
328          * {@link ContentResolver#SYNC_EXTRAS_INITIALIZE},
329          * {@link ContentResolver#SYNC_EXTRAS_FORCE},
330          * {@link ContentResolver#SYNC_EXTRAS_EXPEDITED},
331          * {@link ContentResolver#SYNC_EXTRAS_MANUAL},
332          * {@link ContentResolver#SYNC_EXTRAS_SCHEDULE_AS_EXPEDITED_JOB}
333          * set to true. If any are supplied then an <code>IllegalArgumentException</code> will
334          * be thrown.
335          *
336          * @param pollFrequency the amount of time in seconds that you wish
337          *            to elapse between periodic syncs. A minimum period of 1 hour is enforced.
338          * @param beforeSeconds the amount of flex time in seconds before
339          *            {@code pollFrequency} that you permit for the sync to take
340          *            place. Must be less than {@code pollFrequency} and greater than
341          *            MAX(5% of {@code pollFrequency}, 5 minutes)
342          */
syncPeriodic(long pollFrequency, long beforeSeconds)343         public Builder syncPeriodic(long pollFrequency, long beforeSeconds) {
344             if (mSyncType != SYNC_TYPE_UNKNOWN) {
345                 throw new IllegalArgumentException("Sync type has already been defined.");
346             }
347             mSyncType = SYNC_TYPE_PERIODIC;
348             setupInterval(pollFrequency, beforeSeconds);
349             return this;
350         }
351 
setupInterval(long at, long before)352         private void setupInterval(long at, long before) {
353             if (before > at) {
354                 throw new IllegalArgumentException("Specified run time for the sync must be" +
355                     " after the specified flex time.");
356             }
357             mSyncRunTimeSecs = at;
358             mSyncFlexTimeSecs = before;
359         }
360 
361         /**
362          * Will throw an <code>IllegalArgumentException</code> if called and
363          * {@link #setIgnoreSettings(boolean ignoreSettings)} has already been called.
364          * @param disallow true to allow this transfer on metered networks. Default false.
365          *
366          */
setDisallowMetered(boolean disallow)367         public Builder setDisallowMetered(boolean disallow) {
368             if (mIgnoreSettings && disallow) {
369                 throw new IllegalArgumentException("setDisallowMetered(true) after having"
370                         + " specified that settings are ignored.");
371             }
372             mDisallowMetered = disallow;
373             return this;
374         }
375 
376         /**
377          * Specify whether the sync requires the phone to be plugged in.
378          * @param requiresCharging true if sync requires the phone to be plugged in. Default false.
379          */
setRequiresCharging(boolean requiresCharging)380         public Builder setRequiresCharging(boolean requiresCharging) {
381             mRequiresCharging = requiresCharging;
382             return this;
383         }
384 
385         /**
386          * Specify an authority and account for this transfer.
387          *
388          * @param authority A String identifying the content provider to be synced.
389          * @param account Account to sync. Can be null unless this is a periodic
390          *            sync, for which verification by the ContentResolver will
391          *            fail. If a sync is performed without an account, the
392          */
setSyncAdapter(Account account, String authority)393         public Builder setSyncAdapter(Account account, String authority) {
394             if (mSyncTarget != SYNC_TARGET_UNKNOWN) {
395                 throw new IllegalArgumentException("Sync target has already been defined.");
396             }
397             if (authority != null && authority.length() == 0) {
398                 throw new IllegalArgumentException("Authority must be non-empty");
399             }
400             mSyncTarget = SYNC_TARGET_ADAPTER;
401             mAccount = account;
402             mAuthority = authority;
403             return this;
404         }
405 
406         /**
407          * Developer-provided extras handed back when sync actually occurs. This bundle is copied
408          * into the SyncRequest returned by {@link #build()}.
409          *
410          * Example:
411          * <pre>
412          *   String[] syncItems = {"dog", "cat", "frog", "child"};
413          *   SyncRequest.Builder builder =
414          *     new SyncRequest.Builder()
415          *       .setSyncAdapter(dummyAccount, dummyProvider)
416          *       .syncOnce();
417          *
418          *   for (String syncData : syncItems) {
419          *     Bundle extras = new Bundle();
420          *     extras.setString("data", syncData);
421          *     builder.setExtras(extras);
422          *     ContentResolver.sync(builder.build()); // Each sync() request creates a unique sync.
423          *   }
424          * </pre>
425          * Only values of the following types may be used in the extras bundle:
426          * <ul>
427          * <li>Integer</li>
428          * <li>Long</li>
429          * <li>Boolean</li>
430          * <li>Float</li>
431          * <li>Double</li>
432          * <li>String</li>
433          * <li>Account</li>
434          * <li>null</li>
435          * </ul>
436          * If any data is present in the bundle not of this type, build() will
437          * throw a runtime exception.
438          *
439          * @param bundle extras bundle to set.
440          */
setExtras(Bundle bundle)441         public Builder setExtras(Bundle bundle) {
442             mCustomExtras = bundle;
443             return this;
444         }
445 
446         /**
447          * Convenience function for setting {@link ContentResolver#SYNC_EXTRAS_DO_NOT_RETRY}.
448          *
449          * A one-off sync operation that fails will be retried with exponential back-off unless
450          * this is set to false. Not valid for periodic sync and will throw an
451          * <code>IllegalArgumentException</code> in build().
452          *
453          * @param noRetry true to not retry a failed sync. Default false.
454          */
setNoRetry(boolean noRetry)455         public Builder setNoRetry(boolean noRetry) {
456             mNoRetry = noRetry;
457             return this;
458         }
459 
460         /**
461          * Convenience function for setting {@link ContentResolver#SYNC_EXTRAS_IGNORE_SETTINGS}.
462          *
463          * Not valid for periodic sync and will throw an <code>IllegalArgumentException</code> in
464          * {@link #build()}.
465          * <p>Throws <code>IllegalArgumentException</code> if called and
466          * {@link #setDisallowMetered(boolean)} has been set.
467          *
468          *
469          * @param ignoreSettings true to ignore the sync automatically settings. Default false.
470          */
setIgnoreSettings(boolean ignoreSettings)471         public Builder setIgnoreSettings(boolean ignoreSettings) {
472             if (mDisallowMetered && ignoreSettings) {
473                 throw new IllegalArgumentException("setIgnoreSettings(true) after having specified"
474                         + " sync settings with this builder.");
475             }
476             mIgnoreSettings = ignoreSettings;
477             return this;
478         }
479 
480         /**
481          * Convenience function for setting {@link ContentResolver#SYNC_EXTRAS_IGNORE_BACKOFF}.
482          *
483          * Ignoring back-off will force the sync scheduling process to ignore any back-off that was
484          * the result of a failed sync, as well as to invalidate any {@link SyncResult#delayUntil}
485          * value that may have been set by the adapter. Successive failures will not honor this
486          * flag. Not valid for periodic sync and will throw an <code>IllegalArgumentException</code>
487          * in {@link #build()}.
488          *
489          * @param ignoreBackoff ignore back off settings. Default false.
490          */
setIgnoreBackoff(boolean ignoreBackoff)491         public Builder setIgnoreBackoff(boolean ignoreBackoff) {
492             mIgnoreBackoff = ignoreBackoff;
493             return this;
494         }
495 
496         /**
497          * Convenience function for setting {@link ContentResolver#SYNC_EXTRAS_MANUAL}.
498          *
499          * Not valid for periodic sync and will throw an <code>IllegalArgumentException</code> in
500          * {@link #build()}.
501          *
502          * @param isManual User-initiated sync or not. Default false.
503          */
setManual(boolean isManual)504         public Builder setManual(boolean isManual) {
505             mIsManual = isManual;
506             return this;
507         }
508 
509         /**
510          * An expedited sync runs immediately and can preempt other non-expedited running syncs.
511          *
512          * Not valid for periodic sync and will throw an <code>IllegalArgumentException</code> in
513          * {@link #build()}.
514          *
515          * @param expedited whether to run expedited. Default false.
516          */
setExpedited(boolean expedited)517         public Builder setExpedited(boolean expedited) {
518             mExpedited = expedited;
519             return this;
520         }
521 
522         /**
523          * Convenience function for setting
524          * {@link ContentResolver#SYNC_EXTRAS_SCHEDULE_AS_EXPEDITED_JOB}.
525          *
526          * <p> Not to be confused with {@link ContentResolver#SYNC_EXTRAS_EXPEDITED}.
527          *
528          * <p> Not valid for periodic syncs, expedited syncs, and syncs that require charging - an
529          * <code>IllegalArgumentException</code> will be thrown in {@link #build()}.
530          *
531          * @param scheduleAsExpeditedJob whether to schedule as an expedited job. Default false.
532          */
setScheduleAsExpeditedJob(boolean scheduleAsExpeditedJob)533         public @NonNull Builder setScheduleAsExpeditedJob(boolean scheduleAsExpeditedJob) {
534             mScheduleAsExpeditedJob = scheduleAsExpeditedJob;
535             return this;
536         }
537 
538         /**
539          * Performs validation over the request and throws the runtime exception
540          * <code>IllegalArgumentException</code> if this validation fails.
541          *
542          * @return a SyncRequest with the information contained within this
543          *         builder.
544          */
build()545         public SyncRequest build() {
546             // Combine builder extra flags into the config bundle.
547             mSyncConfigExtras = new Bundle();
548             if (mIgnoreBackoff) {
549                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_BACKOFF, true);
550             }
551             if (mDisallowMetered) {
552                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_DISALLOW_METERED, true);
553             }
554             if (mRequiresCharging) {
555                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_REQUIRE_CHARGING, true);
556             }
557             if (mIgnoreSettings) {
558                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_SETTINGS, true);
559             }
560             if (mNoRetry) {
561                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_DO_NOT_RETRY, true);
562             }
563             if (mExpedited) {
564                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_EXPEDITED, true);
565             }
566             if (mScheduleAsExpeditedJob) {
567                 mSyncConfigExtras.putBoolean(
568                         ContentResolver.SYNC_EXTRAS_SCHEDULE_AS_EXPEDITED_JOB, true);
569             }
570             if (mIsManual) {
571                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_BACKOFF, true);
572                 mSyncConfigExtras.putBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_SETTINGS, true);
573             }
574 
575             if (mCustomExtras == null) {
576                 mCustomExtras = new Bundle();
577             }
578             // Validate the extras bundles
579             ContentResolver.validateSyncExtrasBundle(mCustomExtras);
580             // If this is a periodic sync ensure than invalid extras were not set.
581             if (mSyncType == SYNC_TYPE_PERIODIC) {
582                 if (ContentResolver.invalidPeriodicExtras(mCustomExtras) ||
583                         ContentResolver.invalidPeriodicExtras(mSyncConfigExtras)) {
584                     throw new IllegalArgumentException("Illegal extras were set");
585                 }
586             }
587             // If this sync is scheduled as an EJ, ensure that invalid extras were not set.
588             if (mCustomExtras.getBoolean(ContentResolver.SYNC_EXTRAS_SCHEDULE_AS_EXPEDITED_JOB)
589                     || mScheduleAsExpeditedJob) {
590                 if (ContentResolver.hasInvalidScheduleAsEjExtras(mCustomExtras)
591                         || ContentResolver.hasInvalidScheduleAsEjExtras(mSyncConfigExtras)) {
592                     throw new IllegalArgumentException("Illegal extras were set");
593                 }
594             }
595             // Ensure that a target for the sync has been set.
596             if (mSyncTarget == SYNC_TARGET_UNKNOWN) {
597                 throw new IllegalArgumentException("Must specify an adapter with" +
598                         " setSyncAdapter(Account, String");
599             }
600             return new SyncRequest(this);
601         }
602     }
603 }
604