summaryrefslogtreecommitdiffstats
path: root/scp.c
diff options
context:
space:
mode:
authordjm@openbsd.org <djm@openbsd.org>2021-08-10 01:49:31 +0200
committerDamien Miller <djm@mindrot.org>2021-08-10 04:47:46 +0200
commit2f7a3b51cef689ad9e93d0c6c17db5a194eb5555 (patch)
tree4e2907a57f676a8fbfe957fc9ed80a8739fd0609 /scp.c
parentupstream: SFTP protocol extension to allow the server to expand (diff)
downloadopenssh-2f7a3b51cef689ad9e93d0c6c17db5a194eb5555.tar.xz
openssh-2f7a3b51cef689ad9e93d0c6c17db5a194eb5555.zip
upstream: make scp in SFTP mode try to use relative paths as much
as possible. Previosuly, it would try to make relative and ~/-rooted paths absolute before requesting transfers. prompted by and much discussion deraadt@ ok markus@ OpenBSD-Commit-ID: 46639d382ea99546a4914b545fa7b00fa1be5566
Diffstat (limited to 'scp.c')
-rw-r--r--scp.c97
1 files changed, 26 insertions, 71 deletions
diff --git a/scp.c b/scp.c
index a0377c6c3..cb8d049b8 100644
--- a/scp.c
+++ b/scp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: scp.c,v 1.227 2021/08/09 23:47:44 djm Exp $ */
+/* $OpenBSD: scp.c,v 1.228 2021/08/09 23:49:31 djm Exp $ */
/*
* scp - secure remote copy. This is basically patched BSD rcp which
* uses ssh to do the data transfer (instead of using rcmd).
@@ -435,10 +435,10 @@ void tolocal(int, char *[], enum scp_mode_e, char *sftp_direct);
void toremote(int, char *[], enum scp_mode_e, char *sftp_direct);
void usage(void);
-void source_sftp(int, char *, char *, struct sftp_conn *, char **);
+void source_sftp(int, char *, char *, struct sftp_conn *);
void sink_sftp(int, char *, const char *, struct sftp_conn *);
void throughlocal_sftp(struct sftp_conn *, struct sftp_conn *,
- char *, char *, char **);
+ char *, char *);
int
main(int argc, char **argv)
@@ -982,7 +982,6 @@ toremote(int argc, char **argv, enum scp_mode_e mode, char *sftp_direct)
{
char *suser = NULL, *host = NULL, *src = NULL;
char *bp, *tuser, *thost, *targ;
- char *remote_path = NULL;
int sport = -1, tport = -1;
struct sftp_conn *conn = NULL, *conn2 = NULL;
arglist alist;
@@ -1056,8 +1055,7 @@ toremote(int argc, char **argv, enum scp_mode_e mode, char *sftp_direct)
}
debug3_f("destination in %d out %d pid %ld",
remin2, remout2, (long)do_cmd_pid2);
- throughlocal_sftp(conn2, conn, src, targ,
- &remote_path);
+ throughlocal_sftp(conn2, conn, src, targ);
(void) close(remin2);
(void) close(remout2);
remin2 = remout2 = -1;
@@ -1142,8 +1140,7 @@ toremote(int argc, char **argv, enum scp_mode_e mode, char *sftp_direct)
}
/* The protocol */
- source_sftp(1, argv[i], targ, conn,
- &remote_path);
+ source_sftp(1, argv[i], targ, conn);
continue;
}
/* SCP */
@@ -1161,10 +1158,8 @@ toremote(int argc, char **argv, enum scp_mode_e mode, char *sftp_direct)
}
}
out:
- if (mode == MODE_SFTP) {
- free(remote_path);
+ if (mode == MODE_SFTP)
free(conn);
- }
free(tuser);
free(thost);
free(targ);
@@ -1253,46 +1248,30 @@ tolocal(int argc, char **argv, enum scp_mode_e mode, char *sftp_direct)
free(src);
}
-/* Canonicalise a remote path, handling ~ by assuming cwd is the homedir */
+/* Prepare remote path, handling ~ by assuming cwd is the homedir */
static char *
-absolute_remote_path(struct sftp_conn *conn, const char *path,
- const char *remote_path)
+prepare_remote_path(struct sftp_conn *conn, const char *path)
{
- char *ret;
-
- if (can_expand_path(conn))
- return do_expand_path(conn, path);
-
/* Handle ~ prefixed paths */
if (*path != '~')
- ret = xstrdup(path);
- else {
- if (strcmp(path, "~") == 0)
- ret = xstrdup("");
- else if (strncmp(path, "~/", 2) == 0)
- ret = xstrdup(path + 2);
- else {
- /* XXX could be supported with protocol extension */
- error("~user paths are not currently supported");
- return NULL;
- }
- }
- return make_absolute(ret, remote_path);
+ return xstrdup(path);
+ if (*path == '\0' || strcmp(path, "~") == 0)
+ return xstrdup(".");
+ if (strncmp(path, "~/", 2) == 0)
+ return xstrdup(path + 2);
+ if (can_expand_path(conn))
+ return do_expand_path(conn, path);
+ /* No protocol extension */
+ error("~user paths are not currently supported");
+ return NULL;
}
void
-source_sftp(int argc, char *src, char *targ,
- struct sftp_conn *conn, char **remote_path)
+source_sftp(int argc, char *src, char *targ, struct sftp_conn *conn)
{
char *target = NULL, *filename = NULL, *abs_dst = NULL;
int target_is_dir;
- if (*remote_path == NULL) {
- *remote_path = do_realpath(conn, ".");
- if (*remote_path == NULL)
- fatal("Unable to determine remote working directory");
- }
-
if ((filename = basename(src)) == NULL)
fatal("basename %s: %s", src, strerror(errno));
@@ -1300,7 +1279,7 @@ source_sftp(int argc, char *src, char *targ,
* No need to glob here - the local shell already took care of
* the expansions
*/
- if ((target = absolute_remote_path(conn, targ, *remote_path)) == NULL)
+ if ((target = prepare_remote_path(conn, targ)) == NULL)
cleanup_exit(255);
target_is_dir = remote_is_dir(conn, target);
if (targetshouldbedirectory && !target_is_dir) {
@@ -1495,7 +1474,7 @@ sink_sftp(int argc, char *dst, const char *src, struct sftp_conn *conn)
char *abs_src = NULL;
char *abs_dst = NULL;
glob_t g;
- char *filename, *tmp = NULL, *remote_path = NULL;
+ char *filename, *tmp = NULL;
int i, r, err = 0;
memset(&g, 0, sizeof(g));
@@ -1504,19 +1483,10 @@ sink_sftp(int argc, char *dst, const char *src, struct sftp_conn *conn)
* expansions
*/
- remote_path = do_realpath(conn, ".");
- if (remote_path == NULL) {
- error("Could not obtain remote working directory");
- /* TODO - gracefully degrade by using relative paths ? */
- err = -1;
- goto out;
- }
-
- if ((abs_src = absolute_remote_path(conn, src, remote_path)) == NULL) {
+ if ((abs_src = prepare_remote_path(conn, src)) == NULL) {
err = -1;
goto out;
}
- free(remote_path);
debug3_f("copying remote %s to local %s", abs_src, dst);
if ((r = remote_glob(conn, abs_src, GLOB_MARK, NULL, &g)) != 0) {
@@ -1901,34 +1871,19 @@ screwup:
void
throughlocal_sftp(struct sftp_conn *from, struct sftp_conn *to,
- char *src, char *targ, char **to_remote_path)
+ char *src, char *targ)
{
char *target = NULL, *filename = NULL, *abs_dst = NULL;
- char *abs_src = NULL, *tmp = NULL, *from_remote_path;
+ char *abs_src = NULL, *tmp = NULL;
glob_t g;
int i, r, targetisdir, err = 0;
- if (*to_remote_path == NULL) {
- *to_remote_path = do_realpath(to, ".");
- if (*to_remote_path == NULL) {
- fatal("Unable to determine destination remote "
- "working directory");
- }
- }
-
- if ((from_remote_path = do_realpath(from, ".")) == NULL) {
- fatal("Unable to determine source remote "
- "working directory");
- }
-
if ((filename = basename(src)) == NULL)
fatal("basename %s: %s", src, strerror(errno));
- if ((abs_src = absolute_remote_path(from, src,
- from_remote_path)) == NULL ||
- (target = absolute_remote_path(to, targ, *to_remote_path)) == NULL)
+ if ((abs_src = prepare_remote_path(from, src)) == NULL ||
+ (target = prepare_remote_path(to, targ)) == NULL)
cleanup_exit(255);
- free(from_remote_path);
memset(&g, 0, sizeof(g));
targetisdir = remote_is_dir(to, target);