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