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