1 /* $OpenBSD: sftp.c,v 1.158 2013/11/20 20:54:10 deraadt Exp $ */ 2 /* $FreeBSD$ */ 3 /* 4 * Copyright (c) 2001-2004 Damien Miller <djm@openbsd.org> 5 * 6 * Permission to use, copy, modify, and distribute this software for any 7 * purpose with or without fee is hereby granted, provided that the above 8 * copyright notice and this permission notice appear in all copies. 9 * 10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 17 */ 18 19 #include "includes.h" 20 21 #include <sys/types.h> 22 #include <sys/ioctl.h> 23 #ifdef HAVE_SYS_STAT_H 24 # include <sys/stat.h> 25 #endif 26 #include <sys/param.h> 27 #include <sys/socket.h> 28 #include <sys/wait.h> 29 #ifdef HAVE_SYS_STATVFS_H 30 #include <sys/statvfs.h> 31 #endif 32 33 #include <ctype.h> 34 #include <errno.h> 35 36 #ifdef HAVE_PATHS_H 37 # include <paths.h> 38 #endif 39 #ifdef HAVE_LIBGEN_H 40 #include <libgen.h> 41 #endif 42 #ifdef HAVE_LOCALE_H 43 # include <locale.h> 44 #endif 45 #ifdef USE_LIBEDIT 46 #include <histedit.h> 47 #else 48 typedef void EditLine; 49 #endif 50 #include <signal.h> 51 #include <stdlib.h> 52 #include <stdio.h> 53 #include <string.h> 54 #include <unistd.h> 55 #include <stdarg.h> 56 57 #ifdef HAVE_UTIL_H 58 # include <util.h> 59 #endif 60 61 #include "xmalloc.h" 62 #include "log.h" 63 #include "pathnames.h" 64 #include "misc.h" 65 66 #include "sftp.h" 67 #include "buffer.h" 68 #include "sftp-common.h" 69 #include "sftp-client.h" 70 71 #define DEFAULT_COPY_BUFLEN 32768 /* Size of buffer for up/download */ 72 #define DEFAULT_NUM_REQUESTS 256 /* # concurrent outstanding requests */ 73 74 /* File to read commands from */ 75 FILE* infile; 76 77 /* Are we in batchfile mode? */ 78 int batchmode = 0; 79 80 /* PID of ssh transport process */ 81 static pid_t sshpid = -1; 82 83 /* Suppress diagnositic messages */ 84 int quiet = 0; 85 86 /* This is set to 0 if the progressmeter is not desired. */ 87 int showprogress = 1; 88 89 /* When this option is set, we always recursively download/upload directories */ 90 int global_rflag = 0; 91 92 /* When this option is set, we resume download if possible */ 93 int global_aflag = 0; 94 95 /* When this option is set, the file transfers will always preserve times */ 96 int global_pflag = 0; 97 98 /* When this option is set, transfers will have fsync() called on each file */ 99 int global_fflag = 0; 100 101 /* SIGINT received during command processing */ 102 volatile sig_atomic_t interrupted = 0; 103 104 /* I wish qsort() took a separate ctx for the comparison function...*/ 105 int sort_flag; 106 107 /* Context used for commandline completion */ 108 struct complete_ctx { 109 struct sftp_conn *conn; 110 char **remote_pathp; 111 }; 112 113 int remote_glob(struct sftp_conn *, const char *, int, 114 int (*)(const char *, int), glob_t *); /* proto for sftp-glob.c */ 115 116 extern char *__progname; 117 118 /* Separators for interactive commands */ 119 #define WHITESPACE " \t\r\n" 120 121 /* ls flags */ 122 #define LS_LONG_VIEW 0x0001 /* Full view ala ls -l */ 123 #define LS_SHORT_VIEW 0x0002 /* Single row view ala ls -1 */ 124 #define LS_NUMERIC_VIEW 0x0004 /* Long view with numeric uid/gid */ 125 #define LS_NAME_SORT 0x0008 /* Sort by name (default) */ 126 #define LS_TIME_SORT 0x0010 /* Sort by mtime */ 127 #define LS_SIZE_SORT 0x0020 /* Sort by file size */ 128 #define LS_REVERSE_SORT 0x0040 /* Reverse sort order */ 129 #define LS_SHOW_ALL 0x0080 /* Don't skip filenames starting with '.' */ 130 #define LS_SI_UNITS 0x0100 /* Display sizes as K, M, G, etc. */ 131 132 #define VIEW_FLAGS (LS_LONG_VIEW|LS_SHORT_VIEW|LS_NUMERIC_VIEW|LS_SI_UNITS) 133 #define SORT_FLAGS (LS_NAME_SORT|LS_TIME_SORT|LS_SIZE_SORT) 134 135 /* Commands for interactive mode */ 136 enum sftp_command { 137 I_CHDIR = 1, 138 I_CHGRP, 139 I_CHMOD, 140 I_CHOWN, 141 I_DF, 142 I_GET, 143 I_HELP, 144 I_LCHDIR, 145 I_LINK, 146 I_LLS, 147 I_LMKDIR, 148 I_LPWD, 149 I_LS, 150 I_LUMASK, 151 I_MKDIR, 152 I_PUT, 153 I_PWD, 154 I_QUIT, 155 I_RENAME, 156 I_RM, 157 I_RMDIR, 158 I_SHELL, 159 I_SYMLINK, 160 I_VERSION, 161 I_PROGRESS, 162 I_REGET, 163 }; 164 165 struct CMD { 166 const char *c; 167 const int n; 168 const int t; 169 }; 170 171 /* Type of completion */ 172 #define NOARGS 0 173 #define REMOTE 1 174 #define LOCAL 2 175 176 static const struct CMD cmds[] = { 177 { "bye", I_QUIT, NOARGS }, 178 { "cd", I_CHDIR, REMOTE }, 179 { "chdir", I_CHDIR, REMOTE }, 180 { "chgrp", I_CHGRP, REMOTE }, 181 { "chmod", I_CHMOD, REMOTE }, 182 { "chown", I_CHOWN, REMOTE }, 183 { "df", I_DF, REMOTE }, 184 { "dir", I_LS, REMOTE }, 185 { "exit", I_QUIT, NOARGS }, 186 { "get", I_GET, REMOTE }, 187 { "help", I_HELP, NOARGS }, 188 { "lcd", I_LCHDIR, LOCAL }, 189 { "lchdir", I_LCHDIR, LOCAL }, 190 { "lls", I_LLS, LOCAL }, 191 { "lmkdir", I_LMKDIR, LOCAL }, 192 { "ln", I_LINK, REMOTE }, 193 { "lpwd", I_LPWD, LOCAL }, 194 { "ls", I_LS, REMOTE }, 195 { "lumask", I_LUMASK, NOARGS }, 196 { "mkdir", I_MKDIR, REMOTE }, 197 { "mget", I_GET, REMOTE }, 198 { "mput", I_PUT, LOCAL }, 199 { "progress", I_PROGRESS, NOARGS }, 200 { "put", I_PUT, LOCAL }, 201 { "pwd", I_PWD, REMOTE }, 202 { "quit", I_QUIT, NOARGS }, 203 { "reget", I_REGET, REMOTE }, 204 { "rename", I_RENAME, REMOTE }, 205 { "rm", I_RM, REMOTE }, 206 { "rmdir", I_RMDIR, REMOTE }, 207 { "symlink", I_SYMLINK, REMOTE }, 208 { "version", I_VERSION, NOARGS }, 209 { "!", I_SHELL, NOARGS }, 210 { "?", I_HELP, NOARGS }, 211 { NULL, -1, -1 } 212 }; 213 214 int interactive_loop(struct sftp_conn *, char *file1, char *file2); 215 216 /* ARGSUSED */ 217 static void 218 killchild(int signo) 219 { 220 if (sshpid > 1) { 221 kill(sshpid, SIGTERM); 222 waitpid(sshpid, NULL, 0); 223 } 224 225 _exit(1); 226 } 227 228 /* ARGSUSED */ 229 static void 230 cmd_interrupt(int signo) 231 { 232 const char msg[] = "\rInterrupt \n"; 233 int olderrno = errno; 234 235 (void)write(STDERR_FILENO, msg, sizeof(msg) - 1); 236 interrupted = 1; 237 errno = olderrno; 238 } 239 240 static void 241 help(void) 242 { 243 printf("Available commands:\n" 244 "bye Quit sftp\n" 245 "cd path Change remote directory to 'path'\n" 246 "chgrp grp path Change group of file 'path' to 'grp'\n" 247 "chmod mode path Change permissions of file 'path' to 'mode'\n" 248 "chown own path Change owner of file 'path' to 'own'\n" 249 "df [-hi] [path] Display statistics for current directory or\n" 250 " filesystem containing 'path'\n" 251 "exit Quit sftp\n" 252 "get [-Ppr] remote [local] Download file\n" 253 "reget remote [local] Resume download file\n" 254 "help Display this help text\n" 255 "lcd path Change local directory to 'path'\n" 256 "lls [ls-options [path]] Display local directory listing\n" 257 "lmkdir path Create local directory\n" 258 "ln [-s] oldpath newpath Link remote file (-s for symlink)\n" 259 "lpwd Print local working directory\n" 260 "ls [-1afhlnrSt] [path] Display remote directory listing\n" 261 "lumask umask Set local umask to 'umask'\n" 262 "mkdir path Create remote directory\n" 263 "progress Toggle display of progress meter\n" 264 "put [-Ppr] local [remote] Upload file\n" 265 "pwd Display remote working directory\n" 266 "quit Quit sftp\n" 267 "rename oldpath newpath Rename remote file\n" 268 "rm path Delete remote file\n" 269 "rmdir path Remove remote directory\n" 270 "symlink oldpath newpath Symlink remote file\n" 271 "version Show SFTP version\n" 272 "!command Execute 'command' in local shell\n" 273 "! Escape to local shell\n" 274 "? Synonym for help\n"); 275 } 276 277 static void 278 local_do_shell(const char *args) 279 { 280 int status; 281 char *shell; 282 pid_t pid; 283 284 if (!*args) 285 args = NULL; 286 287 if ((shell = getenv("SHELL")) == NULL || *shell == '\0') 288 shell = _PATH_BSHELL; 289 290 if ((pid = fork()) == -1) 291 fatal("Couldn't fork: %s", strerror(errno)); 292 293 if (pid == 0) { 294 /* XXX: child has pipe fds to ssh subproc open - issue? */ 295 if (args) { 296 debug3("Executing %s -c \"%s\"", shell, args); 297 execl(shell, shell, "-c", args, (char *)NULL); 298 } else { 299 debug3("Executing %s", shell); 300 execl(shell, shell, (char *)NULL); 301 } 302 fprintf(stderr, "Couldn't execute \"%s\": %s\n", shell, 303 strerror(errno)); 304 _exit(1); 305 } 306 while (waitpid(pid, &status, 0) == -1) 307 if (errno != EINTR) 308 fatal("Couldn't wait for child: %s", strerror(errno)); 309 if (!WIFEXITED(status)) 310 error("Shell exited abnormally"); 311 else if (WEXITSTATUS(status)) 312 error("Shell exited with status %d", WEXITSTATUS(status)); 313 } 314 315 static void 316 local_do_ls(const char *args) 317 { 318 if (!args || !*args) 319 local_do_shell(_PATH_LS); 320 else { 321 int len = strlen(_PATH_LS " ") + strlen(args) + 1; 322 char *buf = xmalloc(len); 323 324 /* XXX: quoting - rip quoting code from ftp? */ 325 snprintf(buf, len, _PATH_LS " %s", args); 326 local_do_shell(buf); 327 free(buf); 328 } 329 } 330 331 /* Strip one path (usually the pwd) from the start of another */ 332 static char * 333 path_strip(char *path, char *strip) 334 { 335 size_t len; 336 337 if (strip == NULL) 338 return (xstrdup(path)); 339 340 len = strlen(strip); 341 if (strncmp(path, strip, len) == 0) { 342 if (strip[len - 1] != '/' && path[len] == '/') 343 len++; 344 return (xstrdup(path + len)); 345 } 346 347 return (xstrdup(path)); 348 } 349 350 static char * 351 make_absolute(char *p, char *pwd) 352 { 353 char *abs_str; 354 355 /* Derelativise */ 356 if (p && p[0] != '/') { 357 abs_str = path_append(pwd, p); 358 free(p); 359 return(abs_str); 360 } else 361 return(p); 362 } 363 364 static int 365 parse_getput_flags(const char *cmd, char **argv, int argc, 366 int *aflag, int *fflag, int *pflag, int *rflag) 367 { 368 extern int opterr, optind, optopt, optreset; 369 int ch; 370 371 optind = optreset = 1; 372 opterr = 0; 373 374 *aflag = *fflag = *rflag = *pflag = 0; 375 while ((ch = getopt(argc, argv, "afPpRr")) != -1) { 376 switch (ch) { 377 case 'a': 378 *aflag = 1; 379 break; 380 case 'f': 381 *fflag = 1; 382 break; 383 case 'p': 384 case 'P': 385 *pflag = 1; 386 break; 387 case 'r': 388 case 'R': 389 *rflag = 1; 390 break; 391 default: 392 error("%s: Invalid flag -%c", cmd, optopt); 393 return -1; 394 } 395 } 396 397 return optind; 398 } 399 400 static int 401 parse_link_flags(const char *cmd, char **argv, int argc, int *sflag) 402 { 403 extern int opterr, optind, optopt, optreset; 404 int ch; 405 406 optind = optreset = 1; 407 opterr = 0; 408 409 *sflag = 0; 410 while ((ch = getopt(argc, argv, "s")) != -1) { 411 switch (ch) { 412 case 's': 413 *sflag = 1; 414 break; 415 default: 416 error("%s: Invalid flag -%c", cmd, optopt); 417 return -1; 418 } 419 } 420 421 return optind; 422 } 423 424 static int 425 parse_rename_flags(const char *cmd, char **argv, int argc, int *lflag) 426 { 427 extern int opterr, optind, optopt, optreset; 428 int ch; 429 430 optind = optreset = 1; 431 opterr = 0; 432 433 *lflag = 0; 434 while ((ch = getopt(argc, argv, "l")) != -1) { 435 switch (ch) { 436 case 'l': 437 *lflag = 1; 438 break; 439 default: 440 error("%s: Invalid flag -%c", cmd, optopt); 441 return -1; 442 } 443 } 444 445 return optind; 446 } 447 448 static int 449 parse_ls_flags(char **argv, int argc, int *lflag) 450 { 451 extern int opterr, optind, optopt, optreset; 452 int ch; 453 454 optind = optreset = 1; 455 opterr = 0; 456 457 *lflag = LS_NAME_SORT; 458 while ((ch = getopt(argc, argv, "1Safhlnrt")) != -1) { 459 switch (ch) { 460 case '1': 461 *lflag &= ~VIEW_FLAGS; 462 *lflag |= LS_SHORT_VIEW; 463 break; 464 case 'S': 465 *lflag &= ~SORT_FLAGS; 466 *lflag |= LS_SIZE_SORT; 467 break; 468 case 'a': 469 *lflag |= LS_SHOW_ALL; 470 break; 471 case 'f': 472 *lflag &= ~SORT_FLAGS; 473 break; 474 case 'h': 475 *lflag |= LS_SI_UNITS; 476 break; 477 case 'l': 478 *lflag &= ~LS_SHORT_VIEW; 479 *lflag |= LS_LONG_VIEW; 480 break; 481 case 'n': 482 *lflag &= ~LS_SHORT_VIEW; 483 *lflag |= LS_NUMERIC_VIEW|LS_LONG_VIEW; 484 break; 485 case 'r': 486 *lflag |= LS_REVERSE_SORT; 487 break; 488 case 't': 489 *lflag &= ~SORT_FLAGS; 490 *lflag |= LS_TIME_SORT; 491 break; 492 default: 493 error("ls: Invalid flag -%c", optopt); 494 return -1; 495 } 496 } 497 498 return optind; 499 } 500 501 static int 502 parse_df_flags(const char *cmd, char **argv, int argc, int *hflag, int *iflag) 503 { 504 extern int opterr, optind, optopt, optreset; 505 int ch; 506 507 optind = optreset = 1; 508 opterr = 0; 509 510 *hflag = *iflag = 0; 511 while ((ch = getopt(argc, argv, "hi")) != -1) { 512 switch (ch) { 513 case 'h': 514 *hflag = 1; 515 break; 516 case 'i': 517 *iflag = 1; 518 break; 519 default: 520 error("%s: Invalid flag -%c", cmd, optopt); 521 return -1; 522 } 523 } 524 525 return optind; 526 } 527 528 static int 529 parse_no_flags(const char *cmd, char **argv, int argc) 530 { 531 extern int opterr, optind, optopt, optreset; 532 int ch; 533 534 optind = optreset = 1; 535 opterr = 0; 536 537 while ((ch = getopt(argc, argv, "")) != -1) { 538 switch (ch) { 539 default: 540 error("%s: Invalid flag -%c", cmd, optopt); 541 return -1; 542 } 543 } 544 545 return optind; 546 } 547 548 static int 549 is_dir(char *path) 550 { 551 struct stat sb; 552 553 /* XXX: report errors? */ 554 if (stat(path, &sb) == -1) 555 return(0); 556 557 return(S_ISDIR(sb.st_mode)); 558 } 559 560 static int 561 remote_is_dir(struct sftp_conn *conn, char *path) 562 { 563 Attrib *a; 564 565 /* XXX: report errors? */ 566 if ((a = do_stat(conn, path, 1)) == NULL) 567 return(0); 568 if (!(a->flags & SSH2_FILEXFER_ATTR_PERMISSIONS)) 569 return(0); 570 return(S_ISDIR(a->perm)); 571 } 572 573 /* Check whether path returned from glob(..., GLOB_MARK, ...) is a directory */ 574 static int 575 pathname_is_dir(char *pathname) 576 { 577 size_t l = strlen(pathname); 578 579 return l > 0 && pathname[l - 1] == '/'; 580 } 581 582 static int 583 process_get(struct sftp_conn *conn, char *src, char *dst, char *pwd, 584 int pflag, int rflag, int resume, int fflag) 585 { 586 char *abs_src = NULL; 587 char *abs_dst = NULL; 588 glob_t g; 589 char *filename, *tmp=NULL; 590 int i, err = 0; 591 592 abs_src = xstrdup(src); 593 abs_src = make_absolute(abs_src, pwd); 594 memset(&g, 0, sizeof(g)); 595 596 debug3("Looking up %s", abs_src); 597 if (remote_glob(conn, abs_src, GLOB_MARK, NULL, &g)) { 598 error("File \"%s\" not found.", abs_src); 599 err = -1; 600 goto out; 601 } 602 603 /* 604 * If multiple matches then dst must be a directory or 605 * unspecified. 606 */ 607 if (g.gl_matchc > 1 && dst != NULL && !is_dir(dst)) { 608 error("Multiple source paths, but destination " 609 "\"%s\" is not a directory", dst); 610 err = -1; 611 goto out; 612 } 613 614 for (i = 0; g.gl_pathv[i] && !interrupted; i++) { 615 tmp = xstrdup(g.gl_pathv[i]); 616 if ((filename = basename(tmp)) == NULL) { 617 error("basename %s: %s", tmp, strerror(errno)); 618 free(tmp); 619 err = -1; 620 goto out; 621 } 622 623 if (g.gl_matchc == 1 && dst) { 624 if (is_dir(dst)) { 625 abs_dst = path_append(dst, filename); 626 } else { 627 abs_dst = xstrdup(dst); 628 } 629 } else if (dst) { 630 abs_dst = path_append(dst, filename); 631 } else { 632 abs_dst = xstrdup(filename); 633 } 634 free(tmp); 635 636 resume |= global_aflag; 637 if (!quiet && resume) 638 printf("Resuming %s to %s\n", g.gl_pathv[i], abs_dst); 639 else if (!quiet && !resume) 640 printf("Fetching %s to %s\n", g.gl_pathv[i], abs_dst); 641 if (pathname_is_dir(g.gl_pathv[i]) && (rflag || global_rflag)) { 642 if (download_dir(conn, g.gl_pathv[i], abs_dst, NULL, 643 pflag || global_pflag, 1, resume, 644 fflag || global_fflag) == -1) 645 err = -1; 646 } else { 647 if (do_download(conn, g.gl_pathv[i], abs_dst, NULL, 648 pflag || global_pflag, resume, 649 fflag || global_fflag) == -1) 650 err = -1; 651 } 652 free(abs_dst); 653 abs_dst = NULL; 654 } 655 656 out: 657 free(abs_src); 658 globfree(&g); 659 return(err); 660 } 661 662 static int 663 process_put(struct sftp_conn *conn, char *src, char *dst, char *pwd, 664 int pflag, int rflag, int fflag) 665 { 666 char *tmp_dst = NULL; 667 char *abs_dst = NULL; 668 char *tmp = NULL, *filename = NULL; 669 glob_t g; 670 int err = 0; 671 int i, dst_is_dir = 1; 672 struct stat sb; 673 674 if (dst) { 675 tmp_dst = xstrdup(dst); 676 tmp_dst = make_absolute(tmp_dst, pwd); 677 } 678 679 memset(&g, 0, sizeof(g)); 680 debug3("Looking up %s", src); 681 if (glob(src, GLOB_NOCHECK | GLOB_MARK, NULL, &g)) { 682 error("File \"%s\" not found.", src); 683 err = -1; 684 goto out; 685 } 686 687 /* If we aren't fetching to pwd then stash this status for later */ 688 if (tmp_dst != NULL) 689 dst_is_dir = remote_is_dir(conn, tmp_dst); 690 691 /* If multiple matches, dst may be directory or unspecified */ 692 if (g.gl_matchc > 1 && tmp_dst && !dst_is_dir) { 693 error("Multiple paths match, but destination " 694 "\"%s\" is not a directory", tmp_dst); 695 err = -1; 696 goto out; 697 } 698 699 for (i = 0; g.gl_pathv[i] && !interrupted; i++) { 700 if (stat(g.gl_pathv[i], &sb) == -1) { 701 err = -1; 702 error("stat %s: %s", g.gl_pathv[i], strerror(errno)); 703 continue; 704 } 705 706 tmp = xstrdup(g.gl_pathv[i]); 707 if ((filename = basename(tmp)) == NULL) { 708 error("basename %s: %s", tmp, strerror(errno)); 709 free(tmp); 710 err = -1; 711 goto out; 712 } 713 714 if (g.gl_matchc == 1 && tmp_dst) { 715 /* If directory specified, append filename */ 716 if (dst_is_dir) 717 abs_dst = path_append(tmp_dst, filename); 718 else 719 abs_dst = xstrdup(tmp_dst); 720 } else if (tmp_dst) { 721 abs_dst = path_append(tmp_dst, filename); 722 } else { 723 abs_dst = make_absolute(xstrdup(filename), pwd); 724 } 725 free(tmp); 726 727 if (!quiet) 728 printf("Uploading %s to %s\n", g.gl_pathv[i], abs_dst); 729 if (pathname_is_dir(g.gl_pathv[i]) && (rflag || global_rflag)) { 730 if (upload_dir(conn, g.gl_pathv[i], abs_dst, 731 pflag || global_pflag, 1, 732 fflag || global_fflag) == -1) 733 err = -1; 734 } else { 735 if (do_upload(conn, g.gl_pathv[i], abs_dst, 736 pflag || global_pflag, 737 fflag || global_fflag) == -1) 738 err = -1; 739 } 740 } 741 742 out: 743 free(abs_dst); 744 free(tmp_dst); 745 globfree(&g); 746 return(err); 747 } 748 749 static int 750 sdirent_comp(const void *aa, const void *bb) 751 { 752 SFTP_DIRENT *a = *(SFTP_DIRENT **)aa; 753 SFTP_DIRENT *b = *(SFTP_DIRENT **)bb; 754 int rmul = sort_flag & LS_REVERSE_SORT ? -1 : 1; 755 756 #define NCMP(a,b) (a == b ? 0 : (a < b ? 1 : -1)) 757 if (sort_flag & LS_NAME_SORT) 758 return (rmul * strcmp(a->filename, b->filename)); 759 else if (sort_flag & LS_TIME_SORT) 760 return (rmul * NCMP(a->a.mtime, b->a.mtime)); 761 else if (sort_flag & LS_SIZE_SORT) 762 return (rmul * NCMP(a->a.size, b->a.size)); 763 764 fatal("Unknown ls sort type"); 765 } 766 767 /* sftp ls.1 replacement for directories */ 768 static int 769 do_ls_dir(struct sftp_conn *conn, char *path, char *strip_path, int lflag) 770 { 771 int n; 772 u_int c = 1, colspace = 0, columns = 1; 773 SFTP_DIRENT **d; 774 775 if ((n = do_readdir(conn, path, &d)) != 0) 776 return (n); 777 778 if (!(lflag & LS_SHORT_VIEW)) { 779 u_int m = 0, width = 80; 780 struct winsize ws; 781 char *tmp; 782 783 /* Count entries for sort and find longest filename */ 784 for (n = 0; d[n] != NULL; n++) { 785 if (d[n]->filename[0] != '.' || (lflag & LS_SHOW_ALL)) 786 m = MAX(m, strlen(d[n]->filename)); 787 } 788 789 /* Add any subpath that also needs to be counted */ 790 tmp = path_strip(path, strip_path); 791 m += strlen(tmp); 792 free(tmp); 793 794 if (ioctl(fileno(stdin), TIOCGWINSZ, &ws) != -1) 795 width = ws.ws_col; 796 797 columns = width / (m + 2); 798 columns = MAX(columns, 1); 799 colspace = width / columns; 800 colspace = MIN(colspace, width); 801 } 802 803 if (lflag & SORT_FLAGS) { 804 for (n = 0; d[n] != NULL; n++) 805 ; /* count entries */ 806 sort_flag = lflag & (SORT_FLAGS|LS_REVERSE_SORT); 807 qsort(d, n, sizeof(*d), sdirent_comp); 808 } 809 810 for (n = 0; d[n] != NULL && !interrupted; n++) { 811 char *tmp, *fname; 812 813 if (d[n]->filename[0] == '.' && !(lflag & LS_SHOW_ALL)) 814 continue; 815 816 tmp = path_append(path, d[n]->filename); 817 fname = path_strip(tmp, strip_path); 818 free(tmp); 819 820 if (lflag & LS_LONG_VIEW) { 821 if (lflag & (LS_NUMERIC_VIEW|LS_SI_UNITS)) { 822 char *lname; 823 struct stat sb; 824 825 memset(&sb, 0, sizeof(sb)); 826 attrib_to_stat(&d[n]->a, &sb); 827 lname = ls_file(fname, &sb, 1, 828 (lflag & LS_SI_UNITS)); 829 printf("%s\n", lname); 830 free(lname); 831 } else 832 printf("%s\n", d[n]->longname); 833 } else { 834 printf("%-*s", colspace, fname); 835 if (c >= columns) { 836 printf("\n"); 837 c = 1; 838 } else 839 c++; 840 } 841 842 free(fname); 843 } 844 845 if (!(lflag & LS_LONG_VIEW) && (c != 1)) 846 printf("\n"); 847 848 free_sftp_dirents(d); 849 return (0); 850 } 851 852 /* sftp ls.1 replacement which handles path globs */ 853 static int 854 do_globbed_ls(struct sftp_conn *conn, char *path, char *strip_path, 855 int lflag) 856 { 857 char *fname, *lname; 858 glob_t g; 859 int err; 860 struct winsize ws; 861 u_int i, c = 1, colspace = 0, columns = 1, m = 0, width = 80; 862 863 memset(&g, 0, sizeof(g)); 864 865 if (remote_glob(conn, path, 866 GLOB_MARK|GLOB_NOCHECK|GLOB_BRACE|GLOB_KEEPSTAT|GLOB_NOSORT, 867 NULL, &g) || 868 (g.gl_pathc && !g.gl_matchc)) { 869 if (g.gl_pathc) 870 globfree(&g); 871 error("Can't ls: \"%s\" not found", path); 872 return -1; 873 } 874 875 if (interrupted) 876 goto out; 877 878 /* 879 * If the glob returns a single match and it is a directory, 880 * then just list its contents. 881 */ 882 if (g.gl_matchc == 1 && g.gl_statv[0] != NULL && 883 S_ISDIR(g.gl_statv[0]->st_mode)) { 884 err = do_ls_dir(conn, g.gl_pathv[0], strip_path, lflag); 885 globfree(&g); 886 return err; 887 } 888 889 if (ioctl(fileno(stdin), TIOCGWINSZ, &ws) != -1) 890 width = ws.ws_col; 891 892 if (!(lflag & LS_SHORT_VIEW)) { 893 /* Count entries for sort and find longest filename */ 894 for (i = 0; g.gl_pathv[i]; i++) 895 m = MAX(m, strlen(g.gl_pathv[i])); 896 897 columns = width / (m + 2); 898 columns = MAX(columns, 1); 899 colspace = width / columns; 900 } 901 902 for (i = 0; g.gl_pathv[i] && !interrupted; i++) { 903 fname = path_strip(g.gl_pathv[i], strip_path); 904 if (lflag & LS_LONG_VIEW) { 905 if (g.gl_statv[i] == NULL) { 906 error("no stat information for %s", fname); 907 continue; 908 } 909 lname = ls_file(fname, g.gl_statv[i], 1, 910 (lflag & LS_SI_UNITS)); 911 printf("%s\n", lname); 912 free(lname); 913 } else { 914 printf("%-*s", colspace, fname); 915 if (c >= columns) { 916 printf("\n"); 917 c = 1; 918 } else 919 c++; 920 } 921 free(fname); 922 } 923 924 if (!(lflag & LS_LONG_VIEW) && (c != 1)) 925 printf("\n"); 926 927 out: 928 if (g.gl_pathc) 929 globfree(&g); 930 931 return 0; 932 } 933 934 static int 935 do_df(struct sftp_conn *conn, char *path, int hflag, int iflag) 936 { 937 struct sftp_statvfs st; 938 char s_used[FMT_SCALED_STRSIZE]; 939 char s_avail[FMT_SCALED_STRSIZE]; 940 char s_root[FMT_SCALED_STRSIZE]; 941 char s_total[FMT_SCALED_STRSIZE]; 942 unsigned long long ffree; 943 944 if (do_statvfs(conn, path, &st, 1) == -1) 945 return -1; 946 if (iflag) { 947 ffree = st.f_files ? (100 * (st.f_files - st.f_ffree) / st.f_files) : 0; 948 printf(" Inodes Used Avail " 949 "(root) %%Capacity\n"); 950 printf("%11llu %11llu %11llu %11llu %3llu%%\n", 951 (unsigned long long)st.f_files, 952 (unsigned long long)(st.f_files - st.f_ffree), 953 (unsigned long long)st.f_favail, 954 (unsigned long long)st.f_ffree, ffree); 955 } else if (hflag) { 956 strlcpy(s_used, "error", sizeof(s_used)); 957 strlcpy(s_avail, "error", sizeof(s_avail)); 958 strlcpy(s_root, "error", sizeof(s_root)); 959 strlcpy(s_total, "error", sizeof(s_total)); 960 fmt_scaled((st.f_blocks - st.f_bfree) * st.f_frsize, s_used); 961 fmt_scaled(st.f_bavail * st.f_frsize, s_avail); 962 fmt_scaled(st.f_bfree * st.f_frsize, s_root); 963 fmt_scaled(st.f_blocks * st.f_frsize, s_total); 964 printf(" Size Used Avail (root) %%Capacity\n"); 965 printf("%7sB %7sB %7sB %7sB %3llu%%\n", 966 s_total, s_used, s_avail, s_root, 967 (unsigned long long)(100 * (st.f_blocks - st.f_bfree) / 968 st.f_blocks)); 969 } else { 970 printf(" Size Used Avail " 971 "(root) %%Capacity\n"); 972 printf("%12llu %12llu %12llu %12llu %3llu%%\n", 973 (unsigned long long)(st.f_frsize * st.f_blocks / 1024), 974 (unsigned long long)(st.f_frsize * 975 (st.f_blocks - st.f_bfree) / 1024), 976 (unsigned long long)(st.f_frsize * st.f_bavail / 1024), 977 (unsigned long long)(st.f_frsize * st.f_bfree / 1024), 978 (unsigned long long)(100 * (st.f_blocks - st.f_bfree) / 979 st.f_blocks)); 980 } 981 return 0; 982 } 983 984 /* 985 * Undo escaping of glob sequences in place. Used to undo extra escaping 986 * applied in makeargv() when the string is destined for a function that 987 * does not glob it. 988 */ 989 static void 990 undo_glob_escape(char *s) 991 { 992 size_t i, j; 993 994 for (i = j = 0;;) { 995 if (s[i] == '\0') { 996 s[j] = '\0'; 997 return; 998 } 999 if (s[i] != '\\') { 1000 s[j++] = s[i++]; 1001 continue; 1002 } 1003 /* s[i] == '\\' */ 1004 ++i; 1005 switch (s[i]) { 1006 case '?': 1007 case '[': 1008 case '*': 1009 case '\\': 1010 s[j++] = s[i++]; 1011 break; 1012 case '\0': 1013 s[j++] = '\\'; 1014 s[j] = '\0'; 1015 return; 1016 default: 1017 s[j++] = '\\'; 1018 s[j++] = s[i++]; 1019 break; 1020 } 1021 } 1022 } 1023 1024 /* 1025 * Split a string into an argument vector using sh(1)-style quoting, 1026 * comment and escaping rules, but with some tweaks to handle glob(3) 1027 * wildcards. 1028 * The "sloppy" flag allows for recovery from missing terminating quote, for 1029 * use in parsing incomplete commandlines during tab autocompletion. 1030 * 1031 * Returns NULL on error or a NULL-terminated array of arguments. 1032 * 1033 * If "lastquote" is not NULL, the quoting character used for the last 1034 * argument is placed in *lastquote ("\0", "'" or "\""). 1035 * 1036 * If "terminated" is not NULL, *terminated will be set to 1 when the 1037 * last argument's quote has been properly terminated or 0 otherwise. 1038 * This parameter is only of use if "sloppy" is set. 1039 */ 1040 #define MAXARGS 128 1041 #define MAXARGLEN 8192 1042 static char ** 1043 makeargv(const char *arg, int *argcp, int sloppy, char *lastquote, 1044 u_int *terminated) 1045 { 1046 int argc, quot; 1047 size_t i, j; 1048 static char argvs[MAXARGLEN]; 1049 static char *argv[MAXARGS + 1]; 1050 enum { MA_START, MA_SQUOTE, MA_DQUOTE, MA_UNQUOTED } state, q; 1051 1052 *argcp = argc = 0; 1053 if (strlen(arg) > sizeof(argvs) - 1) { 1054 args_too_longs: 1055 error("string too long"); 1056 return NULL; 1057 } 1058 if (terminated != NULL) 1059 *terminated = 1; 1060 if (lastquote != NULL) 1061 *lastquote = '\0'; 1062 state = MA_START; 1063 i = j = 0; 1064 for (;;) { 1065 if ((size_t)argc >= sizeof(argv) / sizeof(*argv)){ 1066 error("Too many arguments."); 1067 return NULL; 1068 } 1069 if (isspace((unsigned char)arg[i])) { 1070 if (state == MA_UNQUOTED) { 1071 /* Terminate current argument */ 1072 argvs[j++] = '\0'; 1073 argc++; 1074 state = MA_START; 1075 } else if (state != MA_START) 1076 argvs[j++] = arg[i]; 1077 } else if (arg[i] == '"' || arg[i] == '\'') { 1078 q = arg[i] == '"' ? MA_DQUOTE : MA_SQUOTE; 1079 if (state == MA_START) { 1080 argv[argc] = argvs + j; 1081 state = q; 1082 if (lastquote != NULL) 1083 *lastquote = arg[i]; 1084 } else if (state == MA_UNQUOTED) 1085 state = q; 1086 else if (state == q) 1087 state = MA_UNQUOTED; 1088 else 1089 argvs[j++] = arg[i]; 1090 } else if (arg[i] == '\\') { 1091 if (state == MA_SQUOTE || state == MA_DQUOTE) { 1092 quot = state == MA_SQUOTE ? '\'' : '"'; 1093 /* Unescape quote we are in */ 1094 /* XXX support \n and friends? */ 1095 if (arg[i + 1] == quot) { 1096 i++; 1097 argvs[j++] = arg[i]; 1098 } else if (arg[i + 1] == '?' || 1099 arg[i + 1] == '[' || arg[i + 1] == '*') { 1100 /* 1101 * Special case for sftp: append 1102 * double-escaped glob sequence - 1103 * glob will undo one level of 1104 * escaping. NB. string can grow here. 1105 */ 1106 if (j >= sizeof(argvs) - 5) 1107 goto args_too_longs; 1108 argvs[j++] = '\\'; 1109 argvs[j++] = arg[i++]; 1110 argvs[j++] = '\\'; 1111 argvs[j++] = arg[i]; 1112 } else { 1113 argvs[j++] = arg[i++]; 1114 argvs[j++] = arg[i]; 1115 } 1116 } else { 1117 if (state == MA_START) { 1118 argv[argc] = argvs + j; 1119 state = MA_UNQUOTED; 1120 if (lastquote != NULL) 1121 *lastquote = '\0'; 1122 } 1123 if (arg[i + 1] == '?' || arg[i + 1] == '[' || 1124 arg[i + 1] == '*' || arg[i + 1] == '\\') { 1125 /* 1126 * Special case for sftp: append 1127 * escaped glob sequence - 1128 * glob will undo one level of 1129 * escaping. 1130 */ 1131 argvs[j++] = arg[i++]; 1132 argvs[j++] = arg[i]; 1133 } else { 1134 /* Unescape everything */ 1135 /* XXX support \n and friends? */ 1136 i++; 1137 argvs[j++] = arg[i]; 1138 } 1139 } 1140 } else if (arg[i] == '#') { 1141 if (state == MA_SQUOTE || state == MA_DQUOTE) 1142 argvs[j++] = arg[i]; 1143 else 1144 goto string_done; 1145 } else if (arg[i] == '\0') { 1146 if (state == MA_SQUOTE || state == MA_DQUOTE) { 1147 if (sloppy) { 1148 state = MA_UNQUOTED; 1149 if (terminated != NULL) 1150 *terminated = 0; 1151 goto string_done; 1152 } 1153 error("Unterminated quoted argument"); 1154 return NULL; 1155 } 1156 string_done: 1157 if (state == MA_UNQUOTED) { 1158 argvs[j++] = '\0'; 1159 argc++; 1160 } 1161 break; 1162 } else { 1163 if (state == MA_START) { 1164 argv[argc] = argvs + j; 1165 state = MA_UNQUOTED; 1166 if (lastquote != NULL) 1167 *lastquote = '\0'; 1168 } 1169 if ((state == MA_SQUOTE || state == MA_DQUOTE) && 1170 (arg[i] == '?' || arg[i] == '[' || arg[i] == '*')) { 1171 /* 1172 * Special case for sftp: escape quoted 1173 * glob(3) wildcards. NB. string can grow 1174 * here. 1175 */ 1176 if (j >= sizeof(argvs) - 3) 1177 goto args_too_longs; 1178 argvs[j++] = '\\'; 1179 argvs[j++] = arg[i]; 1180 } else 1181 argvs[j++] = arg[i]; 1182 } 1183 i++; 1184 } 1185 *argcp = argc; 1186 return argv; 1187 } 1188 1189 static int 1190 parse_args(const char **cpp, int *ignore_errors, int *aflag, int *fflag, 1191 int *hflag, int *iflag, int *lflag, int *pflag, int *rflag, int *sflag, 1192 unsigned long *n_arg, char **path1, char **path2) 1193 { 1194 const char *cmd, *cp = *cpp; 1195 char *cp2, **argv; 1196 int base = 0; 1197 long l; 1198 int i, cmdnum, optidx, argc; 1199 1200 /* Skip leading whitespace */ 1201 cp = cp + strspn(cp, WHITESPACE); 1202 1203 /* Check for leading '-' (disable error processing) */ 1204 *ignore_errors = 0; 1205 if (*cp == '-') { 1206 *ignore_errors = 1; 1207 cp++; 1208 cp = cp + strspn(cp, WHITESPACE); 1209 } 1210 1211 /* Ignore blank lines and lines which begin with comment '#' char */ 1212 if (*cp == '\0' || *cp == '#') 1213 return (0); 1214 1215 if ((argv = makeargv(cp, &argc, 0, NULL, NULL)) == NULL) 1216 return -1; 1217 1218 /* Figure out which command we have */ 1219 for (i = 0; cmds[i].c != NULL; i++) { 1220 if (argv[0] != NULL && strcasecmp(cmds[i].c, argv[0]) == 0) 1221 break; 1222 } 1223 cmdnum = cmds[i].n; 1224 cmd = cmds[i].c; 1225 1226 /* Special case */ 1227 if (*cp == '!') { 1228 cp++; 1229 cmdnum = I_SHELL; 1230 } else if (cmdnum == -1) { 1231 error("Invalid command."); 1232 return -1; 1233 } 1234 1235 /* Get arguments and parse flags */ 1236 *aflag = *fflag = *hflag = *iflag = *lflag = *pflag = 0; 1237 *rflag = *sflag = 0; 1238 *path1 = *path2 = NULL; 1239 optidx = 1; 1240 switch (cmdnum) { 1241 case I_GET: 1242 case I_REGET: 1243 case I_PUT: 1244 if ((optidx = parse_getput_flags(cmd, argv, argc, 1245 aflag, fflag, pflag, rflag)) == -1) 1246 return -1; 1247 /* Get first pathname (mandatory) */ 1248 if (argc - optidx < 1) { 1249 error("You must specify at least one path after a " 1250 "%s command.", cmd); 1251 return -1; 1252 } 1253 *path1 = xstrdup(argv[optidx]); 1254 /* Get second pathname (optional) */ 1255 if (argc - optidx > 1) { 1256 *path2 = xstrdup(argv[optidx + 1]); 1257 /* Destination is not globbed */ 1258 undo_glob_escape(*path2); 1259 } 1260 if (*aflag && cmdnum == I_PUT) { 1261 /* XXX implement resume for uploads */ 1262 error("Resume is not supported for uploads"); 1263 return -1; 1264 } 1265 break; 1266 case I_LINK: 1267 if ((optidx = parse_link_flags(cmd, argv, argc, sflag)) == -1) 1268 return -1; 1269 goto parse_two_paths; 1270 case I_RENAME: 1271 if ((optidx = parse_rename_flags(cmd, argv, argc, lflag)) == -1) 1272 return -1; 1273 goto parse_two_paths; 1274 case I_SYMLINK: 1275 if ((optidx = parse_no_flags(cmd, argv, argc)) == -1) 1276 return -1; 1277 parse_two_paths: 1278 if (argc - optidx < 2) { 1279 error("You must specify two paths after a %s " 1280 "command.", cmd); 1281 return -1; 1282 } 1283 *path1 = xstrdup(argv[optidx]); 1284 *path2 = xstrdup(argv[optidx + 1]); 1285 /* Paths are not globbed */ 1286 undo_glob_escape(*path1); 1287 undo_glob_escape(*path2); 1288 break; 1289 case I_RM: 1290 case I_MKDIR: 1291 case I_RMDIR: 1292 case I_CHDIR: 1293 case I_LCHDIR: 1294 case I_LMKDIR: 1295 if ((optidx = parse_no_flags(cmd, argv, argc)) == -1) 1296 return -1; 1297 /* Get pathname (mandatory) */ 1298 if (argc - optidx < 1) { 1299 error("You must specify a path after a %s command.", 1300 cmd); 1301 return -1; 1302 } 1303 *path1 = xstrdup(argv[optidx]); 1304 /* Only "rm" globs */ 1305 if (cmdnum != I_RM) 1306 undo_glob_escape(*path1); 1307 break; 1308 case I_DF: 1309 if ((optidx = parse_df_flags(cmd, argv, argc, hflag, 1310 iflag)) == -1) 1311 return -1; 1312 /* Default to current directory if no path specified */ 1313 if (argc - optidx < 1) 1314 *path1 = NULL; 1315 else { 1316 *path1 = xstrdup(argv[optidx]); 1317 undo_glob_escape(*path1); 1318 } 1319 break; 1320 case I_LS: 1321 if ((optidx = parse_ls_flags(argv, argc, lflag)) == -1) 1322 return(-1); 1323 /* Path is optional */ 1324 if (argc - optidx > 0) 1325 *path1 = xstrdup(argv[optidx]); 1326 break; 1327 case I_LLS: 1328 /* Skip ls command and following whitespace */ 1329 cp = cp + strlen(cmd) + strspn(cp, WHITESPACE); 1330 case I_SHELL: 1331 /* Uses the rest of the line */ 1332 break; 1333 case I_LUMASK: 1334 case I_CHMOD: 1335 base = 8; 1336 case I_CHOWN: 1337 case I_CHGRP: 1338 if ((optidx = parse_no_flags(cmd, argv, argc)) == -1) 1339 return -1; 1340 /* Get numeric arg (mandatory) */ 1341 if (argc - optidx < 1) 1342 goto need_num_arg; 1343 errno = 0; 1344 l = strtol(argv[optidx], &cp2, base); 1345 if (cp2 == argv[optidx] || *cp2 != '\0' || 1346 ((l == LONG_MIN || l == LONG_MAX) && errno == ERANGE) || 1347 l < 0) { 1348 need_num_arg: 1349 error("You must supply a numeric argument " 1350 "to the %s command.", cmd); 1351 return -1; 1352 } 1353 *n_arg = l; 1354 if (cmdnum == I_LUMASK) 1355 break; 1356 /* Get pathname (mandatory) */ 1357 if (argc - optidx < 2) { 1358 error("You must specify a path after a %s command.", 1359 cmd); 1360 return -1; 1361 } 1362 *path1 = xstrdup(argv[optidx + 1]); 1363 break; 1364 case I_QUIT: 1365 case I_PWD: 1366 case I_LPWD: 1367 case I_HELP: 1368 case I_VERSION: 1369 case I_PROGRESS: 1370 if ((optidx = parse_no_flags(cmd, argv, argc)) == -1) 1371 return -1; 1372 break; 1373 default: 1374 fatal("Command not implemented"); 1375 } 1376 1377 *cpp = cp; 1378 return(cmdnum); 1379 } 1380 1381 static int 1382 parse_dispatch_command(struct sftp_conn *conn, const char *cmd, char **pwd, 1383 int err_abort) 1384 { 1385 char *path1, *path2, *tmp; 1386 int ignore_errors = 0, aflag = 0, fflag = 0, hflag = 0, iflag = 0; 1387 int lflag = 0, pflag = 0, rflag = 0, sflag = 0; 1388 int cmdnum, i; 1389 unsigned long n_arg = 0; 1390 Attrib a, *aa; 1391 char path_buf[MAXPATHLEN]; 1392 int err = 0; 1393 glob_t g; 1394 1395 path1 = path2 = NULL; 1396 cmdnum = parse_args(&cmd, &ignore_errors, &aflag, &fflag, &hflag, 1397 &iflag, &lflag, &pflag, &rflag, &sflag, &n_arg, &path1, &path2); 1398 if (ignore_errors != 0) 1399 err_abort = 0; 1400 1401 memset(&g, 0, sizeof(g)); 1402 1403 /* Perform command */ 1404 switch (cmdnum) { 1405 case 0: 1406 /* Blank line */ 1407 break; 1408 case -1: 1409 /* Unrecognized command */ 1410 err = -1; 1411 break; 1412 case I_REGET: 1413 aflag = 1; 1414 /* FALLTHROUGH */ 1415 case I_GET: 1416 err = process_get(conn, path1, path2, *pwd, pflag, 1417 rflag, aflag, fflag); 1418 break; 1419 case I_PUT: 1420 err = process_put(conn, path1, path2, *pwd, pflag, 1421 rflag, fflag); 1422 break; 1423 case I_RENAME: 1424 path1 = make_absolute(path1, *pwd); 1425 path2 = make_absolute(path2, *pwd); 1426 err = do_rename(conn, path1, path2, lflag); 1427 break; 1428 case I_SYMLINK: 1429 sflag = 1; 1430 case I_LINK: 1431 if (!sflag) 1432 path1 = make_absolute(path1, *pwd); 1433 path2 = make_absolute(path2, *pwd); 1434 err = (sflag ? do_symlink : do_hardlink)(conn, path1, path2); 1435 break; 1436 case I_RM: 1437 path1 = make_absolute(path1, *pwd); 1438 remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g); 1439 for (i = 0; g.gl_pathv[i] && !interrupted; i++) { 1440 if (!quiet) 1441 printf("Removing %s\n", g.gl_pathv[i]); 1442 err = do_rm(conn, g.gl_pathv[i]); 1443 if (err != 0 && err_abort) 1444 break; 1445 } 1446 break; 1447 case I_MKDIR: 1448 path1 = make_absolute(path1, *pwd); 1449 attrib_clear(&a); 1450 a.flags |= SSH2_FILEXFER_ATTR_PERMISSIONS; 1451 a.perm = 0777; 1452 err = do_mkdir(conn, path1, &a, 1); 1453 break; 1454 case I_RMDIR: 1455 path1 = make_absolute(path1, *pwd); 1456 err = do_rmdir(conn, path1); 1457 break; 1458 case I_CHDIR: 1459 path1 = make_absolute(path1, *pwd); 1460 if ((tmp = do_realpath(conn, path1)) == NULL) { 1461 err = 1; 1462 break; 1463 } 1464 if ((aa = do_stat(conn, tmp, 0)) == NULL) { 1465 free(tmp); 1466 err = 1; 1467 break; 1468 } 1469 if (!(aa->flags & SSH2_FILEXFER_ATTR_PERMISSIONS)) { 1470 error("Can't change directory: Can't check target"); 1471 free(tmp); 1472 err = 1; 1473 break; 1474 } 1475 if (!S_ISDIR(aa->perm)) { 1476 error("Can't change directory: \"%s\" is not " 1477 "a directory", tmp); 1478 free(tmp); 1479 err = 1; 1480 break; 1481 } 1482 free(*pwd); 1483 *pwd = tmp; 1484 break; 1485 case I_LS: 1486 if (!path1) { 1487 do_ls_dir(conn, *pwd, *pwd, lflag); 1488 break; 1489 } 1490 1491 /* Strip pwd off beginning of non-absolute paths */ 1492 tmp = NULL; 1493 if (*path1 != '/') 1494 tmp = *pwd; 1495 1496 path1 = make_absolute(path1, *pwd); 1497 err = do_globbed_ls(conn, path1, tmp, lflag); 1498 break; 1499 case I_DF: 1500 /* Default to current directory if no path specified */ 1501 if (path1 == NULL) 1502 path1 = xstrdup(*pwd); 1503 path1 = make_absolute(path1, *pwd); 1504 err = do_df(conn, path1, hflag, iflag); 1505 break; 1506 case I_LCHDIR: 1507 if (chdir(path1) == -1) { 1508 error("Couldn't change local directory to " 1509 "\"%s\": %s", path1, strerror(errno)); 1510 err = 1; 1511 } 1512 break; 1513 case I_LMKDIR: 1514 if (mkdir(path1, 0777) == -1) { 1515 error("Couldn't create local directory " 1516 "\"%s\": %s", path1, strerror(errno)); 1517 err = 1; 1518 } 1519 break; 1520 case I_LLS: 1521 local_do_ls(cmd); 1522 break; 1523 case I_SHELL: 1524 local_do_shell(cmd); 1525 break; 1526 case I_LUMASK: 1527 umask(n_arg); 1528 printf("Local umask: %03lo\n", n_arg); 1529 break; 1530 case I_CHMOD: 1531 path1 = make_absolute(path1, *pwd); 1532 attrib_clear(&a); 1533 a.flags |= SSH2_FILEXFER_ATTR_PERMISSIONS; 1534 a.perm = n_arg; 1535 remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g); 1536 for (i = 0; g.gl_pathv[i] && !interrupted; i++) { 1537 if (!quiet) 1538 printf("Changing mode on %s\n", g.gl_pathv[i]); 1539 err = do_setstat(conn, g.gl_pathv[i], &a); 1540 if (err != 0 && err_abort) 1541 break; 1542 } 1543 break; 1544 case I_CHOWN: 1545 case I_CHGRP: 1546 path1 = make_absolute(path1, *pwd); 1547 remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g); 1548 for (i = 0; g.gl_pathv[i] && !interrupted; i++) { 1549 if (!(aa = do_stat(conn, g.gl_pathv[i], 0))) { 1550 if (err_abort) { 1551 err = -1; 1552 break; 1553 } else 1554 continue; 1555 } 1556 if (!(aa->flags & SSH2_FILEXFER_ATTR_UIDGID)) { 1557 error("Can't get current ownership of " 1558 "remote file \"%s\"", g.gl_pathv[i]); 1559 if (err_abort) { 1560 err = -1; 1561 break; 1562 } else 1563 continue; 1564 } 1565 aa->flags &= SSH2_FILEXFER_ATTR_UIDGID; 1566 if (cmdnum == I_CHOWN) { 1567 if (!quiet) 1568 printf("Changing owner on %s\n", 1569 g.gl_pathv[i]); 1570 aa->uid = n_arg; 1571 } else { 1572 if (!quiet) 1573 printf("Changing group on %s\n", 1574 g.gl_pathv[i]); 1575 aa->gid = n_arg; 1576 } 1577 err = do_setstat(conn, g.gl_pathv[i], aa); 1578 if (err != 0 && err_abort) 1579 break; 1580 } 1581 break; 1582 case I_PWD: 1583 printf("Remote working directory: %s\n", *pwd); 1584 break; 1585 case I_LPWD: 1586 if (!getcwd(path_buf, sizeof(path_buf))) { 1587 error("Couldn't get local cwd: %s", strerror(errno)); 1588 err = -1; 1589 break; 1590 } 1591 printf("Local working directory: %s\n", path_buf); 1592 break; 1593 case I_QUIT: 1594 /* Processed below */ 1595 break; 1596 case I_HELP: 1597 help(); 1598 break; 1599 case I_VERSION: 1600 printf("SFTP protocol version %u\n", sftp_proto_version(conn)); 1601 break; 1602 case I_PROGRESS: 1603 showprogress = !showprogress; 1604 if (showprogress) 1605 printf("Progress meter enabled\n"); 1606 else 1607 printf("Progress meter disabled\n"); 1608 break; 1609 default: 1610 fatal("%d is not implemented", cmdnum); 1611 } 1612 1613 if (g.gl_pathc) 1614 globfree(&g); 1615 free(path1); 1616 free(path2); 1617 1618 /* If an unignored error occurs in batch mode we should abort. */ 1619 if (err_abort && err != 0) 1620 return (-1); 1621 else if (cmdnum == I_QUIT) 1622 return (1); 1623 1624 return (0); 1625 } 1626 1627 #ifdef USE_LIBEDIT 1628 static char * 1629 prompt(EditLine *el) 1630 { 1631 return ("sftp> "); 1632 } 1633 1634 /* Display entries in 'list' after skipping the first 'len' chars */ 1635 static void 1636 complete_display(char **list, u_int len) 1637 { 1638 u_int y, m = 0, width = 80, columns = 1, colspace = 0, llen; 1639 struct winsize ws; 1640 char *tmp; 1641 1642 /* Count entries for sort and find longest */ 1643 for (y = 0; list[y]; y++) 1644 m = MAX(m, strlen(list[y])); 1645 1646 if (ioctl(fileno(stdin), TIOCGWINSZ, &ws) != -1) 1647 width = ws.ws_col; 1648 1649 m = m > len ? m - len : 0; 1650 columns = width / (m + 2); 1651 columns = MAX(columns, 1); 1652 colspace = width / columns; 1653 colspace = MIN(colspace, width); 1654 1655 printf("\n"); 1656 m = 1; 1657 for (y = 0; list[y]; y++) { 1658 llen = strlen(list[y]); 1659 tmp = llen > len ? list[y] + len : ""; 1660 printf("%-*s", colspace, tmp); 1661 if (m >= columns) { 1662 printf("\n"); 1663 m = 1; 1664 } else 1665 m++; 1666 } 1667 printf("\n"); 1668 } 1669 1670 /* 1671 * Given a "list" of words that begin with a common prefix of "word", 1672 * attempt to find an autocompletion to extends "word" by the next 1673 * characters common to all entries in "list". 1674 */ 1675 static char * 1676 complete_ambiguous(const char *word, char **list, size_t count) 1677 { 1678 if (word == NULL) 1679 return NULL; 1680 1681 if (count > 0) { 1682 u_int y, matchlen = strlen(list[0]); 1683 1684 /* Find length of common stem */ 1685 for (y = 1; list[y]; y++) { 1686 u_int x; 1687 1688 for (x = 0; x < matchlen; x++) 1689 if (list[0][x] != list[y][x]) 1690 break; 1691 1692 matchlen = x; 1693 } 1694 1695 if (matchlen > strlen(word)) { 1696 char *tmp = xstrdup(list[0]); 1697 1698 tmp[matchlen] = '\0'; 1699 return tmp; 1700 } 1701 } 1702 1703 return xstrdup(word); 1704 } 1705 1706 /* Autocomplete a sftp command */ 1707 static int 1708 complete_cmd_parse(EditLine *el, char *cmd, int lastarg, char quote, 1709 int terminated) 1710 { 1711 u_int y, count = 0, cmdlen, tmplen; 1712 char *tmp, **list, argterm[3]; 1713 const LineInfo *lf; 1714 1715 list = xcalloc((sizeof(cmds) / sizeof(*cmds)) + 1, sizeof(char *)); 1716 1717 /* No command specified: display all available commands */ 1718 if (cmd == NULL) { 1719 for (y = 0; cmds[y].c; y++) 1720 list[count++] = xstrdup(cmds[y].c); 1721 1722 list[count] = NULL; 1723 complete_display(list, 0); 1724 1725 for (y = 0; list[y] != NULL; y++) 1726 free(list[y]); 1727 free(list); 1728 return count; 1729 } 1730 1731 /* Prepare subset of commands that start with "cmd" */ 1732 cmdlen = strlen(cmd); 1733 for (y = 0; cmds[y].c; y++) { 1734 if (!strncasecmp(cmd, cmds[y].c, cmdlen)) 1735 list[count++] = xstrdup(cmds[y].c); 1736 } 1737 list[count] = NULL; 1738 1739 if (count == 0) { 1740 free(list); 1741 return 0; 1742 } 1743 1744 /* Complete ambigious command */ 1745 tmp = complete_ambiguous(cmd, list, count); 1746 if (count > 1) 1747 complete_display(list, 0); 1748 1749 for (y = 0; list[y]; y++) 1750 free(list[y]); 1751 free(list); 1752 1753 if (tmp != NULL) { 1754 tmplen = strlen(tmp); 1755 cmdlen = strlen(cmd); 1756 /* If cmd may be extended then do so */ 1757 if (tmplen > cmdlen) 1758 if (el_insertstr(el, tmp + cmdlen) == -1) 1759 fatal("el_insertstr failed."); 1760 lf = el_line(el); 1761 /* Terminate argument cleanly */ 1762 if (count == 1) { 1763 y = 0; 1764 if (!terminated) 1765 argterm[y++] = quote; 1766 if (lastarg || *(lf->cursor) != ' ') 1767 argterm[y++] = ' '; 1768 argterm[y] = '\0'; 1769 if (y > 0 && el_insertstr(el, argterm) == -1) 1770 fatal("el_insertstr failed."); 1771 } 1772 free(tmp); 1773 } 1774 1775 return count; 1776 } 1777 1778 /* 1779 * Determine whether a particular sftp command's arguments (if any) 1780 * represent local or remote files. 1781 */ 1782 static int 1783 complete_is_remote(char *cmd) { 1784 int i; 1785 1786 if (cmd == NULL) 1787 return -1; 1788 1789 for (i = 0; cmds[i].c; i++) { 1790 if (!strncasecmp(cmd, cmds[i].c, strlen(cmds[i].c))) 1791 return cmds[i].t; 1792 } 1793 1794 return -1; 1795 } 1796 1797 /* Autocomplete a filename "file" */ 1798 static int 1799 complete_match(EditLine *el, struct sftp_conn *conn, char *remote_path, 1800 char *file, int remote, int lastarg, char quote, int terminated) 1801 { 1802 glob_t g; 1803 char *tmp, *tmp2, ins[8]; 1804 u_int i, hadglob, pwdlen, len, tmplen, filelen, cesc, isesc, isabs; 1805 int clen; 1806 const LineInfo *lf; 1807 1808 /* Glob from "file" location */ 1809 if (file == NULL) 1810 tmp = xstrdup("*"); 1811 else 1812 xasprintf(&tmp, "%s*", file); 1813 1814 /* Check if the path is absolute. */ 1815 isabs = tmp[0] == '/'; 1816 1817 memset(&g, 0, sizeof(g)); 1818 if (remote != LOCAL) { 1819 tmp = make_absolute(tmp, remote_path); 1820 remote_glob(conn, tmp, GLOB_DOOFFS|GLOB_MARK, NULL, &g); 1821 } else 1822 glob(tmp, GLOB_DOOFFS|GLOB_MARK, NULL, &g); 1823 1824 /* Determine length of pwd so we can trim completion display */ 1825 for (hadglob = tmplen = pwdlen = 0; tmp[tmplen] != 0; tmplen++) { 1826 /* Terminate counting on first unescaped glob metacharacter */ 1827 if (tmp[tmplen] == '*' || tmp[tmplen] == '?') { 1828 if (tmp[tmplen] != '*' || tmp[tmplen + 1] != '\0') 1829 hadglob = 1; 1830 break; 1831 } 1832 if (tmp[tmplen] == '\\' && tmp[tmplen + 1] != '\0') 1833 tmplen++; 1834 if (tmp[tmplen] == '/') 1835 pwdlen = tmplen + 1; /* track last seen '/' */ 1836 } 1837 free(tmp); 1838 1839 if (g.gl_matchc == 0) 1840 goto out; 1841 1842 if (g.gl_matchc > 1) 1843 complete_display(g.gl_pathv, pwdlen); 1844 1845 tmp = NULL; 1846 /* Don't try to extend globs */ 1847 if (file == NULL || hadglob) 1848 goto out; 1849 1850 tmp2 = complete_ambiguous(file, g.gl_pathv, g.gl_matchc); 1851 tmp = path_strip(tmp2, isabs ? NULL : remote_path); 1852 free(tmp2); 1853 1854 if (tmp == NULL) 1855 goto out; 1856 1857 tmplen = strlen(tmp); 1858 filelen = strlen(file); 1859 1860 /* Count the number of escaped characters in the input string. */ 1861 cesc = isesc = 0; 1862 for (i = 0; i < filelen; i++) { 1863 if (!isesc && file[i] == '\\' && i + 1 < filelen){ 1864 isesc = 1; 1865 cesc++; 1866 } else 1867 isesc = 0; 1868 } 1869 1870 if (tmplen > (filelen - cesc)) { 1871 tmp2 = tmp + filelen - cesc; 1872 len = strlen(tmp2); 1873 /* quote argument on way out */ 1874 for (i = 0; i < len; i += clen) { 1875 if ((clen = mblen(tmp2 + i, len - i)) < 0 || 1876 (size_t)clen > sizeof(ins) - 2) 1877 fatal("invalid multibyte character"); 1878 ins[0] = '\\'; 1879 memcpy(ins + 1, tmp2 + i, clen); 1880 ins[clen + 1] = '\0'; 1881 switch (tmp2[i]) { 1882 case '\'': 1883 case '"': 1884 case '\\': 1885 case '\t': 1886 case '[': 1887 case ' ': 1888 case '#': 1889 case '*': 1890 if (quote == '\0' || tmp2[i] == quote) { 1891 if (el_insertstr(el, ins) == -1) 1892 fatal("el_insertstr " 1893 "failed."); 1894 break; 1895 } 1896 /* FALLTHROUGH */ 1897 default: 1898 if (el_insertstr(el, ins + 1) == -1) 1899 fatal("el_insertstr failed."); 1900 break; 1901 } 1902 } 1903 } 1904 1905 lf = el_line(el); 1906 if (g.gl_matchc == 1) { 1907 i = 0; 1908 if (!terminated) 1909 ins[i++] = quote; 1910 if (*(lf->cursor - 1) != '/' && 1911 (lastarg || *(lf->cursor) != ' ')) 1912 ins[i++] = ' '; 1913 ins[i] = '\0'; 1914 if (i > 0 && el_insertstr(el, ins) == -1) 1915 fatal("el_insertstr failed."); 1916 } 1917 free(tmp); 1918 1919 out: 1920 globfree(&g); 1921 return g.gl_matchc; 1922 } 1923 1924 /* tab-completion hook function, called via libedit */ 1925 static unsigned char 1926 complete(EditLine *el, int ch) 1927 { 1928 char **argv, *line, quote; 1929 int argc, carg; 1930 u_int cursor, len, terminated, ret = CC_ERROR; 1931 const LineInfo *lf; 1932 struct complete_ctx *complete_ctx; 1933 1934 lf = el_line(el); 1935 if (el_get(el, EL_CLIENTDATA, (void**)&complete_ctx) != 0) 1936 fatal("%s: el_get failed", __func__); 1937 1938 /* Figure out which argument the cursor points to */ 1939 cursor = lf->cursor - lf->buffer; 1940 line = (char *)xmalloc(cursor + 1); 1941 memcpy(line, lf->buffer, cursor); 1942 line[cursor] = '\0'; 1943 argv = makeargv(line, &carg, 1, "e, &terminated); 1944 free(line); 1945 1946 /* Get all the arguments on the line */ 1947 len = lf->lastchar - lf->buffer; 1948 line = (char *)xmalloc(len + 1); 1949 memcpy(line, lf->buffer, len); 1950 line[len] = '\0'; 1951 argv = makeargv(line, &argc, 1, NULL, NULL); 1952 1953 /* Ensure cursor is at EOL or a argument boundary */ 1954 if (line[cursor] != ' ' && line[cursor] != '\0' && 1955 line[cursor] != '\n') { 1956 free(line); 1957 return ret; 1958 } 1959 1960 if (carg == 0) { 1961 /* Show all available commands */ 1962 complete_cmd_parse(el, NULL, argc == carg, '\0', 1); 1963 ret = CC_REDISPLAY; 1964 } else if (carg == 1 && cursor > 0 && line[cursor - 1] != ' ') { 1965 /* Handle the command parsing */ 1966 if (complete_cmd_parse(el, argv[0], argc == carg, 1967 quote, terminated) != 0) 1968 ret = CC_REDISPLAY; 1969 } else if (carg >= 1) { 1970 /* Handle file parsing */ 1971 int remote = complete_is_remote(argv[0]); 1972 char *filematch = NULL; 1973 1974 if (carg > 1 && line[cursor-1] != ' ') 1975 filematch = argv[carg - 1]; 1976 1977 if (remote != 0 && 1978 complete_match(el, complete_ctx->conn, 1979 *complete_ctx->remote_pathp, filematch, 1980 remote, carg == argc, quote, terminated) != 0) 1981 ret = CC_REDISPLAY; 1982 } 1983 1984 free(line); 1985 return ret; 1986 } 1987 #endif /* USE_LIBEDIT */ 1988 1989 int 1990 interactive_loop(struct sftp_conn *conn, char *file1, char *file2) 1991 { 1992 char *remote_path; 1993 char *dir = NULL; 1994 char cmd[2048]; 1995 int err, interactive; 1996 EditLine *el = NULL; 1997 #ifdef USE_LIBEDIT 1998 History *hl = NULL; 1999 HistEvent hev; 2000 extern char *__progname; 2001 struct complete_ctx complete_ctx; 2002 2003 if (!batchmode && isatty(STDIN_FILENO)) { 2004 if ((el = el_init(__progname, stdin, stdout, stderr)) == NULL) 2005 fatal("Couldn't initialise editline"); 2006 if ((hl = history_init()) == NULL) 2007 fatal("Couldn't initialise editline history"); 2008 history(hl, &hev, H_SETSIZE, 100); 2009 el_set(el, EL_HIST, history, hl); 2010 2011 el_set(el, EL_PROMPT, prompt); 2012 el_set(el, EL_EDITOR, "emacs"); 2013 el_set(el, EL_TERMINAL, NULL); 2014 el_set(el, EL_SIGNAL, 1); 2015 el_source(el, NULL); 2016 2017 /* Tab Completion */ 2018 el_set(el, EL_ADDFN, "ftp-complete", 2019 "Context sensitive argument completion", complete); 2020 complete_ctx.conn = conn; 2021 complete_ctx.remote_pathp = &remote_path; 2022 el_set(el, EL_CLIENTDATA, (void*)&complete_ctx); 2023 el_set(el, EL_BIND, "^I", "ftp-complete", NULL); 2024 /* enable ctrl-left-arrow and ctrl-right-arrow */ 2025 el_set(el, EL_BIND, "\\e[1;5C", "em-next-word", NULL); 2026 el_set(el, EL_BIND, "\\e[5C", "em-next-word", NULL); 2027 el_set(el, EL_BIND, "\\e[1;5D", "ed-prev-word", NULL); 2028 el_set(el, EL_BIND, "\\e\\e[D", "ed-prev-word", NULL); 2029 /* make ^w match ksh behaviour */ 2030 el_set(el, EL_BIND, "^w", "ed-delete-prev-word", NULL); 2031 } 2032 #endif /* USE_LIBEDIT */ 2033 2034 remote_path = do_realpath(conn, "."); 2035 if (remote_path == NULL) 2036 fatal("Need cwd"); 2037 2038 if (file1 != NULL) { 2039 dir = xstrdup(file1); 2040 dir = make_absolute(dir, remote_path); 2041 2042 if (remote_is_dir(conn, dir) && file2 == NULL) { 2043 if (!quiet) 2044 printf("Changing to: %s\n", dir); 2045 snprintf(cmd, sizeof cmd, "cd \"%s\"", dir); 2046 if (parse_dispatch_command(conn, cmd, 2047 &remote_path, 1) != 0) { 2048 free(dir); 2049 free(remote_path); 2050 free(conn); 2051 return (-1); 2052 } 2053 } else { 2054 /* XXX this is wrong wrt quoting */ 2055 snprintf(cmd, sizeof cmd, "get%s %s%s%s", 2056 global_aflag ? " -a" : "", dir, 2057 file2 == NULL ? "" : " ", 2058 file2 == NULL ? "" : file2); 2059 err = parse_dispatch_command(conn, cmd, 2060 &remote_path, 1); 2061 free(dir); 2062 free(remote_path); 2063 free(conn); 2064 return (err); 2065 } 2066 free(dir); 2067 } 2068 2069 setlinebuf(stdout); 2070 setlinebuf(infile); 2071 2072 interactive = !batchmode && isatty(STDIN_FILENO); 2073 err = 0; 2074 for (;;) { 2075 char *cp; 2076 2077 signal(SIGINT, SIG_IGN); 2078 2079 if (el == NULL) { 2080 if (interactive) 2081 printf("sftp> "); 2082 if (fgets(cmd, sizeof(cmd), infile) == NULL) { 2083 if (interactive) 2084 printf("\n"); 2085 break; 2086 } 2087 if (!interactive) { /* Echo command */ 2088 printf("sftp> %s", cmd); 2089 if (strlen(cmd) > 0 && 2090 cmd[strlen(cmd) - 1] != '\n') 2091 printf("\n"); 2092 } 2093 } else { 2094 #ifdef USE_LIBEDIT 2095 const char *line; 2096 int count = 0; 2097 2098 if ((line = el_gets(el, &count)) == NULL || 2099 count <= 0) { 2100 printf("\n"); 2101 break; 2102 } 2103 history(hl, &hev, H_ENTER, line); 2104 if (strlcpy(cmd, line, sizeof(cmd)) >= sizeof(cmd)) { 2105 fprintf(stderr, "Error: input line too long\n"); 2106 continue; 2107 } 2108 #endif /* USE_LIBEDIT */ 2109 } 2110 2111 cp = strrchr(cmd, '\n'); 2112 if (cp) 2113 *cp = '\0'; 2114 2115 /* Handle user interrupts gracefully during commands */ 2116 interrupted = 0; 2117 signal(SIGINT, cmd_interrupt); 2118 2119 err = parse_dispatch_command(conn, cmd, &remote_path, 2120 batchmode); 2121 if (err != 0) 2122 break; 2123 } 2124 free(remote_path); 2125 free(conn); 2126 2127 #ifdef USE_LIBEDIT 2128 if (el != NULL) 2129 el_end(el); 2130 #endif /* USE_LIBEDIT */ 2131 2132 /* err == 1 signifies normal "quit" exit */ 2133 return (err >= 0 ? 0 : -1); 2134 } 2135 2136 static void 2137 connect_to_server(char *path, char **args, int *in, int *out) 2138 { 2139 int c_in, c_out; 2140 2141 #ifdef USE_PIPES 2142 int pin[2], pout[2]; 2143 2144 if ((pipe(pin) == -1) || (pipe(pout) == -1)) 2145 fatal("pipe: %s", strerror(errno)); 2146 *in = pin[0]; 2147 *out = pout[1]; 2148 c_in = pout[0]; 2149 c_out = pin[1]; 2150 #else /* USE_PIPES */ 2151 int inout[2]; 2152 2153 if (socketpair(AF_UNIX, SOCK_STREAM, 0, inout) == -1) 2154 fatal("socketpair: %s", strerror(errno)); 2155 *in = *out = inout[0]; 2156 c_in = c_out = inout[1]; 2157 #endif /* USE_PIPES */ 2158 2159 if ((sshpid = fork()) == -1) 2160 fatal("fork: %s", strerror(errno)); 2161 else if (sshpid == 0) { 2162 if ((dup2(c_in, STDIN_FILENO) == -1) || 2163 (dup2(c_out, STDOUT_FILENO) == -1)) { 2164 fprintf(stderr, "dup2: %s\n", strerror(errno)); 2165 _exit(1); 2166 } 2167 close(*in); 2168 close(*out); 2169 close(c_in); 2170 close(c_out); 2171 2172 /* 2173 * The underlying ssh is in the same process group, so we must 2174 * ignore SIGINT if we want to gracefully abort commands, 2175 * otherwise the signal will make it to the ssh process and 2176 * kill it too. Contrawise, since sftp sends SIGTERMs to the 2177 * underlying ssh, it must *not* ignore that signal. 2178 */ 2179 signal(SIGINT, SIG_IGN); 2180 signal(SIGTERM, SIG_DFL); 2181 execvp(path, args); 2182 fprintf(stderr, "exec: %s: %s\n", path, strerror(errno)); 2183 _exit(1); 2184 } 2185 2186 signal(SIGTERM, killchild); 2187 signal(SIGINT, killchild); 2188 signal(SIGHUP, killchild); 2189 close(c_in); 2190 close(c_out); 2191 } 2192 2193 static void 2194 usage(void) 2195 { 2196 extern char *__progname; 2197 2198 fprintf(stderr, 2199 "usage: %s [-1246aCfpqrv] [-B buffer_size] [-b batchfile] [-c cipher]\n" 2200 " [-D sftp_server_path] [-F ssh_config] " 2201 "[-i identity_file] [-l limit]\n" 2202 " [-o ssh_option] [-P port] [-R num_requests] " 2203 "[-S program]\n" 2204 " [-s subsystem | sftp_server] host\n" 2205 " %s [user@]host[:file ...]\n" 2206 " %s [user@]host[:dir[/]]\n" 2207 " %s -b batchfile [user@]host\n", 2208 __progname, __progname, __progname, __progname); 2209 exit(1); 2210 } 2211 2212 int 2213 main(int argc, char **argv) 2214 { 2215 int in, out, ch, err; 2216 char *host = NULL, *userhost, *cp, *file2 = NULL; 2217 int debug_level = 0, sshver = 2; 2218 char *file1 = NULL, *sftp_server = NULL; 2219 char *ssh_program = _PATH_SSH_PROGRAM, *sftp_direct = NULL; 2220 const char *errstr; 2221 LogLevel ll = SYSLOG_LEVEL_INFO; 2222 arglist args; 2223 extern int optind; 2224 extern char *optarg; 2225 struct sftp_conn *conn; 2226 size_t copy_buffer_len = DEFAULT_COPY_BUFLEN; 2227 size_t num_requests = DEFAULT_NUM_REQUESTS; 2228 long long limit_kbps = 0; 2229 2230 /* Ensure that fds 0, 1 and 2 are open or directed to /dev/null */ 2231 sanitise_stdfd(); 2232 setlocale(LC_CTYPE, ""); 2233 2234 __progname = ssh_get_progname(argv[0]); 2235 memset(&args, '\0', sizeof(args)); 2236 args.list = NULL; 2237 addargs(&args, "%s", ssh_program); 2238 addargs(&args, "-oForwardX11 no"); 2239 addargs(&args, "-oForwardAgent no"); 2240 addargs(&args, "-oPermitLocalCommand no"); 2241 addargs(&args, "-oClearAllForwardings yes"); 2242 2243 ll = SYSLOG_LEVEL_INFO; 2244 infile = stdin; 2245 2246 while ((ch = getopt(argc, argv, 2247 "1246afhpqrvCc:D:i:l:o:s:S:b:B:F:P:R:")) != -1) { 2248 switch (ch) { 2249 /* Passed through to ssh(1) */ 2250 case '4': 2251 case '6': 2252 case 'C': 2253 addargs(&args, "-%c", ch); 2254 break; 2255 /* Passed through to ssh(1) with argument */ 2256 case 'F': 2257 case 'c': 2258 case 'i': 2259 case 'o': 2260 addargs(&args, "-%c", ch); 2261 addargs(&args, "%s", optarg); 2262 break; 2263 case 'q': 2264 ll = SYSLOG_LEVEL_ERROR; 2265 quiet = 1; 2266 showprogress = 0; 2267 addargs(&args, "-%c", ch); 2268 break; 2269 case 'P': 2270 addargs(&args, "-oPort %s", optarg); 2271 break; 2272 case 'v': 2273 if (debug_level < 3) { 2274 addargs(&args, "-v"); 2275 ll = SYSLOG_LEVEL_DEBUG1 + debug_level; 2276 } 2277 debug_level++; 2278 break; 2279 case '1': 2280 sshver = 1; 2281 if (sftp_server == NULL) 2282 sftp_server = _PATH_SFTP_SERVER; 2283 break; 2284 case '2': 2285 sshver = 2; 2286 break; 2287 case 'a': 2288 global_aflag = 1; 2289 break; 2290 case 'B': 2291 copy_buffer_len = strtol(optarg, &cp, 10); 2292 if (copy_buffer_len == 0 || *cp != '\0') 2293 fatal("Invalid buffer size \"%s\"", optarg); 2294 break; 2295 case 'b': 2296 if (batchmode) 2297 fatal("Batch file already specified."); 2298 2299 /* Allow "-" as stdin */ 2300 if (strcmp(optarg, "-") != 0 && 2301 (infile = fopen(optarg, "r")) == NULL) 2302 fatal("%s (%s).", strerror(errno), optarg); 2303 showprogress = 0; 2304 quiet = batchmode = 1; 2305 addargs(&args, "-obatchmode yes"); 2306 break; 2307 case 'f': 2308 global_fflag = 1; 2309 break; 2310 case 'p': 2311 global_pflag = 1; 2312 break; 2313 case 'D': 2314 sftp_direct = optarg; 2315 break; 2316 case 'l': 2317 limit_kbps = strtonum(optarg, 1, 100 * 1024 * 1024, 2318 &errstr); 2319 if (errstr != NULL) 2320 usage(); 2321 limit_kbps *= 1024; /* kbps */ 2322 break; 2323 case 'r': 2324 global_rflag = 1; 2325 break; 2326 case 'R': 2327 num_requests = strtol(optarg, &cp, 10); 2328 if (num_requests == 0 || *cp != '\0') 2329 fatal("Invalid number of requests \"%s\"", 2330 optarg); 2331 break; 2332 case 's': 2333 sftp_server = optarg; 2334 break; 2335 case 'S': 2336 ssh_program = optarg; 2337 replacearg(&args, 0, "%s", ssh_program); 2338 break; 2339 case 'h': 2340 default: 2341 usage(); 2342 } 2343 } 2344 2345 if (!isatty(STDERR_FILENO)) 2346 showprogress = 0; 2347 2348 log_init(argv[0], ll, SYSLOG_FACILITY_USER, 1); 2349 2350 if (sftp_direct == NULL) { 2351 if (optind == argc || argc > (optind + 2)) 2352 usage(); 2353 2354 userhost = xstrdup(argv[optind]); 2355 file2 = argv[optind+1]; 2356 2357 if ((host = strrchr(userhost, '@')) == NULL) 2358 host = userhost; 2359 else { 2360 *host++ = '\0'; 2361 if (!userhost[0]) { 2362 fprintf(stderr, "Missing username\n"); 2363 usage(); 2364 } 2365 addargs(&args, "-l"); 2366 addargs(&args, "%s", userhost); 2367 } 2368 2369 if ((cp = colon(host)) != NULL) { 2370 *cp++ = '\0'; 2371 file1 = cp; 2372 } 2373 2374 host = cleanhostname(host); 2375 if (!*host) { 2376 fprintf(stderr, "Missing hostname\n"); 2377 usage(); 2378 } 2379 2380 addargs(&args, "-oProtocol %d", sshver); 2381 2382 /* no subsystem if the server-spec contains a '/' */ 2383 if (sftp_server == NULL || strchr(sftp_server, '/') == NULL) 2384 addargs(&args, "-s"); 2385 2386 addargs(&args, "--"); 2387 addargs(&args, "%s", host); 2388 addargs(&args, "%s", (sftp_server != NULL ? 2389 sftp_server : "sftp")); 2390 2391 connect_to_server(ssh_program, args.list, &in, &out); 2392 } else { 2393 args.list = NULL; 2394 addargs(&args, "sftp-server"); 2395 2396 connect_to_server(sftp_direct, args.list, &in, &out); 2397 } 2398 freeargs(&args); 2399 2400 conn = do_init(in, out, copy_buffer_len, num_requests, limit_kbps); 2401 if (conn == NULL) 2402 fatal("Couldn't initialise connection to server"); 2403 2404 if (!quiet) { 2405 if (sftp_direct == NULL) 2406 fprintf(stderr, "Connected to %s.\n", host); 2407 else 2408 fprintf(stderr, "Attached to %s.\n", sftp_direct); 2409 } 2410 2411 err = interactive_loop(conn, file1, file2); 2412 2413 #if !defined(USE_PIPES) 2414 shutdown(in, SHUT_RDWR); 2415 shutdown(out, SHUT_RDWR); 2416 #endif 2417 2418 close(in); 2419 close(out); 2420 if (batchmode) 2421 fclose(infile); 2422 2423 while (waitpid(sshpid, NULL, 0) == -1) 2424 if (errno != EINTR) 2425 fatal("Couldn't wait for ssh process: %s", 2426 strerror(errno)); 2427 2428 exit(err == 0 ? 0 : 1); 2429 } 2430