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