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