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