Mention the latest fix.
[rsync/rsync.git] / options.c
CommitLineData
7a24c346 1/* -*- c-file-style: "linux" -*-
dfa32483 2 *
dafe63ca
MP
3 * Copyright (C) 1998-2001 by Andrew Tridgell <tridge@samba.org>
4 * Copyright (C) 2000, 2001, 2002 by Martin Pool <mbp@samba.org>
dfa32483 5 *
dafe63ca
MP
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
dfa32483 10 *
dafe63ca
MP
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
dfa32483 15 *
dafe63ca
MP
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19 */
7a6421fa 20
7a6421fa 21#include "rsync.h"
2855f61f 22#include "popt.h"
7a6421fa 23
7bc90b30 24extern int module_id;
7be73df4 25extern int sanitize_paths;
7842418b
WD
26extern struct filter_list_struct filter_list;
27extern struct filter_list_struct server_filter_list;
8645af1d 28
7a6421fa 29int make_backups = 0;
1bfbf40b
MP
30
31/**
dfa32483 32 * If 1, send the whole file as literal data rather than trying to
dafe63ca 33 * create an incremental diff.
1bfbf40b 34 *
dfa32483 35 * If -1, then look at whether we're local or remote and go by that.
dafe63ca
MP
36 *
37 * @sa disable_deltas_p()
1bfbf40b 38 **/
dfa32483 39int whole_file = -1;
1bfbf40b 40
a015788d 41int append_mode = 0;
716e73d4 42int keep_dirlinks = 0;
7a6421fa
AT
43int copy_links = 0;
44int preserve_links = 0;
45int preserve_hard_links = 0;
46int preserve_perms = 0;
47int preserve_devices = 0;
48int preserve_uid = 0;
49int preserve_gid = 0;
50int preserve_times = 0;
20fb7b91 51int omit_dir_times = 0;
7a6421fa
AT
52int update_only = 0;
53int cvs_exclude = 0;
a5c11139 54int dry_run = 0;
11e758a4 55int do_xfers = 1;
a5c11139
WD
56int ignore_times = 0;
57int delete_mode = 0;
a51b3168 58int delete_during = 0;
51d48398
WD
59int delete_before = 0;
60int delete_after = 0;
a5c11139 61int delete_excluded = 0;
07c6ae7d 62int remove_sent_files = 0;
a5c11139 63int one_file_system = 0;
4f3e9a0f 64int protocol_version = PROTOCOL_VERSION;
a5c11139
WD
65int sparse_files = 0;
66int do_compression = 0;
67int am_root = 0;
7f2a1f65
WD
68int am_server = 0;
69int am_sender = 0;
70int am_generator = 0;
71int am_starting_up = 1;
a5c11139 72int orig_umask = 0;
ea5164d1
WD
73int relative_paths = -1;
74int implied_dirs = 1;
7a6421fa
AT
75int numeric_ids = 0;
76int force_delete = 0;
77int io_timeout = 0;
9ac756c6 78int allowed_lull = 0;
ea5164d1
WD
79char *files_from = NULL;
80int filesfrom_fd = -1;
305666bf 81char *filesfrom_host = NULL;
ea5164d1 82int eol_nulls = 0;
7a6421fa 83int recurse = 0;
b6164938 84int xfer_dirs = -1;
1312d9fc
WD
85int am_daemon = 0;
86int daemon_over_rsh = 0;
a5c11139
WD
87int do_stats = 0;
88int do_progress = 0;
89int keep_partial = 0;
90int safe_symlinks = 0;
91int copy_unsafe_links = 0;
92int size_only = 0;
9fb08441 93int daemon_bwlimit = 0;
a5c11139 94int bwlimit = 0;
c4ed1487 95int fuzzy_basis = 0;
3c74c3a3 96size_t bwlimit_writemax = 0;
a5c11139
WD
97int only_existing = 0;
98int opt_ignore_existing = 0;
07c6ae7d 99int need_messages_from_generator = 0;
a5c11139 100int max_delete = 0;
7d5acf1d 101OFF_T max_size = 0;
a5c11139
WD
102int ignore_errors = 0;
103int modify_window = 0;
104int blocking_io = -1;
2289bf64 105int checksum_seed = 0;
a3221d2a 106int inplace = 0;
f06e7082 107int delay_updates = 0;
a06b419d 108long block_size = 0; /* "long" because popt can't set an int32. */
7a6421fa 109
b35d0d8e 110
13e29995 111/** Network address family. **/
4f5b0756 112#ifdef INET6
13e29995 113int default_af_hint = 0; /* Any protocol */
6ab6d4bf 114#else
13e29995 115int default_af_hint = AF_INET; /* Must use IPv4 */
6ab6d4bf 116#endif
06963d0f 117
13e29995
MP
118/** Do not go into the background when run as --daemon. Good
119 * for debugging and required for running as a service on W32,
120 * or under Unix process-monitors. **/
1f35babc
WD
121int no_detach
122#if defined _WIN32 || defined __WIN32__
123 = 1;
124#else
125 = 0;
126#endif
13e29995 127
088aac85
DD
128int write_batch = 0;
129int read_batch = 0;
d175d7e1
WD
130int backup_dir_len = 0;
131int backup_suffix_len;
e0391f81 132unsigned int backup_dir_remainder;
6902ed17 133
d175d7e1 134char *backup_suffix = NULL;
7a6421fa 135char *tmpdir = NULL;
a7260c40 136char *partial_dir = NULL;
e012f858 137char *basis_dir[MAX_BASIS_DIRS+1];
30e8c8e1 138char *config_file = NULL;
7a6421fa 139char *shell_cmd = NULL;
b6062654 140char *log_format = NULL;
65575e96 141char *password_file = NULL;
41bd28fe 142char *rsync_path = RSYNC_PATH;
66203a98 143char *backup_dir = NULL;
e0391f81 144char backup_dir_buf[MAXPATHLEN];
0c56b1ad 145int rsync_port = 0;
dfd7d541 146int compare_dest = 0;
1de3e99b 147int copy_dest = 0;
59c95e42 148int link_dest = 0;
ce0b384f 149int basis_dir_cnt = 0;
f9a9f547 150char *dest_option = NULL;
7a6421fa
AT
151
152int verbose = 0;
b86f0cef 153int quiet = 0;
e7651884 154int log_before_transfer = 0;
684d7582 155int log_format_has_i = 0;
5c5e1598 156int log_format_has_o_or_i = 0;
7a6421fa 157int always_checksum = 0;
f7632fc6 158int list_only = 0;
7a6421fa 159
9b3318b0
WD
160#define MAX_BATCH_NAME_LEN 256 /* Must be less than MAXPATHLEN-13 */
161char *batch_name = NULL;
6902ed17 162
e1add893 163static int daemon_opt; /* sets am_daemon after option error-reporting */
aa4d3b4c 164static int F_option_cnt = 0;
5b56cc19 165static int modify_window_set;
624d6be2 166static int itemize_changes = 0;
702cd15c 167static int refused_delete, refused_archive_part;
3296f91b 168static int refused_partial, refused_progress, refused_delete_before;
a015788d 169static int refused_inplace;
7d5acf1d 170static char *max_size_arg;
3671987f 171static char partialdir_for_delayupdate[] = ".~tmp~";
5b56cc19 172
06963d0f 173/** Local address to bind. As a character string because it's
fdf57ede 174 * interpreted by the IPv6 layer: should be a numeric IP4 or IP6
06963d0f
MP
175 * address, or a hostname. **/
176char *bind_address;
5c9730a4 177
7a24c346 178
27a12348 179static void print_rsync_version(enum logcode f)
7a24c346 180{
dfa32483 181 char const *got_socketpair = "no ";
a3221d2a 182 char const *have_inplace = "no ";
dfa32483
WD
183 char const *hardlinks = "no ";
184 char const *links = "no ";
a358449a 185 char const *ipv6 = "no ";
736a6a29 186 STRUCT_STAT *dumstat;
0c80cd8e 187
4f5b0756 188#ifdef HAVE_SOCKETPAIR
dfa32483 189 got_socketpair = "";
0c80cd8e 190#endif
2855f61f 191
4f5b0756 192#ifdef HAVE_FTRUNCATE
a3221d2a
WD
193 have_inplace = "";
194#endif
195
4f5b0756 196#ifdef SUPPORT_HARD_LINKS
dfa32483 197 hardlinks = "";
2855f61f
MP
198#endif
199
4f5b0756 200#ifdef SUPPORT_LINKS
dfa32483 201 links = "";
2855f61f
MP
202#endif
203
4f5b0756 204#ifdef INET6
a358449a 205 ipv6 = "";
dfa32483 206#endif
a358449a 207
dfa32483
WD
208 rprintf(f, "%s version %s protocol version %d\n",
209 RSYNC_NAME, RSYNC_VERSION, PROTOCOL_VERSION);
210 rprintf(f,
72a90c75 211 "Copyright (C) 1996-2005 by Andrew Tridgell and others\n");
3b4b1984 212 rprintf(f, "<http://rsync.samba.org/>\n");
dfa32483
WD
213 rprintf(f, "Capabilities: %d-bit files, %ssocketpairs, "
214 "%shard links, %ssymlinks, batchfiles, \n",
a5c11139 215 (int) (sizeof (OFF_T) * 8),
dfa32483 216 got_socketpair, hardlinks, links);
2855f61f 217
736a6a29
MP
218 /* Note that this field may not have type ino_t. It depends
219 * on the complicated interaction between largefile feature
220 * macros. */
a3221d2a
WD
221 rprintf(f, " %sinplace, %sIPv6, %d-bit system inums, %d-bit internal inums\n",
222 have_inplace, ipv6,
a5c11139 223 (int) (sizeof dumstat->st_ino * 8),
1490812a 224 (int) (sizeof (int64) * 8));
fc0302cf
MP
225#ifdef MAINTAINER_MODE
226 rprintf(f, " panic action: \"%s\"\n",
227 get_panic_action());
228#endif
736a6a29 229
031fa9ad
WD
230#if SIZEOF_INT64 < 8
231 rprintf(f, "WARNING: no 64-bit integers on this platform!\n");
7a24c346 232#endif
c561edaa
WD
233 if (sizeof (int64) != SIZEOF_INT64) {
234 rprintf(f,
235 "WARNING: size mismatch in SIZEOF_INT64 define (%d != %d)\n",
236 (int) SIZEOF_INT64, (int) sizeof (int64));
237 }
7b329a2d
MP
238
239 rprintf(f,
240"\n"
241"rsync comes with ABSOLUTELY NO WARRANTY. This is free software, and you\n"
242"are welcome to redistribute it under certain conditions. See the GNU\n"
243"General Public Licence for details.\n"
244 );
7a24c346
MP
245}
246
247
0f3203c3 248void usage(enum logcode F)
7a6421fa 249{
2855f61f 250 print_rsync_version(F);
704f908e 251
3ff984d7 252 rprintf(F,"\nrsync is a file transfer program capable of efficient remote update\nvia a fast differencing algorithm.\n\n");
704f908e 253
9ef53907 254 rprintf(F,"Usage: rsync [OPTION]... SRC [SRC]... [USER@]HOST:DEST\n");
d0e94abb 255 rprintf(F," or rsync [OPTION]... [USER@]HOST:SRC [DEST]\n");
9ef53907 256 rprintf(F," or rsync [OPTION]... SRC [SRC]... DEST\n");
14d43f1f 257 rprintf(F," or rsync [OPTION]... [USER@]HOST::SRC [DEST]\n");
9ef53907 258 rprintf(F," or rsync [OPTION]... SRC [SRC]... [USER@]HOST::DEST\n");
14d43f1f 259 rprintf(F," or rsync [OPTION]... rsync://[USER@]HOST[:PORT]/SRC [DEST]\n");
eaa4c150 260 rprintf(F," or rsync [OPTION]... SRC [SRC]... rsync://[USER@]HOST[:PORT]/DEST\n");
9ef53907
DD
261 rprintf(F,"SRC on single-colon remote HOST will be expanded by remote shell\n");
262 rprintf(F,"SRC on server remote HOST may contain shell wildcards or multiple\n");
263 rprintf(F," sources separated by space as long as they have same top-level\n");
704f908e
AT
264 rprintf(F,"\nOptions\n");
265 rprintf(F," -v, --verbose increase verbosity\n");
b3708acf
WD
266 rprintf(F," -q, --quiet suppress non-error messages\n");
267 rprintf(F," -c, --checksum skip based on checksum, not mod-time & size\n");
268 rprintf(F," -a, --archive archive mode; same as -rlptgoD (no -H)\n");
b6164938 269 rprintf(F," --no-OPTION turn of an implied OPTION (e.g. --no-D)\n");
704f908e
AT
270 rprintf(F," -r, --recursive recurse into directories\n");
271 rprintf(F," -R, --relative use relative path names\n");
11bfaf63 272 rprintf(F," --no-implied-dirs don't send implied dirs with --relative\n");
6839140e 273 rprintf(F," -b, --backup make backups (see --suffix & --backup-dir)\n");
b3708acf
WD
274 rprintf(F," --backup-dir=DIR make backups into hierarchy based in DIR\n");
275 rprintf(F," --suffix=SUFFIX set backup suffix (default %s w/o --backup-dir)\n",BACKUP_SUFFIX);
276 rprintf(F," -u, --update skip files that are newer on the receiver\n");
4ce838e1 277 rprintf(F," --inplace update destination files in-place (SEE MAN PAGE)\n");
a015788d 278 rprintf(F," --append append data onto shorter files\n");
65e4cda0 279 rprintf(F," -d, --dirs transfer directories without recursing\n");
13e29995 280 rprintf(F," -l, --links copy symlinks as symlinks\n");
b3708acf
WD
281 rprintf(F," -L, --copy-links transform symlink into referent file/dir\n");
282 rprintf(F," --copy-unsafe-links only \"unsafe\" symlinks are transformed\n");
283 rprintf(F," --safe-links ignore symlinks that point outside the source tree\n");
704f908e 284 rprintf(F," -H, --hard-links preserve hard links\n");
65e4cda0 285 rprintf(F," -K, --keep-dirlinks treat symlinked dir on receiver as dir\n");
704f908e
AT
286 rprintf(F," -p, --perms preserve permissions\n");
287 rprintf(F," -o, --owner preserve owner (root only)\n");
288 rprintf(F," -g, --group preserve group\n");
289 rprintf(F," -D, --devices preserve devices (root only)\n");
dfa32483 290 rprintf(F," -t, --times preserve times\n");
20fb7b91 291 rprintf(F," -O, --omit-dir-times omit directories when preserving times\n");
704f908e
AT
292 rprintf(F," -S, --sparse handle sparse files efficiently\n");
293 rprintf(F," -n, --dry-run show what would have been transferred\n");
98bf61c8 294 rprintf(F," -W, --whole-file copy files whole (without rsync algorithm)\n");
704f908e 295 rprintf(F," -x, --one-file-system don't cross filesystem boundaries\n");
9cd339eb 296 rprintf(F," -B, --block-size=SIZE force a fixed checksum block-size\n");
b3708acf 297 rprintf(F," -e, --rsh=COMMAND specify the remote shell to use\n");
05ee4866 298 rprintf(F," --rsync-path=PROGRAM specify the rsync to run on the remote machine\n");
b3708acf 299 rprintf(F," --existing only update files that already exist on receiver\n");
6839140e 300 rprintf(F," --ignore-existing ignore files that already exist on receiving side\n");
07c6ae7d 301 rprintf(F," --remove-sent-files sent files/symlinks are removed from sending side\n");
3359acb8 302 rprintf(F," --del an alias for --delete-during\n");
704f908e 303 rprintf(F," --delete delete files that don't exist on the sending side\n");
c6eb7fad 304 rprintf(F," --delete-before receiver deletes before transfer (default)\n");
3359acb8
WD
305 rprintf(F," --delete-during receiver deletes during transfer, not before\n");
306 rprintf(F," --delete-after receiver deletes after transfer, not before\n");
51d48398 307 rprintf(F," --delete-excluded also delete excluded files on the receiving side\n");
db2b5cb7 308 rprintf(F," --ignore-errors delete even if there are I/O errors\n");
51d48398 309 rprintf(F," --force force deletion of directories even if not empty\n");
0b73ca12 310 rprintf(F," --max-delete=NUM don't delete more than NUM files\n");
7d5acf1d 311 rprintf(F," --max-size=SIZE don't transfer any file larger than SIZE\n");
c95da96a 312 rprintf(F," --partial keep partially transferred files\n");
a7260c40 313 rprintf(F," --partial-dir=DIR put a partially transferred file into DIR\n");
b3708acf 314 rprintf(F," --delay-updates put all updated files into place at transfer's end\n");
704f908e 315 rprintf(F," --numeric-ids don't map uid/gid values by user/group name\n");
db2b5cb7 316 rprintf(F," --timeout=TIME set I/O timeout in seconds\n");
b3708acf
WD
317 rprintf(F," -I, --ignore-times don't skip files that match in size and mod-time\n");
318 rprintf(F," --size-only skip files that match in size\n");
319 rprintf(F," --modify-window=NUM compare mod-times with reduced accuracy\n");
4db88e5b 320 rprintf(F," -T, --temp-dir=DIR create temporary files in directory DIR\n");
c4ed1487 321 rprintf(F," -y, --fuzzy find similar file for basis if no dest file\n");
375a4556 322 rprintf(F," --compare-dest=DIR also compare destination files relative to DIR\n");
1de3e99b 323 rprintf(F," --copy-dest=DIR ... and include copies of unchanged files\n");
e012f858 324 rprintf(F," --link-dest=DIR hardlink to files in DIR when unchanged\n");
f924946e 325 rprintf(F," -z, --compress compress file data during the transfer\n");
b3708acf 326 rprintf(F," -C, --cvs-exclude auto-ignore files the same way CVS does\n");
aa4d3b4c 327 rprintf(F," -f, --filter=RULE add a file-filtering RULE\n");
9c71f56a 328 rprintf(F," -F same as --filter='dir-merge /.rsync-filter'\n");
aa4d3b4c 329 rprintf(F," repeated: --filter='- .rsync-filter'\n");
2acf81eb 330 rprintf(F," --exclude=PATTERN exclude files matching PATTERN\n");
b3708acf 331 rprintf(F," --exclude-from=FILE read exclude patterns from FILE\n");
2acf81eb 332 rprintf(F," --include=PATTERN don't exclude files matching PATTERN\n");
b3708acf
WD
333 rprintf(F," --include-from=FILE read include patterns from FILE\n");
334 rprintf(F," --files-from=FILE read list of source-file names from FILE\n");
72316028 335 rprintf(F," -0, --from0 all *-from/filter files are delimited by 0s\n");
b4ef0bca 336 rprintf(F," --address=ADDRESS bind address for outgoing socket to daemon\n");
b4713295 337 rprintf(F," --port=PORT specify double-colon alternate port number\n");
db2b5cb7 338 rprintf(F," --blocking-io use blocking I/O for the remote shell\n");
b3708acf 339 rprintf(F," --stats give some file-transfer stats\n");
dfa32483 340 rprintf(F," --progress show progress during transfer\n");
b3708acf 341 rprintf(F," -P same as --partial --progress\n");
b78296cb 342 rprintf(F," -i, --itemize-changes output a change-summary for all updates\n");
4a34c6f1 343 rprintf(F," --log-format=FORMAT output filenames using the specified format\n");
b3708acf 344 rprintf(F," --password-file=FILE read password from FILE\n");
65e4cda0 345 rprintf(F," --list-only list the files instead of copying them\n");
b3708acf
WD
346 rprintf(F," --bwlimit=KBPS limit I/O bandwidth; KBytes per second\n");
347 rprintf(F," --write-batch=FILE write a batched update to FILE\n");
11e758a4 348 rprintf(F," --only-write-batch=FILE like --write-batch but w/o updating destination\n");
b3708acf 349 rprintf(F," --read-batch=FILE read a batched update from FILE\n");
da9f5926 350 rprintf(F," --protocol=NUM force an older protocol version to be used\n");
4f5b0756 351#ifdef INET6
4db88e5b
WD
352 rprintf(F," -4, --ipv4 prefer IPv4\n");
353 rprintf(F," -6, --ipv6 prefer IPv6\n");
06963d0f 354#endif
4a34c6f1 355 rprintf(F," --version print version number\n");
3dd22903 356 rprintf(F," -h, --help show this help screen\n");
7a6421fa 357
3ac7f5d4
WD
358 rprintf(F,"\nUse \"rsync --daemon --help\" to see the daemon-mode command-line options.\n");
359 rprintf(F,"Please see the rsync(1) and rsyncd.conf(5) man pages for full documentation.\n");
2855f61f 360 rprintf(F,"See http://rsync.samba.org/ for updates, bug reports, and answers\n");
7a6421fa
AT
361}
362
3ac7f5d4 363enum {OPT_VERSION = 1000, OPT_DAEMON, OPT_SENDER, OPT_EXCLUDE, OPT_EXCLUDE_FROM,
1de3e99b 364 OPT_FILTER, OPT_COMPARE_DEST, OPT_COPY_DEST, OPT_LINK_DEST,
1f3d6cdd 365 OPT_INCLUDE, OPT_INCLUDE_FROM, OPT_MODIFY_WINDOW,
9ac756c6 366 OPT_READ_BATCH, OPT_WRITE_BATCH, OPT_ONLY_WRITE_BATCH, OPT_MAX_SIZE,
c67d1386 367 OPT_REFUSED_BASE = 9000};
7a6421fa 368
2855f61f
MP
369static struct poptOption long_options[] = {
370 /* longName, shortName, argInfo, argPtr, value, descrip, argDesc */
e86e2fa1 371 {"help", 'h', POPT_ARG_NONE, 0, 'h', 0, 0 },
8db7cc2c 372 {"version", 0, POPT_ARG_NONE, 0, OPT_VERSION, 0, 0},
e86e2fa1 373 {"verbose", 'v', POPT_ARG_NONE, 0, 'v', 0, 0 },
b6164938 374 {"no-verbose", 0, POPT_ARG_VAL, &verbose, 0, 0, 0 },
4afcb709 375 {"no-v", 0, POPT_ARG_VAL, &verbose, 0, 0, 0 },
e86e2fa1
WD
376 {"quiet", 'q', POPT_ARG_NONE, 0, 'q', 0, 0 },
377 {"stats", 0, POPT_ARG_NONE, &do_stats, 0, 0, 0 },
378 {"dry-run", 'n', POPT_ARG_NONE, &dry_run, 0, 0, 0 },
b6164938
WD
379 {"archive", 'a', POPT_ARG_NONE, 0, 'a', 0, 0 },
380 {"recursive", 'r', POPT_ARG_VAL, &recurse, 2, 0, 0 },
381 {"no-recursive", 0, POPT_ARG_VAL, &recurse, 0, 0, 0 },
382 {"no-r", 0, POPT_ARG_VAL, &recurse, 0, 0, 0 },
e86e2fa1 383 {"dirs", 'd', POPT_ARG_VAL, &xfer_dirs, 2, 0, 0 },
b6164938
WD
384 {"no-dirs", 0, POPT_ARG_VAL, &xfer_dirs, 0, 0, 0 },
385 {"no-d", 0, POPT_ARG_VAL, &xfer_dirs, 0, 0, 0 },
386 {"perms", 'p', POPT_ARG_VAL, &preserve_perms, 1, 0, 0 },
387 {"no-perms", 0, POPT_ARG_VAL, &preserve_perms, 0, 0, 0 },
388 {"no-p", 0, POPT_ARG_VAL, &preserve_perms, 0, 0, 0 },
389 {"times", 't', POPT_ARG_VAL, &preserve_times, 1, 0, 0 },
b6164938
WD
390 {"no-times", 0, POPT_ARG_VAL, &preserve_times, 0, 0, 0 },
391 {"no-t", 0, POPT_ARG_VAL, &preserve_times, 0, 0, 0 },
38b9170c
WD
392 {"omit-dir-times", 'O', POPT_ARG_VAL, &omit_dir_times, 2, 0, 0 },
393 {"modify-window", 0, POPT_ARG_INT, &modify_window, OPT_MODIFY_WINDOW, 0, 0 },
b6164938
WD
394 {"owner", 'o', POPT_ARG_VAL, &preserve_uid, 1, 0, 0 },
395 {"no-owner", 0, POPT_ARG_VAL, &preserve_uid, 0, 0, 0 },
396 {"no-o", 0, POPT_ARG_VAL, &preserve_uid, 0, 0, 0 },
397 {"group", 'g', POPT_ARG_VAL, &preserve_gid, 1, 0, 0 },
398 {"no-group", 0, POPT_ARG_VAL, &preserve_gid, 0, 0, 0 },
399 {"no-g", 0, POPT_ARG_VAL, &preserve_gid, 0, 0, 0 },
400 {"devices", 'D', POPT_ARG_VAL, &preserve_devices, 1, 0, 0 },
401 {"no-devices", 0, POPT_ARG_VAL, &preserve_devices, 0, 0, 0 },
402 {"no-D", 0, POPT_ARG_VAL, &preserve_devices, 0, 0, 0 },
e86e2fa1 403 {"links", 'l', POPT_ARG_VAL, &preserve_links, 1, 0, 0 },
b6164938
WD
404 {"no-links", 0, POPT_ARG_VAL, &preserve_links, 0, 0, 0 },
405 {"no-l", 0, POPT_ARG_VAL, &preserve_links, 0, 0, 0 },
e86e2fa1
WD
406 {"copy-links", 'L', POPT_ARG_NONE, &copy_links, 0, 0, 0 },
407 {"copy-unsafe-links",0, POPT_ARG_NONE, &copy_unsafe_links, 0, 0, 0 },
408 {"safe-links", 0, POPT_ARG_NONE, &safe_symlinks, 0, 0, 0 },
409 {"keep-dirlinks", 'K', POPT_ARG_NONE, &keep_dirlinks, 0, 0, 0 },
410 {"hard-links", 'H', POPT_ARG_NONE, &preserve_hard_links, 0, 0, 0 },
411 {"relative", 'R', POPT_ARG_VAL, &relative_paths, 1, 0, 0 },
412 {"no-relative", 0, POPT_ARG_VAL, &relative_paths, 0, 0, 0 },
b6164938 413 {"no-R", 0, POPT_ARG_VAL, &relative_paths, 0, 0, 0 },
e86e2fa1 414 {"no-implied-dirs", 0, POPT_ARG_VAL, &implied_dirs, 0, 0, 0 },
afb6e945
WD
415 {"ignore-times", 'I', POPT_ARG_NONE, &ignore_times, 0, 0, 0 },
416 {"size-only", 0, POPT_ARG_NONE, &size_only, 0, 0, 0 },
afb6e945 417 {"one-file-system", 'x', POPT_ARG_NONE, &one_file_system, 0, 0, 0 },
e86e2fa1 418 {"update", 'u', POPT_ARG_NONE, &update_only, 0, 0, 0 },
afb6e945
WD
419 {"existing", 0, POPT_ARG_NONE, &only_existing, 0, 0, 0 },
420 {"ignore-existing", 0, POPT_ARG_NONE, &opt_ignore_existing, 0, 0, 0 },
e86e2fa1
WD
421 {"max-size", 0, POPT_ARG_STRING, &max_size_arg, OPT_MAX_SIZE, 0, 0 },
422 {"sparse", 'S', POPT_ARG_NONE, &sparse_files, 0, 0, 0 },
423 {"inplace", 0, POPT_ARG_NONE, &inplace, 0, 0, 0 },
424 {"append", 0, POPT_ARG_VAL, &append_mode, 1, 0, 0 },
3671987f 425 {"del", 0, POPT_ARG_NONE, &delete_during, 0, 0, 0 },
3359acb8 426 {"delete", 0, POPT_ARG_NONE, &delete_mode, 0, 0, 0 },
c6eb7fad 427 {"delete-before", 0, POPT_ARG_VAL, &delete_before, 2, 0, 0 },
3359acb8 428 {"delete-during", 0, POPT_ARG_NONE, &delete_during, 0, 0, 0 },
51d48398
WD
429 {"delete-after", 0, POPT_ARG_NONE, &delete_after, 0, 0, 0 },
430 {"delete-excluded", 0, POPT_ARG_NONE, &delete_excluded, 0, 0, 0 },
07c6ae7d 431 {"remove-sent-files",0, POPT_ARG_NONE, &remove_sent_files, 0, 0, 0 },
afb6e945 432 {"force", 0, POPT_ARG_NONE, &force_delete, 0, 0, 0 },
e86e2fa1
WD
433 {"ignore-errors", 0, POPT_ARG_NONE, &ignore_errors, 0, 0, 0 },
434 {"max-delete", 0, POPT_ARG_INT, &max_delete, 0, 0, 0 },
435 {0, 'F', POPT_ARG_NONE, 0, 'F', 0, 0 },
aa4d3b4c 436 {"filter", 'f', POPT_ARG_STRING, 0, OPT_FILTER, 0, 0 },
8db7cc2c
WD
437 {"exclude", 0, POPT_ARG_STRING, 0, OPT_EXCLUDE, 0, 0 },
438 {"include", 0, POPT_ARG_STRING, 0, OPT_INCLUDE, 0, 0 },
439 {"exclude-from", 0, POPT_ARG_STRING, 0, OPT_EXCLUDE_FROM, 0, 0 },
440 {"include-from", 0, POPT_ARG_STRING, 0, OPT_INCLUDE_FROM, 0, 0 },
afb6e945 441 {"cvs-exclude", 'C', POPT_ARG_NONE, &cvs_exclude, 0, 0, 0 },
afb6e945
WD
442 {"whole-file", 'W', POPT_ARG_VAL, &whole_file, 1, 0, 0 },
443 {"no-whole-file", 0, POPT_ARG_VAL, &whole_file, 0, 0, 0 },
b6164938 444 {"no-W", 0, POPT_ARG_VAL, &whole_file, 0, 0, 0 },
afb6e945 445 {"checksum", 'c', POPT_ARG_NONE, &always_checksum, 0, 0, 0 },
a06b419d 446 {"block-size", 'B', POPT_ARG_LONG, &block_size, 0, 0, 0 },
e012f858 447 {"compare-dest", 0, POPT_ARG_STRING, 0, OPT_COMPARE_DEST, 0, 0 },
1de3e99b 448 {"copy-dest", 0, POPT_ARG_STRING, 0, OPT_COPY_DEST, 0, 0 },
e012f858 449 {"link-dest", 0, POPT_ARG_STRING, 0, OPT_LINK_DEST, 0, 0 },
c4ed1487 450 {"fuzzy", 'y', POPT_ARG_NONE, &fuzzy_basis, 0, 0, 0 },
afb6e945 451 {"compress", 'z', POPT_ARG_NONE, &do_compression, 0, 0, 0 },
11bfaf63 452 {0, 'P', POPT_ARG_NONE, 0, 'P', 0, 0 },
b6164938
WD
453 {"progress", 0, POPT_ARG_VAL, &do_progress, 1, 0, 0 },
454 {"no-progress", 0, POPT_ARG_VAL, &do_progress, 0, 0, 0 },
455 {"partial", 0, POPT_ARG_VAL, &keep_partial, 1, 0, 0 },
456 {"no-partial", 0, POPT_ARG_VAL, &keep_partial, 0, 0, 0 },
a7260c40 457 {"partial-dir", 0, POPT_ARG_STRING, &partial_dir, 0, 0, 0 },
f06e7082 458 {"delay-updates", 0, POPT_ARG_NONE, &delay_updates, 0, 0, 0 },
afb6e945 459 {"log-format", 0, POPT_ARG_STRING, &log_format, 0, 0, 0 },
b78296cb 460 {"itemize-changes", 'i', POPT_ARG_NONE, &itemize_changes, 0, 0, 0 },
afb6e945 461 {"bwlimit", 0, POPT_ARG_INT, &bwlimit, 0, 0, 0 },
11bfaf63 462 {"backup", 'b', POPT_ARG_NONE, &make_backups, 0, 0, 0 },
afb6e945 463 {"backup-dir", 0, POPT_ARG_STRING, &backup_dir, 0, 0, 0 },
e86e2fa1
WD
464 {"suffix", 0, POPT_ARG_STRING, &backup_suffix, 0, 0, 0 },
465 {"list-only", 0, POPT_ARG_VAL, &list_only, 2, 0, 0 },
8db7cc2c
WD
466 {"read-batch", 0, POPT_ARG_STRING, &batch_name, OPT_READ_BATCH, 0, 0 },
467 {"write-batch", 0, POPT_ARG_STRING, &batch_name, OPT_WRITE_BATCH, 0, 0 },
11e758a4 468 {"only-write-batch", 0, POPT_ARG_STRING, &batch_name, OPT_ONLY_WRITE_BATCH, 0, 0 },
ea5164d1
WD
469 {"files-from", 0, POPT_ARG_STRING, &files_from, 0, 0, 0 },
470 {"from0", '0', POPT_ARG_NONE, &eol_nulls, 0, 0, 0},
e86e2fa1
WD
471 {"numeric-ids", 0, POPT_ARG_NONE, &numeric_ids, 0, 0, 0 },
472 {"timeout", 0, POPT_ARG_INT, &io_timeout, 0, 0, 0 },
473 {"rsh", 'e', POPT_ARG_STRING, &shell_cmd, 0, 0, 0 },
474 {"rsync-path", 0, POPT_ARG_STRING, &rsync_path, 0, 0, 0 },
475 {"temp-dir", 'T', POPT_ARG_STRING, &tmpdir, 0, 0, 0 },
4f5b0756 476#ifdef INET6
3dd22903
WD
477 {"ipv4", '4', POPT_ARG_VAL, &default_af_hint, AF_INET, 0, 0 },
478 {"ipv6", '6', POPT_ARG_VAL, &default_af_hint, AF_INET6, 0, 0 },
06963d0f 479#endif
e86e2fa1
WD
480 {"address", 0, POPT_ARG_STRING, &bind_address, 0, 0, 0 },
481 {"port", 0, POPT_ARG_INT, &rsync_port, 0, 0, 0 },
482 {"password-file", 0, POPT_ARG_STRING, &password_file, 0, 0, 0 },
483 {"blocking-io", 0, POPT_ARG_VAL, &blocking_io, 1, 0, 0 },
484 {"no-blocking-io", 0, POPT_ARG_VAL, &blocking_io, 0, 0, 0 },
485 {"protocol", 0, POPT_ARG_INT, &protocol_version, 0, 0, 0 },
486 {"checksum-seed", 0, POPT_ARG_INT, &checksum_seed, 0, 0, 0 },
487 {"server", 0, POPT_ARG_NONE, &am_server, 0, 0, 0 },
488 {"sender", 0, POPT_ARG_NONE, 0, OPT_SENDER, 0, 0 },
489 /* All the following options switch us into daemon-mode option-parsing. */
3ac7f5d4
WD
490 {"config", 0, POPT_ARG_STRING, 0, OPT_DAEMON, 0, 0 },
491 {"daemon", 0, POPT_ARG_NONE, 0, OPT_DAEMON, 0, 0 },
1f35babc 492 {"detach", 0, POPT_ARG_NONE, 0, OPT_DAEMON, 0, 0 },
3ac7f5d4 493 {"no-detach", 0, POPT_ARG_NONE, 0, OPT_DAEMON, 0, 0 },
3ac7f5d4
WD
494 {0,0,0,0, 0, 0, 0}
495};
496
497static void daemon_usage(enum logcode F)
498{
499 print_rsync_version(F);
500
501 rprintf(F,"\nUsage: rsync --daemon [OPTION]...\n");
502 rprintf(F," --address=ADDRESS bind to the specified address\n");
b3708acf 503 rprintf(F," --bwlimit=KBPS limit I/O bandwidth; KBytes per second\n");
3ac7f5d4
WD
504 rprintf(F," --config=FILE specify alternate rsyncd.conf file\n");
505 rprintf(F," --no-detach do not detach from the parent\n");
b3708acf 506 rprintf(F," --port=PORT listen on alternate port number\n");
1bd9db74 507 rprintf(F," -v, --verbose increase verbosity\n");
4f5b0756 508#ifdef INET6
3ac7f5d4
WD
509 rprintf(F," -4, --ipv4 prefer IPv4\n");
510 rprintf(F," -6, --ipv6 prefer IPv6\n");
511#endif
512 rprintf(F," -h, --help show this help screen\n");
513
514 rprintf(F,"\nIf you were not trying to invoke rsync as a daemon, avoid using any of the\n");
515 rprintf(F,"daemon-specific rsync options. See also the rsyncd.conf(5) man page.\n");
516}
517
518static struct poptOption long_daemon_options[] = {
519 /* longName, shortName, argInfo, argPtr, value, descrip, argDesc */
520 {"address", 0, POPT_ARG_STRING, &bind_address, 0, 0, 0 },
9fb08441 521 {"bwlimit", 0, POPT_ARG_INT, &daemon_bwlimit, 0, 0, 0 },
3ac7f5d4
WD
522 {"config", 0, POPT_ARG_STRING, &config_file, 0, 0, 0 },
523 {"daemon", 0, POPT_ARG_NONE, &daemon_opt, 0, 0, 0 },
4f5b0756 524#ifdef INET6
3ac7f5d4
WD
525 {"ipv4", '4', POPT_ARG_VAL, &default_af_hint, AF_INET, 0, 0 },
526 {"ipv6", '6', POPT_ARG_VAL, &default_af_hint, AF_INET6, 0, 0 },
527#endif
1f35babc
WD
528 {"detach", 0, POPT_ARG_VAL, &no_detach, 0, 0, 0 },
529 {"no-detach", 0, POPT_ARG_VAL, &no_detach, 1, 0, 0 },
3ac7f5d4
WD
530 {"port", 0, POPT_ARG_INT, &rsync_port, 0, 0, 0 },
531 {"protocol", 0, POPT_ARG_INT, &protocol_version, 0, 0, 0 },
532 {"server", 0, POPT_ARG_NONE, &am_server, 0, 0, 0 },
b8cc3587 533 {"temp-dir", 'T', POPT_ARG_STRING, &tmpdir, 0, 0, 0 },
1bd9db74 534 {"verbose", 'v', POPT_ARG_NONE, 0, 'v', 0, 0 },
b6e22a47
WD
535 {"no-verbose", 0, POPT_ARG_VAL, &verbose, 0, 0, 0 },
536 {"no-v", 0, POPT_ARG_VAL, &verbose, 0, 0, 0 },
3ac7f5d4 537 {"help", 'h', POPT_ARG_NONE, 0, 'h', 0, 0 },
ad715008 538 {0,0,0,0, 0, 0, 0}
2855f61f 539};
7a6421fa 540
06963d0f 541
e51094b7 542static char err_buf[200];
cd8185f2 543
2855f61f 544
dafe63ca
MP
545/**
546 * Store the option error message, if any, so that we can log the
547 * connection attempt (which requires parsing the options), and then
548 * show the error later on.
549 **/
cd8185f2
AT
550void option_error(void)
551{
e51094b7
WD
552 if (!err_buf[0]) {
553 strcpy(err_buf, "Error parsing options: "
554 "option may be supported on client but not on server?\n");
cd8185f2 555 }
e51094b7 556
e51094b7 557 rprintf(FERROR, RSYNC_NAME ": %s", err_buf);
cd8185f2
AT
558}
559
dafe63ca
MP
560
561/**
27ed20f7
WD
562 * Tweak the option table to disable all options that the rsyncd.conf
563 * file has told us to refuse.
dafe63ca 564 **/
27ed20f7 565static void set_refuse_options(char *bp)
cd8185f2 566{
27ed20f7 567 struct poptOption *op;
093e816c 568 char *cp, shortname[2];
d73e7f6e 569 int is_wild, found_match;
093e816c
WD
570
571 shortname[1] = '\0';
27ed20f7
WD
572
573 while (1) {
06a50542
WD
574 while (*bp == ' ') bp++;
575 if (!*bp)
576 break;
27ed20f7
WD
577 if ((cp = strchr(bp, ' ')) != NULL)
578 *cp= '\0';
093e816c 579 is_wild = strpbrk(bp, "*?[") != NULL;
d73e7f6e 580 found_match = 0;
55afbb52 581 for (op = long_options; ; op++) {
093e816c 582 *shortname = op->shortName;
d73e7f6e
WD
583 if (!op->longName && !*shortname)
584 break;
585 if ((op->longName && wildmatch(bp, op->longName))
684d7582 586 || (*shortname && wildmatch(bp, shortname))) {
e57211c5
WD
587 if (op->argInfo == POPT_ARG_VAL)
588 op->argInfo = POPT_ARG_NONE;
06a50542 589 op->val = (op - long_options) + OPT_REFUSED_BASE;
d73e7f6e 590 found_match = 1;
3671987f
WD
591 /* These flags are set to let us easily check
592 * an implied option later in the code. */
593 switch (*shortname) {
3671987f
WD
594 case 'r': case 'd': case 'l': case 'p':
595 case 't': case 'g': case 'o': case 'D':
596 refused_archive_part = op->val;
597 break;
598 case '\0':
599 if (wildmatch("delete", op->longName))
600 refused_delete = op->val;
3296f91b
WD
601 else if (wildmatch("delete-before", op->longName))
602 refused_delete_before = op->val;
3671987f
WD
603 else if (wildmatch("partial", op->longName))
604 refused_partial = op->val;
605 else if (wildmatch("progress", op->longName))
606 refused_progress = op->val;
a015788d
WD
607 else if (wildmatch("inplace", op->longName))
608 refused_inplace = op->val;
3671987f
WD
609 break;
610 }
06a50542
WD
611 if (!is_wild)
612 break;
27ed20f7 613 }
cd8185f2 614 }
d73e7f6e
WD
615 if (!found_match) {
616 rprintf(FLOG, "No match for refuse-options string \"%s\"\n",
617 bp);
618 }
27ed20f7
WD
619 if (!cp)
620 break;
621 *cp = ' ';
622 bp = cp + 1;
cd8185f2 623 }
684d7582
WD
624
625 for (op = long_options; ; op++) {
626 *shortname = op->shortName;
627 if (!op->longName && !*shortname)
628 break;
629 if (op->val == OPT_DAEMON) {
630 if (op->argInfo == POPT_ARG_VAL)
631 op->argInfo = POPT_ARG_NONE;
632 op->val = (op - long_options) + OPT_REFUSED_BASE;
633 }
634 }
cd8185f2
AT
635}
636
637
8db7cc2c 638static int count_args(const char **argv)
2855f61f 639{
dfa32483 640 int i = 0;
2855f61f 641
8db7cc2c
WD
642 if (argv) {
643 while (argv[i] != NULL)
644 i++;
645 }
dfa32483
WD
646
647 return i;
2855f61f
MP
648}
649
650
95e107db
WD
651static OFF_T parse_size_arg(const char *size_arg)
652{
653 const char *arg;
654 OFF_T size;
655
656 for (arg = size_arg; isdigit(*(uchar*)arg); arg++) {}
657 if (*arg == '.')
658 for (arg++; isdigit(*(uchar*)arg); arg++) {}
659 switch (*arg) {
660 case 'k': case 'K':
661 size = atof(size_arg) * 1024;
662 break;
663 case 'm': case 'M':
664 size = atof(size_arg) * 1024*1024;
665 break;
666 case 'g': case 'G':
667 size = atof(size_arg) * 1024*1024*1024;
668 break;
669 case '\0':
670 size = atof(size_arg);
671 break;
672 default:
673 size = 0;
674 break;
675 }
676 return size;
677}
678
679
3671987f
WD
680static void create_refuse_error(int which)
681{
682 /* The "which" value is the index + OPT_REFUSED_BASE. */
683 struct poptOption *op = &long_options[which - OPT_REFUSED_BASE];
684 int n = snprintf(err_buf, sizeof err_buf,
685 "The server is configured to refuse --%s\n",
686 op->longName) - 1;
687 if (op->shortName) {
688 snprintf(err_buf + n, sizeof err_buf - n,
689 " (-%c)\n", op->shortName);
690 }
691}
692
693
dafe63ca
MP
694/**
695 * Process command line arguments. Called on both local and remote.
696 *
697 * @retval 1 if all options are OK; with globals set to appropriate
698 * values
699 *
700 * @retval 0 on error, with err_buf containing an explanation
701 **/
2855f61f 702int parse_arguments(int *argc, const char ***argv, int frommain)
7a6421fa 703{
d853783f 704 int opt;
cd8185f2 705 char *ref = lp_refuse_options(module_id);
7be73df4 706 const char *arg;
dfa32483 707 poptContext pc;
d853783f 708
27ed20f7
WD
709 if (ref && *ref)
710 set_refuse_options(ref);
711
dfa32483 712 /* TODO: Call poptReadDefaultConfig; handle errors. */
cd8185f2 713
dfa32483
WD
714 /* The context leaks in case of an error, but if there's a
715 * problem we always exit anyhow. */
716 pc = poptGetContext(RSYNC_NAME, *argc, *argv, long_options, 0);
9fb08441 717 poptReadDefaultConfig(pc, 0);
2855f61f
MP
718
719 while ((opt = poptGetNextOpt(pc)) != -1) {
dfa32483
WD
720 /* most options are handled automatically by popt;
721 * only special cases are returned and listed here. */
2855f61f 722
d853783f
AT
723 switch (opt) {
724 case OPT_VERSION:
dfa32483 725 print_rsync_version(FINFO);
d853783f 726 exit_cleanup(0);
dfa32483 727
3ac7f5d4
WD
728 case OPT_DAEMON:
729 if (am_daemon) {
730 strcpy(err_buf, "Attempt to hack rsync thwarted!\n");
731 return 0;
732 }
733 poptFreeContext(pc);
734 pc = poptGetContext(RSYNC_NAME, *argc, *argv,
735 long_daemon_options, 0);
736 while ((opt = poptGetNextOpt(pc)) != -1) {
737 switch (opt) {
738 case 'h':
739 daemon_usage(FINFO);
740 exit_cleanup(0);
741
1bd9db74
WD
742 case 'v':
743 verbose++;
744 break;
745
3ac7f5d4
WD
746 default:
747 rprintf(FERROR,
748 "rsync: %s: %s (in daemon mode)\n",
749 poptBadOption(pc, POPT_BADOPTION_NOALIAS),
750 poptStrerror(opt));
751 goto daemon_error;
752 }
753 }
b6e22a47
WD
754
755 if (tmpdir && strlen(tmpdir) >= MAXPATHLEN - 10) {
756 snprintf(err_buf, sizeof err_buf,
757 "the --temp-dir path is WAY too long.\n");
758 return 0;
759 }
760
3ac7f5d4
WD
761 if (!daemon_opt) {
762 rprintf(FERROR, "Daemon option(s) used without --daemon.\n");
763 daemon_error:
764 rprintf(FERROR,
765 "(Type \"rsync --daemon --help\" for assistance with daemon mode.)\n");
766 exit_cleanup(RERR_SYNTAX);
767 }
b6e22a47 768
3ac7f5d4
WD
769 *argv = poptGetArgs(pc);
770 *argc = count_args(*argv);
7f2a1f65 771 am_starting_up = 0;
3ac7f5d4
WD
772 daemon_opt = 0;
773 am_daemon = 1;
774 return 1;
775
5b56cc19 776 case OPT_MODIFY_WINDOW:
dfa32483
WD
777 /* The value has already been set by popt, but
778 * we need to remember that we're using a
779 * non-default setting. */
5b56cc19
AT
780 modify_window_set = 1;
781 break;
1de50993 782
aa4d3b4c 783 case OPT_FILTER:
3a5e9224 784 parse_rule(&filter_list, poptGetOptArg(pc), 0, 0);
d853783f
AT
785 break;
786
aa4d3b4c 787 case OPT_EXCLUDE:
3a5e9224 788 parse_rule(&filter_list, poptGetOptArg(pc),
0a68f869 789 0, XFLG_OLD_PREFIXES);
aa4d3b4c
WD
790 break;
791
d853783f 792 case OPT_INCLUDE:
3a5e9224 793 parse_rule(&filter_list, poptGetOptArg(pc),
0a68f869 794 MATCHFLG_INCLUDE, XFLG_OLD_PREFIXES);
d853783f
AT
795 break;
796
797 case OPT_EXCLUDE_FROM:
93695764 798 case OPT_INCLUDE_FROM:
7be73df4 799 arg = poptGetOptArg(pc);
ba3db479
WD
800 if (sanitize_paths)
801 arg = sanitize_path(NULL, arg, NULL, 0);
7842418b 802 if (server_filter_list.head) {
ba3db479 803 char *cp = (char *)arg;
3671987f
WD
804 if (!*cp)
805 goto options_rejected;
ba3db479 806 clean_fname(cp, 1);
7842418b 807 if (check_filter(&server_filter_list, cp, 0) < 0)
ba3db479
WD
808 goto options_rejected;
809 }
3a5e9224
WD
810 parse_filter_file(&filter_list, arg,
811 opt == OPT_INCLUDE_FROM ? MATCHFLG_INCLUDE : 0,
812 XFLG_FATAL_ERRORS | XFLG_OLD_PREFIXES);
93695764
DD
813 break;
814
b6164938
WD
815 case 'a':
816 if (refused_archive_part) {
817 create_refuse_error(refused_archive_part);
818 return 0;
819 }
820 if (!recurse) /* preserve recurse == 2 */
821 recurse = 1;
822#ifdef SUPPORT_LINKS
823 preserve_links = 1;
824#endif
825 preserve_perms = 1;
826 preserve_times = 1;
827 preserve_gid = 1;
828 preserve_uid = 1;
829 preserve_devices = 1;
830 break;
831
d853783f
AT
832 case 'h':
833 usage(FINFO);
834 exit_cleanup(0);
835
d853783f
AT
836 case 'v':
837 verbose++;
838 break;
7a6421fa 839
b86f0cef 840 case 'q':
f98c60bf
WD
841 if (frommain)
842 quiet++;
b86f0cef
DD
843 break;
844
d853783f
AT
845 case OPT_SENDER:
846 if (!am_server) {
847 usage(FERROR);
65417579 848 exit_cleanup(RERR_SYNTAX);
d853783f
AT
849 }
850 am_sender = 1;
851 break;
852
aa4d3b4c
WD
853 case 'F':
854 switch (++F_option_cnt) {
855 case 1:
3a5e9224 856 parse_rule(&filter_list,": /.rsync-filter",0,0);
aa4d3b4c
WD
857 break;
858 case 2:
3a5e9224 859 parse_rule(&filter_list,"- .rsync-filter",0,0);
aa4d3b4c
WD
860 break;
861 }
862 break;
863
d9fcc198 864 case 'P':
3671987f
WD
865 if (refused_partial || refused_progress) {
866 create_refuse_error(refused_partial
867 ? refused_partial : refused_progress);
868 return 0;
869 }
d9fcc198
AT
870 do_progress = 1;
871 keep_partial = 1;
872 break;
873
088aac85 874 case OPT_WRITE_BATCH:
9b3318b0 875 /* batch_name is already set */
088aac85
DD
876 write_batch = 1;
877 break;
878
11e758a4
WD
879 case OPT_ONLY_WRITE_BATCH:
880 /* batch_name is already set */
881 write_batch = -1;
882 break;
883
76f79ba7 884 case OPT_READ_BATCH:
9b3318b0 885 /* batch_name is already set */
6902ed17
MP
886 read_batch = 1;
887 break;
ea5164d1 888
7d5acf1d 889 case OPT_MAX_SIZE:
95e107db 890 if ((max_size = parse_size_arg(max_size_arg)) <= 0) {
e012f858 891 snprintf(err_buf, sizeof err_buf,
7d5acf1d
WD
892 "--max-size value is invalid: %s\n",
893 max_size_arg);
e012f858 894 return 0;
7d5acf1d
WD
895 }
896 break;
897
59c95e42 898 case OPT_LINK_DEST:
4f5b0756 899#ifdef HAVE_LINK
59c95e42 900 link_dest = 1;
e012f858
WD
901 dest_option = "--link-dest";
902 goto set_dest_dir;
59c95e42 903#else
d175d7e1 904 snprintf(err_buf, sizeof err_buf,
dfa32483 905 "hard links are not supported on this %s\n",
59c95e42 906 am_server ? "server" : "client");
59c95e42
DD
907 return 0;
908#endif
909
1de3e99b
WD
910 case OPT_COPY_DEST:
911 copy_dest = 1;
912 dest_option = "--copy-dest";
913 goto set_dest_dir;
914
e012f858
WD
915 case OPT_COMPARE_DEST:
916 compare_dest = 1;
917 dest_option = "--compare-dest";
918 set_dest_dir:
3b26bba0 919 if (basis_dir_cnt >= MAX_BASIS_DIRS) {
e012f858
WD
920 snprintf(err_buf, sizeof err_buf,
921 "ERROR: at most %d %s args may be specified\n",
922 MAX_BASIS_DIRS, dest_option);
923 return 0;
924 }
925 arg = poptGetOptArg(pc);
926 if (sanitize_paths)
927 arg = sanitize_path(NULL, arg, NULL, 0);
928 basis_dir[basis_dir_cnt++] = (char *)arg;
929 break;
930
d853783f 931 default:
55afbb52 932 /* A large opt value means that set_refuse_options()
3671987f 933 * turned this option off. */
c67d1386 934 if (opt >= OPT_REFUSED_BASE) {
3671987f
WD
935 create_refuse_error(opt);
936 return 0;
27ed20f7 937 }
3671987f
WD
938 snprintf(err_buf, sizeof err_buf, "%s%s: %s\n",
939 am_server ? "on remote machine: " : "",
940 poptBadOption(pc, POPT_BADOPTION_NOALIAS),
941 poptStrerror(opt));
dfa32483 942 return 0;
d853783f 943 }
7a6421fa 944 }
2855f61f 945
4f5b0756 946#ifndef SUPPORT_LINKS
54e87b4b 947 if (preserve_links && !am_sender) {
e1add893
WD
948 snprintf(err_buf, sizeof err_buf,
949 "symlinks are not supported on this %s\n",
950 am_server ? "server" : "client");
e1add893
WD
951 return 0;
952 }
953#endif
954
4f5b0756 955#ifndef SUPPORT_HARD_LINKS
e1add893
WD
956 if (preserve_hard_links) {
957 snprintf(err_buf, sizeof err_buf,
958 "hard links are not supported on this %s\n",
959 am_server ? "server" : "client");
e1add893
WD
960 return 0;
961 }
962#endif
963
088aac85 964 if (write_batch && read_batch) {
c3ea0990 965 snprintf(err_buf, sizeof err_buf,
36119893 966 "--write-batch and --read-batch can not be used together\n");
c3ea0990 967 return 0;
088aac85 968 }
11e758a4 969 if (write_batch > 0 || read_batch) {
94327ff0
WD
970 if (am_server) {
971 rprintf(FINFO,
972 "ignoring --%s-batch option sent to server\n",
973 write_batch ? "write" : "read");
974 /* We don't actually exit_cleanup(), so that we can
975 * still service older version clients that still send
976 * batch args to server. */
977 read_batch = write_batch = 0;
978 batch_name = NULL;
254ee3ba
WD
979 } else if (dry_run)
980 write_batch = 0;
b9f592fb 981 }
36119893 982 if (read_batch && files_from) {
c3ea0990 983 snprintf(err_buf, sizeof err_buf,
36119893 984 "--read-batch cannot be used with --files-from\n");
c3ea0990 985 return 0;
36119893 986 }
9b3318b0 987 if (batch_name && strlen(batch_name) > MAX_BATCH_NAME_LEN) {
c3ea0990 988 snprintf(err_buf, sizeof err_buf,
9b3318b0
WD
989 "the batch-file name must be %d characters or less.\n",
990 MAX_BATCH_NAME_LEN);
c3ea0990 991 return 0;
beb93684 992 }
088aac85 993
ce58b1b4 994 if (tmpdir && strlen(tmpdir) >= MAXPATHLEN - 10) {
c3ea0990
WD
995 snprintf(err_buf, sizeof err_buf,
996 "the --temp-dir path is WAY too long.\n");
997 return 0;
ce58b1b4
WD
998 }
999
1de3e99b 1000 if (compare_dest + copy_dest + link_dest > 1) {
e012f858 1001 snprintf(err_buf, sizeof err_buf,
1de3e99b 1002 "You may not mix --compare-dest, --copy-dest, and --link-dest.\n");
e012f858
WD
1003 return 0;
1004 }
1005
b6164938
WD
1006 if (files_from) {
1007 if (recurse == 1) /* preserve recurse == 2 */
1008 recurse = 0;
1009 if (xfer_dirs < 0)
1010 xfer_dirs = 1;
dfa32483 1011 }
51d48398 1012
b6164938
WD
1013 if (xfer_dirs < 1)
1014 xfer_dirs = recurse || list_only;
dfa32483 1015
ea5164d1
WD
1016 if (relative_paths < 0)
1017 relative_paths = files_from? 1 : 0;
89f2a4c2
WD
1018 if (!relative_paths)
1019 implied_dirs = 0;
ea5164d1 1020
c6eb7fad 1021 if (!!delete_before + delete_during + delete_after > 1) {
3359acb8 1022 snprintf(err_buf, sizeof err_buf,
c6eb7fad 1023 "You may not combine multiple --delete-WHEN options.\n");
3359acb8
WD
1024 return 0;
1025 }
78fc60cd
WD
1026 if (!recurse) {
1027 delete_before = delete_during = delete_after = 0;
1028 delete_mode = delete_excluded = 0;
1029 } else if (delete_before || delete_during || delete_after)
51d48398 1030 delete_mode = 1;
3296f91b
WD
1031 else if (delete_mode || delete_excluded) {
1032 if (refused_delete_before) {
1033 create_refuse_error(refused_delete_before);
1034 return 0;
1035 }
3359acb8 1036 delete_mode = delete_before = 1;
3296f91b 1037 }
51d48398 1038
3671987f
WD
1039 if (delete_mode && refused_delete) {
1040 create_refuse_error(refused_delete);
1041 return 0;
1042 }
1043
07c6ae7d
WD
1044 if (remove_sent_files) {
1045 /* We only want to infer this refusal of --remove-sent-files
1046 * via the refusal of "delete", not any of the "delete-FOO"
1047 * options. */
1048 if (refused_delete && am_sender) {
1049 create_refuse_error(refused_delete);
1050 return 0;
1051 }
1052 need_messages_from_generator = 1;
1053 }
1054
7be73df4 1055 *argv = poptGetArgs(pc);
8db7cc2c 1056 *argc = count_args(*argv);
7be73df4
WD
1057
1058 if (sanitize_paths) {
1059 int i;
1060 for (i = *argc; i-- > 0; )
10796f4b 1061 (*argv)[i] = sanitize_path(NULL, (*argv)[i], "", 0);
7be73df4 1062 if (tmpdir)
10796f4b 1063 tmpdir = sanitize_path(NULL, tmpdir, NULL, 0);
a7260c40 1064 if (partial_dir)
10796f4b 1065 partial_dir = sanitize_path(NULL, partial_dir, NULL, 0);
7be73df4 1066 if (backup_dir)
10796f4b 1067 backup_dir = sanitize_path(NULL, backup_dir, NULL, 0);
7be73df4 1068 }
7842418b
WD
1069 if (server_filter_list.head && !am_sender) {
1070 struct filter_list_struct *elp = &server_filter_list;
e012f858 1071 int i;
c3ea0990 1072 if (tmpdir) {
3671987f
WD
1073 if (!*tmpdir)
1074 goto options_rejected;
58b1999e 1075 clean_fname(tmpdir, 1);
7842418b 1076 if (check_filter(elp, tmpdir, 1) < 0)
c3ea0990
WD
1077 goto options_rejected;
1078 }
3671987f 1079 if (partial_dir && *partial_dir) {
58b1999e 1080 clean_fname(partial_dir, 1);
7842418b 1081 if (check_filter(elp, partial_dir, 1) < 0)
c3ea0990
WD
1082 goto options_rejected;
1083 }
e012f858 1084 for (i = 0; i < basis_dir_cnt; i++) {
3671987f
WD
1085 if (!*basis_dir[i])
1086 goto options_rejected;
e012f858 1087 clean_fname(basis_dir[i], 1);
7842418b 1088 if (check_filter(elp, basis_dir[i], 1) < 0)
c3ea0990
WD
1089 goto options_rejected;
1090 }
1091 if (backup_dir) {
3671987f
WD
1092 if (!*backup_dir)
1093 goto options_rejected;
58b1999e 1094 clean_fname(backup_dir, 1);
305666bf
WD
1095 if (check_filter(elp, backup_dir, 1) < 0) {
1096 options_rejected:
1097 snprintf(err_buf, sizeof err_buf,
1098 "Your options have been rejected by the server.\n");
1099 return 0;
1100 }
c3ea0990
WD
1101 }
1102 }
7be73df4 1103
d175d7e1 1104 if (!backup_suffix)
e0391f81 1105 backup_suffix = backup_dir ? "" : BACKUP_SUFFIX;
d175d7e1 1106 backup_suffix_len = strlen(backup_suffix);
80ddadb7 1107 if (strchr(backup_suffix, '/') != NULL) {
c3ea0990
WD
1108 snprintf(err_buf, sizeof err_buf,
1109 "--suffix cannot contain slashes: %s\n",
80ddadb7 1110 backup_suffix);
c3ea0990 1111 return 0;
80ddadb7 1112 }
e0391f81
WD
1113 if (backup_dir) {
1114 backup_dir_len = strlcpy(backup_dir_buf, backup_dir, sizeof backup_dir_buf);
1115 backup_dir_remainder = sizeof backup_dir_buf - backup_dir_len;
1116 if (backup_dir_remainder < 32) {
c3ea0990
WD
1117 snprintf(err_buf, sizeof err_buf,
1118 "the --backup-dir path is WAY too long.\n");
1119 return 0;
e0391f81
WD
1120 }
1121 if (backup_dir_buf[backup_dir_len - 1] != '/') {
1122 backup_dir_buf[backup_dir_len++] = '/';
1123 backup_dir_buf[backup_dir_len] = '\0';
1124 }
4875d6b6
WD
1125 if (verbose > 1 && !am_sender) {
1126 rprintf(FINFO, "backup_dir is %s\n",
1127 safe_fname(backup_dir_buf));
1128 }
8dcf9335 1129 } else if (!backup_suffix_len && (!am_server || !am_sender)) {
c3ea0990 1130 snprintf(err_buf, sizeof err_buf,
d175d7e1 1131 "--suffix cannot be a null string without --backup-dir\n");
c3ea0990 1132 return 0;
d175d7e1 1133 }
d4021b6d
WD
1134 if (make_backups && !backup_dir)
1135 omit_dir_times = 1;
d175d7e1 1136
e7651884 1137 if (log_format) {
624d6be2 1138 if (log_format_has(log_format, 'i'))
684d7582 1139 log_format_has_i = 1;
624d6be2
WD
1140 if (!log_format_has(log_format, 'b')
1141 && !log_format_has(log_format, 'c'))
e7651884
WD
1142 log_before_transfer = !am_server;
1143 } else if (itemize_changes) {
1144 log_format = "%i %n%L";
684d7582 1145 log_format_has_i = 1;
e7651884
WD
1146 log_before_transfer = !am_server;
1147 }
87383697
WD
1148
1149 if ((do_progress || dry_run) && !verbose && !log_before_transfer
702cd15c 1150 && !am_server)
e2559dbe 1151 verbose = 1;
87383697 1152
11e758a4
WD
1153 if (dry_run)
1154 do_xfers = 0;
1155
9ac756c6
WD
1156 set_io_timeout(io_timeout);
1157
87383697
WD
1158 if (verbose && !log_format) {
1159 log_format = "%n%L";
1160 log_before_transfer = !am_server;
3671987f 1161 }
624d6be2 1162 if (log_format_has_i || log_format_has(log_format, 'o'))
5c5e1598 1163 log_format_has_o_or_i = 1;
e2559dbe 1164
9fb08441
WD
1165 if (daemon_bwlimit && (!bwlimit || bwlimit > daemon_bwlimit))
1166 bwlimit = daemon_bwlimit;
3c74c3a3
WD
1167 if (bwlimit) {
1168 bwlimit_writemax = (size_t)bwlimit * 128;
1169 if (bwlimit_writemax < 512)
1170 bwlimit_writemax = 512;
1171 }
1172
cfce9f6d 1173 if (sparse_files && inplace) {
c3851185 1174 /* Note: we don't check for this below, because --append is
cfce9f6d
WD
1175 * OK with --sparse (as long as redos are handled right). */
1176 snprintf(err_buf, sizeof err_buf,
1177 "--sparse cannot be used with --inplace\n");
1178 return 0;
1179 }
1180
a015788d
WD
1181 if (append_mode) {
1182 if (whole_file > 0) {
1183 snprintf(err_buf, sizeof err_buf,
1184 "--append cannot be used with --whole-file\n");
1185 return 0;
1186 }
1187 if (refused_inplace) {
1188 create_refuse_error(refused_inplace);
1189 return 0;
1190 }
1191 inplace = 1;
1192 }
1193
f06e7082 1194 if (delay_updates && !partial_dir)
3671987f 1195 partial_dir = partialdir_for_delayupdate;
f06e7082 1196
a3221d2a 1197 if (inplace) {
4f5b0756 1198#ifdef HAVE_FTRUNCATE
a7260c40
WD
1199 if (partial_dir) {
1200 snprintf(err_buf, sizeof err_buf,
a015788d
WD
1201 "--%s cannot be used with --%s\n",
1202 append_mode ? "append" : "inplace",
f06e7082 1203 delay_updates ? "delay-updates" : "partial-dir");
a7260c40
WD
1204 return 0;
1205 }
3671987f
WD
1206 /* --inplace implies --partial for refusal purposes, but we
1207 * clear the keep_partial flag for internal logic purposes. */
1208 if (refused_partial) {
1209 create_refuse_error(refused_partial);
1210 return 0;
1211 }
a3221d2a 1212 keep_partial = 0;
ded4daf0
WD
1213#else
1214 snprintf(err_buf, sizeof err_buf,
a015788d
WD
1215 "--%s is not supported on this %s\n",
1216 append_mode ? "append" : "inplace",
ded4daf0
WD
1217 am_server ? "server" : "client");
1218 return 0;
1219#endif
075aa18f 1220 } else {
3671987f
WD
1221 if (keep_partial && !partial_dir) {
1222 if ((arg = getenv("RSYNC_PARTIAL_DIR")) != NULL && *arg)
1223 partial_dir = strdup(arg);
1224 }
075aa18f 1225 if (partial_dir) {
3671987f
WD
1226 if (*partial_dir)
1227 clean_fname(partial_dir, 1);
075aa18f
WD
1228 if (!*partial_dir || strcmp(partial_dir, ".") == 0)
1229 partial_dir = NULL;
13791b1e 1230 else if (*partial_dir != '/') {
3a5e9224 1231 parse_rule(&filter_list, partial_dir,
0a68f869 1232 MATCHFLG_NO_PREFIXES|MATCHFLG_DIRECTORY, 0);
13791b1e 1233 }
3671987f
WD
1234 if (!partial_dir && refused_partial) {
1235 create_refuse_error(refused_partial);
1236 return 0;
1237 }
075aa18f
WD
1238 keep_partial = 1;
1239 }
a3221d2a
WD
1240 }
1241
ea5164d1 1242 if (files_from) {
305666bf
WD
1243 char *h, *p;
1244 int q;
c3ea0990 1245 if (*argc > 2 || (!am_daemon && *argc == 1)) {
ea5164d1
WD
1246 usage(FERROR);
1247 exit_cleanup(RERR_SYNTAX);
1248 }
63596e1c 1249 if (strcmp(files_from, "-") == 0) {
ea5164d1 1250 filesfrom_fd = 0;
63596e1c 1251 if (am_server)
305666bf
WD
1252 filesfrom_host = ""; /* reading from socket */
1253 } else if ((p = check_for_hostspec(files_from, &h, &q)) != 0) {
ea5164d1 1254 if (am_server) {
305666bf
WD
1255 snprintf(err_buf, sizeof err_buf,
1256 "The --files-from sent to the server cannot specify a host.\n");
1257 return 0;
ea5164d1 1258 }
305666bf
WD
1259 files_from = p;
1260 filesfrom_host = h;
1261 if (strcmp(files_from, "-") == 0) {
c3ea0990
WD
1262 snprintf(err_buf, sizeof err_buf,
1263 "Invalid --files-from remote filename\n");
1264 return 0;
ea5164d1
WD
1265 }
1266 } else {
305666bf
WD
1267 if (sanitize_paths)
1268 files_from = sanitize_path(NULL, files_from, NULL, 0);
1269 if (server_filter_list.head) {
1270 if (!*files_from)
1271 goto options_rejected;
1272 clean_fname(files_from, 1);
1273 if (check_filter(&server_filter_list, files_from, 0) < 0)
1274 goto options_rejected;
1275 }
ea5164d1
WD
1276 filesfrom_fd = open(files_from, O_RDONLY|O_BINARY);
1277 if (filesfrom_fd < 0) {
c3ea0990
WD
1278 snprintf(err_buf, sizeof err_buf,
1279 "failed to open files-from file %s: %s\n",
1280 files_from, strerror(errno));
1281 return 0;
ea5164d1
WD
1282 }
1283 }
1284 }
1285
7f2a1f65
WD
1286 am_starting_up = 0;
1287
b11ed3b1 1288 return 1;
7a6421fa
AT
1289}
1290
1291
dafe63ca
MP
1292/**
1293 * Construct a filtered list of options to pass through from the
1294 * client to the server.
1295 *
1296 * This involves setting options that will tell the server how to
1297 * behave, and also filtering out options that are processed only
1298 * locally.
1299 **/
7a6421fa
AT
1300void server_options(char **args,int *argc)
1301{
74ba98a5 1302 static char argstr[64];
d853783f 1303 int ac = *argc;
f98c60bf 1304 char *arg;
ef5d23eb 1305
d853783f
AT
1306 int i, x;
1307
93689aa5
DD
1308 if (blocking_io == -1)
1309 blocking_io = 0;
1310
d853783f
AT
1311 args[ac++] = "--server";
1312
1312d9fc
WD
1313 if (daemon_over_rsh) {
1314 args[ac++] = "--daemon";
1315 *argc = ac;
1316 /* if we're passing --daemon, we're done */
1317 return;
1318 }
1319
d853783f
AT
1320 if (!am_sender)
1321 args[ac++] = "--sender";
1322
1323 x = 1;
1324 argstr[0] = '-';
f98c60bf 1325 for (i = 0; i < verbose; i++)
d853783f 1326 argstr[x++] = 'v';
f0b36a48 1327
b86f0cef 1328 /* the -q option is intentionally left out */
d853783f
AT
1329 if (make_backups)
1330 argstr[x++] = 'b';
1331 if (update_only)
1332 argstr[x++] = 'u';
4a34c6f1 1333 if (!do_xfers) /* NOT "dry_run"! */
d853783f
AT
1334 argstr[x++] = 'n';
1335 if (preserve_links)
1336 argstr[x++] = 'l';
1337 if (copy_links)
1338 argstr[x++] = 'L';
65e4cda0 1339 if (xfer_dirs > 1)
5454d22a 1340 argstr[x++] = 'd';
716e73d4
WD
1341 if (keep_dirlinks && am_sender)
1342 argstr[x++] = 'K';
1bfbf40b 1343
dfa32483 1344 if (whole_file > 0)
d853783f 1345 argstr[x++] = 'W';
bceec82f
MP
1346 /* We don't need to send --no-whole-file, because it's the
1347 * default for remote transfers, and in any case old versions
1348 * of rsync will not understand it. */
dfa32483 1349
d853783f
AT
1350 if (preserve_hard_links)
1351 argstr[x++] = 'H';
1352 if (preserve_uid)
1353 argstr[x++] = 'o';
1354 if (preserve_gid)
1355 argstr[x++] = 'g';
1356 if (preserve_devices)
1357 argstr[x++] = 'D';
1358 if (preserve_times)
1359 argstr[x++] = 't';
d4021b6d 1360 if (omit_dir_times == 2 && am_sender)
20fb7b91 1361 argstr[x++] = 'O';
d853783f
AT
1362 if (preserve_perms)
1363 argstr[x++] = 'p';
aa7a6e87 1364 if (recurse)
d853783f
AT
1365 argstr[x++] = 'r';
1366 if (always_checksum)
1367 argstr[x++] = 'c';
1368 if (cvs_exclude)
1369 argstr[x++] = 'C';
1370 if (ignore_times)
1371 argstr[x++] = 'I';
1372 if (relative_paths)
1373 argstr[x++] = 'R';
1374 if (one_file_system)
1375 argstr[x++] = 'x';
1376 if (sparse_files)
1377 argstr[x++] = 'S';
1378 if (do_compression)
1379 argstr[x++] = 'z';
f0b36a48 1380
51d48398
WD
1381 /* This is a complete hack - blame Rusty. FIXME!
1382 * This hack is only needed for older rsync versions that
1383 * don't understand the --list-only option. */
da2a6c1f 1384 if (list_only == 1 && !recurse)
f0b36a48
AT
1385 argstr[x++] = 'r';
1386
d853783f
AT
1387 argstr[x] = 0;
1388
f98c60bf
WD
1389 if (x != 1)
1390 args[ac++] = argstr;
d853783f 1391
51d48398
WD
1392 if (list_only > 1)
1393 args[ac++] = "--list-only";
1394
5c5e1598
WD
1395 /* The server side doesn't use our log-format, but in certain
1396 * circumstances they need to know a little about the option. */
684d7582
WD
1397 if (log_format && am_sender) {
1398 if (log_format_has_i)
1399 args[ac++] = "--log-format=%i";
1400 else if (log_format_has_o_or_i)
5c5e1598
WD
1401 args[ac++] = "--log-format=%o";
1402 else if (!verbose)
1403 args[ac++] = "--log-format=X";
1404 }
1f30a674 1405
195bd906 1406 if (block_size) {
a06b419d 1407 if (asprintf(&arg, "-B%lu", block_size) < 0)
f98c60bf
WD
1408 goto oom;
1409 args[ac++] = arg;
dfa32483 1410 }
d853783f 1411
0b73ca12 1412 if (max_delete && am_sender) {
f98c60bf
WD
1413 if (asprintf(&arg, "--max-delete=%d", max_delete) < 0)
1414 goto oom;
1415 args[ac++] = arg;
dfa32483
WD
1416 }
1417
7d5acf1d
WD
1418 if (max_size && am_sender) {
1419 args[ac++] = "--max-size";
1420 args[ac++] = max_size_arg;
1421 }
1422
d853783f 1423 if (io_timeout) {
f98c60bf
WD
1424 if (asprintf(&arg, "--timeout=%d", io_timeout) < 0)
1425 goto oom;
1426 args[ac++] = arg;
dfa32483 1427 }
d853783f 1428
ef5d23eb 1429 if (bwlimit) {
f98c60bf
WD
1430 if (asprintf(&arg, "--bwlimit=%d", bwlimit) < 0)
1431 goto oom;
1432 args[ac++] = arg;
ef5d23eb
DD
1433 }
1434
d175d7e1
WD
1435 if (backup_dir) {
1436 args[ac++] = "--backup-dir";
1437 args[ac++] = backup_dir;
1438 }
1439
1440 /* Only send --suffix if it specifies a non-default value. */
f98c60bf 1441 if (strcmp(backup_suffix, backup_dir ? "" : BACKUP_SUFFIX) != 0) {
191e40da 1442 /* We use the following syntax to avoid weirdness with '~'. */
f98c60bf
WD
1443 if (asprintf(&arg, "--suffix=%s", backup_suffix) < 0)
1444 goto oom;
1445 args[ac++] = arg;
d853783f
AT
1446 }
1447
06a50542
WD
1448 if (am_sender) {
1449 if (delete_excluded)
1450 args[ac++] = "--delete-excluded";
c6eb7fad 1451 else if (delete_before == 1 || delete_after)
06a50542 1452 args[ac++] = "--delete";
57f74bd1 1453 if (delete_before > 1)
c6eb7fad 1454 args[ac++] = "--delete-before";
3359acb8
WD
1455 if (delete_during)
1456 args[ac++] = "--delete-during";
06a50542
WD
1457 if (delete_after)
1458 args[ac++] = "--delete-after";
06a50542
WD
1459 if (force_delete)
1460 args[ac++] = "--force";
11e758a4
WD
1461 if (write_batch < 0)
1462 args[ac++] = "--only-write-batch=X";
06a50542 1463 }
b33b791e 1464
f83f0548
AT
1465 if (size_only)
1466 args[ac++] = "--size-only";
1467
5b56cc19 1468 if (modify_window_set) {
f98c60bf
WD
1469 if (asprintf(&arg, "--modify-window=%d", modify_window) < 0)
1470 goto oom;
1471 args[ac++] = arg;
5b56cc19
AT
1472 }
1473
c8d895de 1474 if (checksum_seed) {
221ddb94 1475 if (asprintf(&arg, "--checksum-seed=%d", checksum_seed) < 0)
c8d895de
WD
1476 goto oom;
1477 args[ac++] = arg;
1478 }
1479
a7260c40 1480 if (partial_dir && am_sender) {
3671987f
WD
1481 if (partial_dir != partialdir_for_delayupdate) {
1482 args[ac++] = "--partial-dir";
1483 args[ac++] = partial_dir;
1484 }
f06e7082
WD
1485 if (delay_updates)
1486 args[ac++] = "--delay-updates";
a7260c40 1487 } else if (keep_partial)
d853783f
AT
1488 args[ac++] = "--partial";
1489
ef55c686
AT
1490 if (ignore_errors)
1491 args[ac++] = "--ignore-errors";
1492
b5313607
DD
1493 if (copy_unsafe_links)
1494 args[ac++] = "--copy-unsafe-links";
1495
d853783f
AT
1496 if (safe_symlinks)
1497 args[ac++] = "--safe-links";
1498
1499 if (numeric_ids)
1500 args[ac++] = "--numeric-ids";
1501
0b73ca12 1502 if (only_existing && am_sender)
1347d512
AT
1503 args[ac++] = "--existing";
1504
dfa32483 1505 if (opt_ignore_existing && am_sender)
3d6feada
MP
1506 args[ac++] = "--ignore-existing";
1507
a015788d
WD
1508 if (append_mode)
1509 args[ac++] = "--append";
1510 else if (inplace)
a3221d2a
WD
1511 args[ac++] = "--inplace";
1512
d853783f
AT
1513 if (tmpdir) {
1514 args[ac++] = "--temp-dir";
1515 args[ac++] = tmpdir;
1516 }
1517
e012f858 1518 if (basis_dir[0] && am_sender) {
375a4556
DD
1519 /* the server only needs this option if it is not the sender,
1520 * and it may be an older version that doesn't know this
1521 * option, so don't send it if client is the sender.
1522 */
e012f858
WD
1523 int i;
1524 for (i = 0; i < basis_dir_cnt; i++) {
1525 args[ac++] = dest_option;
1526 args[ac++] = basis_dir[i];
1527 }
375a4556
DD
1528 }
1529
305666bf
WD
1530 if (files_from && (!am_sender || filesfrom_host)) {
1531 if (filesfrom_host) {
ea5164d1 1532 args[ac++] = "--files-from";
305666bf 1533 args[ac++] = files_from;
ea5164d1
WD
1534 if (eol_nulls)
1535 args[ac++] = "--from0";
1536 } else {
1537 args[ac++] = "--files-from=-";
1538 args[ac++] = "--from0";
1539 }
c0ab28d1
WD
1540 if (!relative_paths)
1541 args[ac++] = "--no-relative";
ea5164d1 1542 }
89f2a4c2 1543 if (relative_paths && !implied_dirs && !am_sender)
3a90ea0a 1544 args[ac++] = "--no-implied-dirs";
ea5164d1 1545
c4ed1487
WD
1546 if (fuzzy_basis && am_sender)
1547 args[ac++] = "--fuzzy";
1548
07c6ae7d
WD
1549 if (remove_sent_files)
1550 args[ac++] = "--remove-sent-files";
1551
d853783f 1552 *argc = ac;
f98c60bf
WD
1553 return;
1554
1555 oom:
1556 out_of_memory("server_options");
7a6421fa
AT
1557}
1558
305666bf
WD
1559/* Look for a HOST specfication of the form "HOST:PATH", "HOST::PATH", or
1560 * "rsync://HOST:PORT/PATH". If found, *host_ptr will be set to some allocated
1561 * memory with the HOST. If a daemon-accessing spec was specified, the value
1562 * of *port_ptr will contain a non-0 port number, otherwise it will be set to
1563 * 0. The return value is a pointer to the PATH. Note that the HOST spec can
1564 * be an IPv6 literal address enclosed in '[' and ']' (such as "[::1]" or
1565 * "[::ffff:127.0.0.1]") which is returned without the '[' and ']'. */
1566char *check_for_hostspec(char *s, char **host_ptr, int *port_ptr)
ea5164d1 1567{
305666bf
WD
1568 char *p;
1569 int not_host;
1570
1571 if (port_ptr && strncasecmp(URL_PREFIX, s, strlen(URL_PREFIX)) == 0) {
1572 char *path;
1573 int hostlen;
1574 s += strlen(URL_PREFIX);
1575 if ((p = strchr(s, '/')) != NULL) {
1576 hostlen = p - s;
1577 path = p + 1;
1578 } else {
1579 hostlen = strlen(s);
1580 path = "";
1581 }
1582 if (*s == '[' && (p = strchr(s, ']')) != NULL) {
1583 s++;
1584 hostlen = p - s;
c60b7056
WD
1585 if (p[1] == ':')
1586 *port_ptr = atoi(p+2);
305666bf
WD
1587 } else {
1588 if ((p = strchr(s, ':')) != NULL) {
1589 hostlen = p - s;
1590 *port_ptr = atoi(p+1);
c60b7056 1591 }
305666bf 1592 }
c60b7056
WD
1593 if (!*port_ptr)
1594 *port_ptr = RSYNC_PORT;
305666bf
WD
1595 *host_ptr = new_array(char, hostlen + 1);
1596 strlcpy(*host_ptr, s, hostlen + 1);
1597 return path;
1598 }
ea5164d1 1599
305666bf
WD
1600 if (*s == '[' && (p = strchr(s, ']')) != NULL && p[1] == ':') {
1601 s++;
1602 *p = '\0';
1603 not_host = strchr(s, '/') || !strchr(s, ':');
1604 *p = ']';
1605 if (not_host)
1606 return NULL;
c60b7056 1607 p++;
305666bf
WD
1608 } else {
1609 if (!(p = strchr(s, ':')))
1610 return NULL;
1611 *p = '\0';
1612 not_host = strchr(s, '/') != NULL;
1613 *p = ':';
1614 if (not_host)
1615 return NULL;
1616 }
1617
1618 *host_ptr = new_array(char, p - s + 1);
1619 strlcpy(*host_ptr, s, p - s + 1);
ea5164d1 1620
305666bf
WD
1621 if (p[1] == ':') {
1622 if (port_ptr && !*port_ptr)
1623 *port_ptr = RSYNC_PORT;
1624 return p + 2;
1625 }
1626 if (port_ptr)
1627 *port_ptr = 0;
ea5164d1 1628
305666bf 1629 return p + 1;
ea5164d1 1630}