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