]> andersk Git - gssapi-openssh.git/blob - openssh/sftp-int.c
6987de9a3888729901f376e6dc4e4d0017221dc9
[gssapi-openssh.git] / openssh / sftp-int.c
1 /*
2  * Copyright (c) 2001,2002 Damien Miller.  All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
14  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
15  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
16  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
17  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
18  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
19  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
20  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
21  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
22  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
23  */
24
25 /* XXX: recursive operations */
26
27 #include "includes.h"
28 RCSID("$OpenBSD: sftp-int.c,v 1.57 2003/03/05 22:33:43 markus Exp $");
29
30 #include "buffer.h"
31 #include "xmalloc.h"
32 #include "log.h"
33 #include "pathnames.h"
34
35 #include "sftp.h"
36 #include "sftp-common.h"
37 #include "sftp-glob.h"
38 #include "sftp-client.h"
39 #include "sftp-int.h"
40
41 /* File to read commands from */
42 extern FILE *infile;
43
44 /* Size of buffer used when copying files */
45 extern size_t copy_buffer_len;
46
47 /* Number of concurrent outstanding requests */
48 extern int num_requests;
49
50 /* This is set to 0 if the progressmeter is not desired. */
51 int showprogress = 1;
52
53 /* Seperators for interactive commands */
54 #define WHITESPACE " \t\r\n"
55
56 /* Commands for interactive mode */
57 #define I_CHDIR         1
58 #define I_CHGRP         2
59 #define I_CHMOD         3
60 #define I_CHOWN         4
61 #define I_GET           5
62 #define I_HELP          6
63 #define I_LCHDIR        7
64 #define I_LLS           8
65 #define I_LMKDIR        9
66 #define I_LPWD          10
67 #define I_LS            11
68 #define I_LUMASK        12
69 #define I_MKDIR         13
70 #define I_PUT           14
71 #define I_PWD           15
72 #define I_QUIT          16
73 #define I_RENAME        17
74 #define I_RM            18
75 #define I_RMDIR         19
76 #define I_SHELL         20
77 #define I_SYMLINK       21
78 #define I_VERSION       22
79 #define I_PROGRESS      23
80
81 struct CMD {
82         const char *c;
83         const int n;
84 };
85
86 static const struct CMD cmds[] = {
87         { "bye",        I_QUIT },
88         { "cd",         I_CHDIR },
89         { "chdir",      I_CHDIR },
90         { "chgrp",      I_CHGRP },
91         { "chmod",      I_CHMOD },
92         { "chown",      I_CHOWN },
93         { "dir",        I_LS },
94         { "exit",       I_QUIT },
95         { "get",        I_GET },
96         { "mget",       I_GET },
97         { "help",       I_HELP },
98         { "lcd",        I_LCHDIR },
99         { "lchdir",     I_LCHDIR },
100         { "lls",        I_LLS },
101         { "lmkdir",     I_LMKDIR },
102         { "ln",         I_SYMLINK },
103         { "lpwd",       I_LPWD },
104         { "ls",         I_LS },
105         { "lumask",     I_LUMASK },
106         { "mkdir",      I_MKDIR },
107         { "progress",   I_PROGRESS },
108         { "put",        I_PUT },
109         { "mput",       I_PUT },
110         { "pwd",        I_PWD },
111         { "quit",       I_QUIT },
112         { "rename",     I_RENAME },
113         { "rm",         I_RM },
114         { "rmdir",      I_RMDIR },
115         { "symlink",    I_SYMLINK },
116         { "version",    I_VERSION },
117         { "!",          I_SHELL },
118         { "?",          I_HELP },
119         { NULL,                 -1}
120 };
121
122 static void
123 help(void)
124 {
125         printf("Available commands:\n");
126         printf("cd path                       Change remote directory to 'path'\n");
127         printf("lcd path                      Change local directory to 'path'\n");
128         printf("chgrp grp path                Change group of file 'path' to 'grp'\n");
129         printf("chmod mode path               Change permissions of file 'path' to 'mode'\n");
130         printf("chown own path                Change owner of file 'path' to 'own'\n");
131         printf("help                          Display this help text\n");
132         printf("get remote-path [local-path]  Download file\n");
133         printf("lls [ls-options [path]]       Display local directory listing\n");
134         printf("ln oldpath newpath            Symlink remote file\n");
135         printf("lmkdir path                   Create local directory\n");
136         printf("lpwd                          Print local working directory\n");
137         printf("ls [path]                     Display remote directory listing\n");
138         printf("lumask umask                  Set local umask to 'umask'\n");
139         printf("mkdir path                    Create remote directory\n");
140         printf("progress                      Toggle display of progress meter\n");
141         printf("put local-path [remote-path]  Upload file\n");
142         printf("pwd                           Display remote working directory\n");
143         printf("exit                          Quit sftp\n");
144         printf("quit                          Quit sftp\n");
145         printf("rename oldpath newpath        Rename remote file\n");
146         printf("rmdir path                    Remove remote directory\n");
147         printf("rm path                       Delete remote file\n");
148         printf("symlink oldpath newpath       Symlink remote file\n");
149         printf("version                       Show SFTP version\n");
150         printf("!command                      Execute 'command' in local shell\n");
151         printf("!                             Escape to local shell\n");
152         printf("?                             Synonym for help\n");
153 }
154
155 static void
156 local_do_shell(const char *args)
157 {
158         int status;
159         char *shell;
160         pid_t pid;
161
162         if (!*args)
163                 args = NULL;
164
165         if ((shell = getenv("SHELL")) == NULL)
166                 shell = _PATH_BSHELL;
167
168         if ((pid = fork()) == -1)
169                 fatal("Couldn't fork: %s", strerror(errno));
170
171         if (pid == 0) {
172                 /* XXX: child has pipe fds to ssh subproc open - issue? */
173                 if (args) {
174                         debug3("Executing %s -c \"%s\"", shell, args);
175                         execl(shell, shell, "-c", args, (char *)NULL);
176                 } else {
177                         debug3("Executing %s", shell);
178                         execl(shell, shell, (char *)NULL);
179                 }
180                 fprintf(stderr, "Couldn't execute \"%s\": %s\n", shell,
181                     strerror(errno));
182                 _exit(1);
183         }
184         while (waitpid(pid, &status, 0) == -1)
185                 if (errno != EINTR)
186                         fatal("Couldn't wait for child: %s", strerror(errno));
187         if (!WIFEXITED(status))
188                 error("Shell exited abormally");
189         else if (WEXITSTATUS(status))
190                 error("Shell exited with status %d", WEXITSTATUS(status));
191 }
192
193 static void
194 local_do_ls(const char *args)
195 {
196         if (!args || !*args)
197                 local_do_shell(_PATH_LS);
198         else {
199                 int len = strlen(_PATH_LS " ") + strlen(args) + 1;
200                 char *buf = xmalloc(len);
201
202                 /* XXX: quoting - rip quoting code from ftp? */
203                 snprintf(buf, len, _PATH_LS " %s", args);
204                 local_do_shell(buf);
205                 xfree(buf);
206         }
207 }
208
209 /* Strip one path (usually the pwd) from the start of another */
210 static char *
211 path_strip(char *path, char *strip)
212 {
213         size_t len;
214
215         if (strip == NULL)
216                 return (xstrdup(path));
217
218         len = strlen(strip);
219         if (strip != NULL && strncmp(path, strip, len) == 0) {
220                 if (strip[len - 1] != '/' && path[len] == '/')
221                         len++;
222                 return (xstrdup(path + len));
223         }
224
225         return (xstrdup(path));
226 }
227
228 static char *
229 path_append(char *p1, char *p2)
230 {
231         char *ret;
232         int len = strlen(p1) + strlen(p2) + 2;
233
234         ret = xmalloc(len);
235         strlcpy(ret, p1, len);
236         if (p1[strlen(p1) - 1] != '/')
237                 strlcat(ret, "/", len);
238         strlcat(ret, p2, len);
239
240         return(ret);
241 }
242
243 static char *
244 make_absolute(char *p, char *pwd)
245 {
246         char *abs;
247
248         /* Derelativise */
249         if (p && p[0] != '/') {
250                 abs = path_append(pwd, p);
251                 xfree(p);
252                 return(abs);
253         } else
254                 return(p);
255 }
256
257 static int
258 infer_path(const char *p, char **ifp)
259 {
260         char *cp;
261
262         cp = strrchr(p, '/');
263         if (cp == NULL) {
264                 *ifp = xstrdup(p);
265                 return(0);
266         }
267
268         if (!cp[1]) {
269                 error("Invalid path");
270                 return(-1);
271         }
272
273         *ifp = xstrdup(cp + 1);
274         return(0);
275 }
276
277 static int
278 parse_getput_flags(const char **cpp, int *pflag)
279 {
280         const char *cp = *cpp;
281
282         /* Check for flags */
283         if (cp[0] == '-' && cp[1] && strchr(WHITESPACE, cp[2])) {
284                 switch (cp[1]) {
285                 case 'p':
286                 case 'P':
287                         *pflag = 1;
288                         break;
289                 default:
290                         error("Invalid flag -%c", cp[1]);
291                         return(-1);
292                 }
293                 cp += 2;
294                 *cpp = cp + strspn(cp, WHITESPACE);
295         }
296
297         return(0);
298 }
299
300 static int
301 parse_ls_flags(const char **cpp, int *lflag)
302 {
303         const char *cp = *cpp;
304
305         /* Check for flags */
306         if (cp++[0] == '-') {
307                 for(; strchr(WHITESPACE, *cp) == NULL; cp++) {
308                         switch (*cp) {
309                         case 'l':
310                                 *lflag = 1;
311                                 break;
312                         default:
313                                 error("Invalid flag -%c", *cp);
314                                 return(-1);
315                         }
316                 }
317                 *cpp = cp + strspn(cp, WHITESPACE);
318         }
319
320         return(0);
321 }
322
323 static int
324 get_pathname(const char **cpp, char **path)
325 {
326         const char *cp = *cpp, *end;
327         char quot;
328         int i;
329
330         cp += strspn(cp, WHITESPACE);
331         if (!*cp) {
332                 *cpp = cp;
333                 *path = NULL;
334                 return (0);
335         }
336
337         /* Check for quoted filenames */
338         if (*cp == '\"' || *cp == '\'') {
339                 quot = *cp++;
340
341                 end = strchr(cp, quot);
342                 if (end == NULL) {
343                         error("Unterminated quote");
344                         goto fail;
345                 }
346                 if (cp == end) {
347                         error("Empty quotes");
348                         goto fail;
349                 }
350                 *cpp = end + 1 + strspn(end + 1, WHITESPACE);
351         } else {
352                 /* Read to end of filename */
353                 end = strpbrk(cp, WHITESPACE);
354                 if (end == NULL)
355                         end = strchr(cp, '\0');
356                 *cpp = end + strspn(end, WHITESPACE);
357         }
358
359         i = end - cp;
360
361         *path = xmalloc(i + 1);
362         memcpy(*path, cp, i);
363         (*path)[i] = '\0';
364         return(0);
365
366  fail:
367         *path = NULL;
368         return (-1);
369 }
370
371 static int
372 is_dir(char *path)
373 {
374         struct stat sb;
375
376         /* XXX: report errors? */
377         if (stat(path, &sb) == -1)
378                 return(0);
379
380         return(sb.st_mode & S_IFDIR);
381 }
382
383 static int
384 is_reg(char *path)
385 {
386         struct stat sb;
387
388         if (stat(path, &sb) == -1)
389                 fatal("stat %s: %s", path, strerror(errno));
390
391         return(S_ISREG(sb.st_mode));
392 }
393
394 static int
395 remote_is_dir(struct sftp_conn *conn, char *path)
396 {
397         Attrib *a;
398
399         /* XXX: report errors? */
400         if ((a = do_stat(conn, path, 1)) == NULL)
401                 return(0);
402         if (!(a->flags & SSH2_FILEXFER_ATTR_PERMISSIONS))
403                 return(0);
404         return(a->perm & S_IFDIR);
405 }
406
407 static int
408 process_get(struct sftp_conn *conn, char *src, char *dst, char *pwd, int pflag)
409 {
410         char *abs_src = NULL;
411         char *abs_dst = NULL;
412         char *tmp;
413         glob_t g;
414         int err = 0;
415         int i;
416
417         abs_src = xstrdup(src);
418         abs_src = make_absolute(abs_src, pwd);
419
420         memset(&g, 0, sizeof(g));
421         debug3("Looking up %s", abs_src);
422         if (remote_glob(conn, abs_src, 0, NULL, &g)) {
423                 error("File \"%s\" not found.", abs_src);
424                 err = -1;
425                 goto out;
426         }
427
428         /* Only one match, dst may be file, directory or unspecified */
429         if (g.gl_pathv[0] && g.gl_matchc == 1) {
430                 if (dst) {
431                         /* If directory specified, append filename */
432                         if (is_dir(dst)) {
433                                 if (infer_path(g.gl_pathv[0], &tmp)) {
434                                         err = 1;
435                                         goto out;
436                                 }
437                                 abs_dst = path_append(dst, tmp);
438                                 xfree(tmp);
439                         } else
440                                 abs_dst = xstrdup(dst);
441                 } else if (infer_path(g.gl_pathv[0], &abs_dst)) {
442                         err = -1;
443                         goto out;
444                 }
445                 err = do_download(conn, g.gl_pathv[0], abs_dst, pflag);
446                 goto out;
447         }
448
449         /* Multiple matches, dst may be directory or unspecified */
450         if (dst && !is_dir(dst)) {
451                 error("Multiple files match, but \"%s\" is not a directory",
452                     dst);
453                 err = -1;
454                 goto out;
455         }
456
457         for (i = 0; g.gl_pathv[i]; i++) {
458                 if (infer_path(g.gl_pathv[i], &tmp)) {
459                         err = -1;
460                         goto out;
461                 }
462                 if (dst) {
463                         abs_dst = path_append(dst, tmp);
464                         xfree(tmp);
465                 } else
466                         abs_dst = tmp;
467
468                 printf("Fetching %s to %s\n", g.gl_pathv[i], abs_dst);
469                 if (do_download(conn, g.gl_pathv[i], abs_dst, pflag) == -1)
470                         err = -1;
471                 xfree(abs_dst);
472                 abs_dst = NULL;
473         }
474
475 out:
476         xfree(abs_src);
477         if (abs_dst)
478                 xfree(abs_dst);
479         globfree(&g);
480         return(err);
481 }
482
483 static int
484 process_put(struct sftp_conn *conn, char *src, char *dst, char *pwd, int pflag)
485 {
486         char *tmp_dst = NULL;
487         char *abs_dst = NULL;
488         char *tmp;
489         glob_t g;
490         int err = 0;
491         int i;
492
493         if (dst) {
494                 tmp_dst = xstrdup(dst);
495                 tmp_dst = make_absolute(tmp_dst, pwd);
496         }
497
498         memset(&g, 0, sizeof(g));
499         debug3("Looking up %s", src);
500         if (glob(src, 0, NULL, &g)) {
501                 error("File \"%s\" not found.", src);
502                 err = -1;
503                 goto out;
504         }
505
506         /* Only one match, dst may be file, directory or unspecified */
507         if (g.gl_pathv[0] && g.gl_matchc == 1) {
508                 if (!is_reg(g.gl_pathv[0])) {
509                         error("Can't upload %s: not a regular file",
510                             g.gl_pathv[0]);
511                         err = 1;
512                         goto out;
513                 }
514                 if (tmp_dst) {
515                         /* If directory specified, append filename */
516                         if (remote_is_dir(conn, tmp_dst)) {
517                                 if (infer_path(g.gl_pathv[0], &tmp)) {
518                                         err = 1;
519                                         goto out;
520                                 }
521                                 abs_dst = path_append(tmp_dst, tmp);
522                                 xfree(tmp);
523                         } else
524                                 abs_dst = xstrdup(tmp_dst);
525                 } else {
526                         if (infer_path(g.gl_pathv[0], &abs_dst)) {
527                                 err = -1;
528                                 goto out;
529                         }
530                         abs_dst = make_absolute(abs_dst, pwd);
531                 }
532                 err = do_upload(conn, g.gl_pathv[0], abs_dst, pflag);
533                 goto out;
534         }
535
536         /* Multiple matches, dst may be directory or unspecified */
537         if (tmp_dst && !remote_is_dir(conn, tmp_dst)) {
538                 error("Multiple files match, but \"%s\" is not a directory",
539                     tmp_dst);
540                 err = -1;
541                 goto out;
542         }
543
544         for (i = 0; g.gl_pathv[i]; i++) {
545                 if (!is_reg(g.gl_pathv[i])) {
546                         error("skipping non-regular file %s", 
547                             g.gl_pathv[i]);
548                         continue;
549                 }
550                 if (infer_path(g.gl_pathv[i], &tmp)) {
551                         err = -1;
552                         goto out;
553                 }
554                 if (tmp_dst) {
555                         abs_dst = path_append(tmp_dst, tmp);
556                         xfree(tmp);
557                 } else
558                         abs_dst = make_absolute(tmp, pwd);
559
560                 printf("Uploading %s to %s\n", g.gl_pathv[i], abs_dst);
561                 if (do_upload(conn, g.gl_pathv[i], abs_dst, pflag) == -1)
562                         err = -1;
563         }
564
565 out:
566         if (abs_dst)
567                 xfree(abs_dst);
568         if (tmp_dst)
569                 xfree(tmp_dst);
570         return(err);
571 }
572
573 static int
574 sdirent_comp(const void *aa, const void *bb)
575 {
576         SFTP_DIRENT *a = *(SFTP_DIRENT **)aa;
577         SFTP_DIRENT *b = *(SFTP_DIRENT **)bb;
578
579         return (strcmp(a->filename, b->filename));
580 }
581
582 /* sftp ls.1 replacement for directories */
583 static int
584 do_ls_dir(struct sftp_conn *conn, char *path, char *strip_path, int lflag)
585 {
586         int n;
587         SFTP_DIRENT **d;
588
589         if ((n = do_readdir(conn, path, &d)) != 0)
590                 return (n);
591
592         /* Count entries for sort */
593         for (n = 0; d[n] != NULL; n++)
594                 ;
595
596         qsort(d, n, sizeof(*d), sdirent_comp);
597
598         for (n = 0; d[n] != NULL; n++) {
599                 char *tmp, *fname;
600
601                 tmp = path_append(path, d[n]->filename);
602                 fname = path_strip(tmp, strip_path);
603                 xfree(tmp);
604
605                 if (lflag) {
606                         char *lname;
607                         struct stat sb;
608
609                         memset(&sb, 0, sizeof(sb));
610                         attrib_to_stat(&d[n]->a, &sb);
611                         lname = ls_file(fname, &sb, 1);
612                         printf("%s\n", lname);
613                         xfree(lname);
614                 } else {
615                         /* XXX - multicolumn display would be nice here */
616                         printf("%s\n", fname);
617                 }
618
619                 xfree(fname);
620         }
621
622         free_sftp_dirents(d);
623         return (0);
624 }
625
626 /* sftp ls.1 replacement which handles path globs */
627 static int
628 do_globbed_ls(struct sftp_conn *conn, char *path, char *strip_path,
629     int lflag)
630 {
631         glob_t g;
632         int i;
633         Attrib *a;
634         struct stat sb;
635
636         memset(&g, 0, sizeof(g));
637
638         if (remote_glob(conn, path, GLOB_MARK|GLOB_NOCHECK|GLOB_BRACE,
639             NULL, &g)) {
640                 error("Can't ls: \"%s\" not found", path);
641                 return (-1);
642         }
643
644         /*
645          * If the glob returns a single match, which is the same as the
646          * input glob, and it is a directory, then just list its contents
647          */
648         if (g.gl_pathc == 1 &&
649             strncmp(path, g.gl_pathv[0], strlen(g.gl_pathv[0]) - 1) == 0) {
650                 if ((a = do_lstat(conn, path, 1)) == NULL) {
651                         globfree(&g);
652                         return (-1);
653                 }
654                 if ((a->flags & SSH2_FILEXFER_ATTR_PERMISSIONS) &&
655                     S_ISDIR(a->perm)) {
656                         globfree(&g);
657                         return (do_ls_dir(conn, path, strip_path, lflag));
658                 }
659         }
660
661         for (i = 0; g.gl_pathv[i]; i++) {
662                 char *fname, *lname;
663
664                 fname = path_strip(g.gl_pathv[i], strip_path);
665
666                 if (lflag) {
667                         /*
668                          * XXX: this is slow - 1 roundtrip per path
669                          * A solution to this is to fork glob() and
670                          * build a sftp specific version which keeps the
671                          * attribs (which currently get thrown away)
672                          * that the server returns as well as the filenames.
673                          */
674                         memset(&sb, 0, sizeof(sb));
675                         a = do_lstat(conn, g.gl_pathv[i], 1);
676                         if (a != NULL)
677                                 attrib_to_stat(a, &sb);
678                         lname = ls_file(fname, &sb, 1);
679                         printf("%s\n", lname);
680                         xfree(lname);
681                 } else {
682                         /* XXX - multicolumn display would be nice here */
683                         printf("%s\n", fname);
684                 }
685                 xfree(fname);
686         }
687
688         if (g.gl_pathc)
689                 globfree(&g);
690
691         return (0);
692 }
693
694 static int
695 parse_args(const char **cpp, int *pflag, int *lflag, int *iflag,
696     unsigned long *n_arg, char **path1, char **path2)
697 {
698         const char *cmd, *cp = *cpp;
699         char *cp2;
700         int base = 0;
701         long l;
702         int i, cmdnum;
703
704         /* Skip leading whitespace */
705         cp = cp + strspn(cp, WHITESPACE);
706
707         /* Ignore blank lines and lines which begin with comment '#' char */
708         if (*cp == '\0' || *cp == '#')
709                 return (0);
710
711         /* Check for leading '-' (disable error processing) */
712         *iflag = 0;
713         if (*cp == '-') {
714                 *iflag = 1;
715                 cp++;
716         }
717                 
718         /* Figure out which command we have */
719         for (i = 0; cmds[i].c; i++) {
720                 int cmdlen = strlen(cmds[i].c);
721
722                 /* Check for command followed by whitespace */
723                 if (!strncasecmp(cp, cmds[i].c, cmdlen) &&
724                     strchr(WHITESPACE, cp[cmdlen])) {
725                         cp += cmdlen;
726                         cp = cp + strspn(cp, WHITESPACE);
727                         break;
728                 }
729         }
730         cmdnum = cmds[i].n;
731         cmd = cmds[i].c;
732
733         /* Special case */
734         if (*cp == '!') {
735                 cp++;
736                 cmdnum = I_SHELL;
737         } else if (cmdnum == -1) {
738                 error("Invalid command.");
739                 return (-1);
740         }
741
742         /* Get arguments and parse flags */
743         *lflag = *pflag = *n_arg = 0;
744         *path1 = *path2 = NULL;
745         switch (cmdnum) {
746         case I_GET:
747         case I_PUT:
748                 if (parse_getput_flags(&cp, pflag))
749                         return(-1);
750                 /* Get first pathname (mandatory) */
751                 if (get_pathname(&cp, path1))
752                         return(-1);
753                 if (*path1 == NULL) {
754                         error("You must specify at least one path after a "
755                             "%s command.", cmd);
756                         return(-1);
757                 }
758                 /* Try to get second pathname (optional) */
759                 if (get_pathname(&cp, path2))
760                         return(-1);
761                 break;
762         case I_RENAME:
763         case I_SYMLINK:
764                 if (get_pathname(&cp, path1))
765                         return(-1);
766                 if (get_pathname(&cp, path2))
767                         return(-1);
768                 if (!*path1 || !*path2) {
769                         error("You must specify two paths after a %s "
770                             "command.", cmd);
771                         return(-1);
772                 }
773                 break;
774         case I_RM:
775         case I_MKDIR:
776         case I_RMDIR:
777         case I_CHDIR:
778         case I_LCHDIR:
779         case I_LMKDIR:
780                 /* Get pathname (mandatory) */
781                 if (get_pathname(&cp, path1))
782                         return(-1);
783                 if (*path1 == NULL) {
784                         error("You must specify a path after a %s command.",
785                             cmd);
786                         return(-1);
787                 }
788                 break;
789         case I_LS:
790                 if (parse_ls_flags(&cp, lflag))
791                         return(-1);
792                 /* Path is optional */
793                 if (get_pathname(&cp, path1))
794                         return(-1);
795                 break;
796         case I_LLS:
797         case I_SHELL:
798                 /* Uses the rest of the line */
799                 break;
800         case I_LUMASK:
801                 base = 8;
802         case I_CHMOD:
803                 base = 8;
804         case I_CHOWN:
805         case I_CHGRP:
806                 /* Get numeric arg (mandatory) */
807                 l = strtol(cp, &cp2, base);
808                 if (cp2 == cp || ((l == LONG_MIN || l == LONG_MAX) &&
809                     errno == ERANGE) || l < 0) {
810                         error("You must supply a numeric argument "
811                             "to the %s command.", cmd);
812                         return(-1);
813                 }
814                 cp = cp2;
815                 *n_arg = l;
816                 if (cmdnum == I_LUMASK && strchr(WHITESPACE, *cp))
817                         break;
818                 if (cmdnum == I_LUMASK || !strchr(WHITESPACE, *cp)) {
819                         error("You must supply a numeric argument "
820                             "to the %s command.", cmd);
821                         return(-1);
822                 }
823                 cp += strspn(cp, WHITESPACE);
824
825                 /* Get pathname (mandatory) */
826                 if (get_pathname(&cp, path1))
827                         return(-1);
828                 if (*path1 == NULL) {
829                         error("You must specify a path after a %s command.",
830                             cmd);
831                         return(-1);
832                 }
833                 break;
834         case I_QUIT:
835         case I_PWD:
836         case I_LPWD:
837         case I_HELP:
838         case I_VERSION:
839         case I_PROGRESS:
840                 break;
841         default:
842                 fatal("Command not implemented");
843         }
844
845         *cpp = cp;
846         return(cmdnum);
847 }
848
849 static int
850 parse_dispatch_command(struct sftp_conn *conn, const char *cmd, char **pwd,
851     int err_abort)
852 {
853         char *path1, *path2, *tmp;
854         int pflag, lflag, iflag, cmdnum, i;
855         unsigned long n_arg;
856         Attrib a, *aa;
857         char path_buf[MAXPATHLEN];
858         int err = 0;
859         glob_t g;
860
861         path1 = path2 = NULL;
862         cmdnum = parse_args(&cmd, &pflag, &lflag, &iflag, &n_arg,
863             &path1, &path2);
864
865         if (iflag != 0)
866                 err_abort = 0;
867
868         memset(&g, 0, sizeof(g));
869
870         /* Perform command */
871         switch (cmdnum) {
872         case 0:
873                 /* Blank line */
874                 break;
875         case -1:
876                 /* Unrecognized command */
877                 err = -1;
878                 break;
879         case I_GET:
880                 err = process_get(conn, path1, path2, *pwd, pflag);
881                 break;
882         case I_PUT:
883                 err = process_put(conn, path1, path2, *pwd, pflag);
884                 break;
885         case I_RENAME:
886                 path1 = make_absolute(path1, *pwd);
887                 path2 = make_absolute(path2, *pwd);
888                 err = do_rename(conn, path1, path2);
889                 break;
890         case I_SYMLINK:
891                 path2 = make_absolute(path2, *pwd);
892                 err = do_symlink(conn, path1, path2);
893                 break;
894         case I_RM:
895                 path1 = make_absolute(path1, *pwd);
896                 remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g);
897                 for (i = 0; g.gl_pathv[i]; i++) {
898                         printf("Removing %s\n", g.gl_pathv[i]);
899                         err = do_rm(conn, g.gl_pathv[i]);
900                         if (err != 0 && err_abort)
901                                 break;
902                 }
903                 break;
904         case I_MKDIR:
905                 path1 = make_absolute(path1, *pwd);
906                 attrib_clear(&a);
907                 a.flags |= SSH2_FILEXFER_ATTR_PERMISSIONS;
908                 a.perm = 0777;
909                 err = do_mkdir(conn, path1, &a);
910                 break;
911         case I_RMDIR:
912                 path1 = make_absolute(path1, *pwd);
913                 err = do_rmdir(conn, path1);
914                 break;
915         case I_CHDIR:
916                 path1 = make_absolute(path1, *pwd);
917                 if ((tmp = do_realpath(conn, path1)) == NULL) {
918                         err = 1;
919                         break;
920                 }
921                 if ((aa = do_stat(conn, tmp, 0)) == NULL) {
922                         xfree(tmp);
923                         err = 1;
924                         break;
925                 }
926                 if (!(aa->flags & SSH2_FILEXFER_ATTR_PERMISSIONS)) {
927                         error("Can't change directory: Can't check target");
928                         xfree(tmp);
929                         err = 1;
930                         break;
931                 }
932                 if (!S_ISDIR(aa->perm)) {
933                         error("Can't change directory: \"%s\" is not "
934                             "a directory", tmp);
935                         xfree(tmp);
936                         err = 1;
937                         break;
938                 }
939                 xfree(*pwd);
940                 *pwd = tmp;
941                 break;
942         case I_LS:
943                 if (!path1) {
944                         do_globbed_ls(conn, *pwd, *pwd, lflag);
945                         break;
946                 }
947
948                 /* Strip pwd off beginning of non-absolute paths */
949                 tmp = NULL;
950                 if (*path1 != '/')
951                         tmp = *pwd;
952
953                 path1 = make_absolute(path1, *pwd);
954                 err = do_globbed_ls(conn, path1, tmp, lflag);
955                 break;
956         case I_LCHDIR:
957                 if (chdir(path1) == -1) {
958                         error("Couldn't change local directory to "
959                             "\"%s\": %s", path1, strerror(errno));
960                         err = 1;
961                 }
962                 break;
963         case I_LMKDIR:
964                 if (mkdir(path1, 0777) == -1) {
965                         error("Couldn't create local directory "
966                             "\"%s\": %s", path1, strerror(errno));
967                         err = 1;
968                 }
969                 break;
970         case I_LLS:
971                 local_do_ls(cmd);
972                 break;
973         case I_SHELL:
974                 local_do_shell(cmd);
975                 break;
976         case I_LUMASK:
977                 umask(n_arg);
978                 printf("Local umask: %03lo\n", n_arg);
979                 break;
980         case I_CHMOD:
981                 path1 = make_absolute(path1, *pwd);
982                 attrib_clear(&a);
983                 a.flags |= SSH2_FILEXFER_ATTR_PERMISSIONS;
984                 a.perm = n_arg;
985                 remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g);
986                 for (i = 0; g.gl_pathv[i]; i++) {
987                         printf("Changing mode on %s\n", g.gl_pathv[i]);
988                         err = do_setstat(conn, g.gl_pathv[i], &a);
989                         if (err != 0 && err_abort)
990                                 break;
991                 }
992                 break;
993         case I_CHOWN:
994         case I_CHGRP:
995                 path1 = make_absolute(path1, *pwd);
996                 remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g);
997                 for (i = 0; g.gl_pathv[i]; i++) {
998                         if (!(aa = do_stat(conn, g.gl_pathv[i], 0))) {
999                                 if (err != 0 && err_abort)
1000                                         break;
1001                                 else
1002                                         continue;
1003                         }
1004                         if (!(aa->flags & SSH2_FILEXFER_ATTR_UIDGID)) {
1005                                 error("Can't get current ownership of "
1006                                     "remote file \"%s\"", g.gl_pathv[i]);
1007                                 if (err != 0 && err_abort)
1008                                         break;
1009                                 else
1010                                         continue;
1011                         }
1012                         aa->flags &= SSH2_FILEXFER_ATTR_UIDGID;
1013                         if (cmdnum == I_CHOWN) {
1014                                 printf("Changing owner on %s\n", g.gl_pathv[i]);
1015                                 aa->uid = n_arg;
1016                         } else {
1017                                 printf("Changing group on %s\n", g.gl_pathv[i]);
1018                                 aa->gid = n_arg;
1019                         }
1020                         err = do_setstat(conn, g.gl_pathv[i], aa);
1021                         if (err != 0 && err_abort)
1022                                 break;
1023                 }
1024                 break;
1025         case I_PWD:
1026                 printf("Remote working directory: %s\n", *pwd);
1027                 break;
1028         case I_LPWD:
1029                 if (!getcwd(path_buf, sizeof(path_buf))) {
1030                         error("Couldn't get local cwd: %s", strerror(errno));
1031                         err = -1;
1032                         break;
1033                 }
1034                 printf("Local working directory: %s\n", path_buf);
1035                 break;
1036         case I_QUIT:
1037                 /* Processed below */
1038                 break;
1039         case I_HELP:
1040                 help();
1041                 break;
1042         case I_VERSION:
1043                 printf("SFTP protocol version %u\n", sftp_proto_version(conn));
1044                 break;
1045         case I_PROGRESS:
1046                 showprogress = !showprogress;
1047                 if (showprogress)
1048                         printf("Progress meter enabled\n");
1049                 else
1050                         printf("Progress meter disabled\n");
1051                 break;
1052         default:
1053                 fatal("%d is not implemented", cmdnum);
1054         }
1055
1056         if (g.gl_pathc)
1057                 globfree(&g);
1058         if (path1)
1059                 xfree(path1);
1060         if (path2)
1061                 xfree(path2);
1062
1063         /* If an unignored error occurs in batch mode we should abort. */
1064         if (err_abort && err != 0)
1065                 return (-1);
1066         else if (cmdnum == I_QUIT)
1067                 return (1);
1068
1069         return (0);
1070 }
1071
1072 int
1073 interactive_loop(int fd_in, int fd_out, char *file1, char *file2)
1074 {
1075         char *pwd;
1076         char *dir = NULL;
1077         char cmd[2048];
1078         struct sftp_conn *conn;
1079         int err;
1080
1081         conn = do_init(fd_in, fd_out, copy_buffer_len, num_requests);
1082         if (conn == NULL)
1083                 fatal("Couldn't initialise connection to server");
1084
1085         pwd = do_realpath(conn, ".");
1086         if (pwd == NULL)
1087                 fatal("Need cwd");
1088
1089         if (file1 != NULL) {
1090                 dir = xstrdup(file1);
1091                 dir = make_absolute(dir, pwd);
1092
1093                 if (remote_is_dir(conn, dir) && file2 == NULL) {
1094                         printf("Changing to: %s\n", dir);
1095                         snprintf(cmd, sizeof cmd, "cd \"%s\"", dir);
1096                         if (parse_dispatch_command(conn, cmd, &pwd, 1) != 0)
1097                                 return (-1);
1098                 } else {
1099                         if (file2 == NULL)
1100                                 snprintf(cmd, sizeof cmd, "get %s", dir);
1101                         else
1102                                 snprintf(cmd, sizeof cmd, "get %s %s", dir,
1103                                     file2);
1104
1105                         err = parse_dispatch_command(conn, cmd, &pwd, 1);
1106                         xfree(dir);
1107                         xfree(pwd);
1108                         return (err);
1109                 }
1110                 xfree(dir);
1111         }
1112
1113 #if HAVE_SETVBUF
1114         setvbuf(stdout, NULL, _IOLBF, 0);
1115         setvbuf(infile, NULL, _IOLBF, 0);
1116 #else
1117         setlinebuf(stdout);
1118         setlinebuf(infile);
1119 #endif
1120
1121         err = 0;
1122         for (;;) {
1123                 char *cp;
1124
1125                 printf("sftp> ");
1126
1127                 /* XXX: use libedit */
1128                 if (fgets(cmd, sizeof(cmd), infile) == NULL) {
1129                         printf("\n");
1130                         break;
1131                 } else if (infile != stdin) /* Bluff typing */
1132                         printf("%s", cmd);
1133
1134                 cp = strrchr(cmd, '\n');
1135                 if (cp)
1136                         *cp = '\0';
1137
1138                 err = parse_dispatch_command(conn, cmd, &pwd, infile != stdin);
1139                 if (err != 0)
1140                         break;
1141         }
1142         xfree(pwd);
1143
1144         /* err == 1 signifies normal "quit" exit */
1145         return (err >= 0 ? 0 : -1);
1146 }
1147
This page took 0.194429 seconds and 3 git commands to generate.