⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sftp-int.c

📁 OpenSSL Source code for SFTP, SSH, and many others
💻 C
📖 第 1 页 / 共 2 页
字号:
			if (infer_path(g.gl_pathv[0], &abs_dst)) {				err = -1;				goto out;			}			abs_dst = make_absolute(abs_dst, pwd);		}		printf("Uploading %s to %s\n", g.gl_pathv[0], abs_dst);		err = do_upload(conn, g.gl_pathv[0], abs_dst, pflag);		goto out;	}	/* Multiple matches, dst may be directory or unspecified */	if (tmp_dst && !remote_is_dir(conn, tmp_dst)) {		error("Multiple files match, but \"%s\" is not a directory",		    tmp_dst);		err = -1;		goto out;	}	for (i = 0; g.gl_pathv[i]; i++) {		if (infer_path(g.gl_pathv[i], &tmp)) {			err = -1;			goto out;		}		if (tmp_dst) {			abs_dst = path_append(tmp_dst, tmp);			xfree(tmp);		} else			abs_dst = make_absolute(tmp, pwd);		printf("Uploading %s to %s\n", g.gl_pathv[i], abs_dst);		if (do_upload(conn, g.gl_pathv[i], abs_dst, pflag) == -1)			err = -1;	}out:	if (abs_dst)		xfree(abs_dst);	if (tmp_dst)		xfree(tmp_dst);	return(err);}static intparse_args(const char **cpp, int *pflag, unsigned long *n_arg,    char **path1, char **path2){	const char *cmd, *cp = *cpp;	char *cp2;	int base = 0;	long l;	int i, cmdnum;	/* Skip leading whitespace */	cp = cp + strspn(cp, WHITESPACE);	/* Ignore blank lines */	if (!*cp)		return(-1);	/* Figure out which command we have */	for (i = 0; cmds[i].c; i++) {		int cmdlen = strlen(cmds[i].c);		/* Check for command followed by whitespace */		if (!strncasecmp(cp, cmds[i].c, cmdlen) &&		    strchr(WHITESPACE, cp[cmdlen])) {			cp += cmdlen;			cp = cp + strspn(cp, WHITESPACE);			break;		}	}	cmdnum = cmds[i].n;	cmd = cmds[i].c;	/* Special case */	if (*cp == '!') {		cp++;		cmdnum = I_SHELL;	} else if (cmdnum == -1) {		error("Invalid command.");		return(-1);	}	/* Get arguments and parse flags */	*pflag = *n_arg = 0;	*path1 = *path2 = NULL;	switch (cmdnum) {	case I_GET:	case I_PUT:		if (parse_getput_flags(&cp, pflag))			return(-1);		/* Get first pathname (mandatory) */		if (get_pathname(&cp, path1))			return(-1);		if (*path1 == NULL) {			error("You must specify at least one path after a "			    "%s command.", cmd);			return(-1);		}		/* Try to get second pathname (optional) */		if (get_pathname(&cp, path2))			return(-1);		break;	case I_RENAME:	case I_SYMLINK:		if (get_pathname(&cp, path1))			return(-1);		if (get_pathname(&cp, path2))			return(-1);		if (!*path1 || !*path2) {			error("You must specify two paths after a %s "			    "command.", cmd);			return(-1);		}		break;	case I_RM:	case I_MKDIR:	case I_RMDIR:	case I_CHDIR:	case I_LCHDIR:	case I_LMKDIR:		/* Get pathname (mandatory) */		if (get_pathname(&cp, path1))			return(-1);		if (*path1 == NULL) {			error("You must specify a path after a %s command.",			    cmd);			return(-1);		}		break;	case I_LS:		/* Path is optional */		if (get_pathname(&cp, path1))			return(-1);		break;	case I_LLS:	case I_SHELL:		/* Uses the rest of the line */		break;	case I_LUMASK:		base = 8;	case I_CHMOD:		base = 8;	case I_CHOWN:	case I_CHGRP:		/* Get numeric arg (mandatory) */		l = strtol(cp, &cp2, base);		if (cp2 == cp || ((l == LONG_MIN || l == LONG_MAX) &&		    errno == ERANGE) || l < 0) {			error("You must supply a numeric argument "			    "to the %s command.", cmd);			return(-1);		}		cp = cp2;		*n_arg = l;		if (cmdnum == I_LUMASK && strchr(WHITESPACE, *cp))			break;		if (cmdnum == I_LUMASK || !strchr(WHITESPACE, *cp)) {			error("You must supply a numeric argument "			    "to the %s command.", cmd);			return(-1);		}		cp += strspn(cp, WHITESPACE);		/* Get pathname (mandatory) */		if (get_pathname(&cp, path1))			return(-1);		if (*path1 == NULL) {			error("You must specify a path after a %s command.",			    cmd);			return(-1);		}		break;	case I_QUIT:	case I_PWD:	case I_LPWD:	case I_HELP:	case I_VERSION:		break;	default:		fatal("Command not implemented");	}	*cpp = cp;	return(cmdnum);}static intparse_dispatch_command(struct sftp_conn *conn, const char *cmd, char **pwd){	char *path1, *path2, *tmp;	int pflag, cmdnum, i;	unsigned long n_arg;	Attrib a, *aa;	char path_buf[MAXPATHLEN];	int err = 0;	glob_t g;	path1 = path2 = NULL;	cmdnum = parse_args(&cmd, &pflag, &n_arg, &path1, &path2);	memset(&g, 0, sizeof(g));	/* Perform command */	switch (cmdnum) {	case -1:		break;	case I_GET:		err = process_get(conn, path1, path2, *pwd, pflag);		break;	case I_PUT:		err = process_put(conn, path1, path2, *pwd, pflag);		break;	case I_RENAME:		path1 = make_absolute(path1, *pwd);		path2 = make_absolute(path2, *pwd);		err = do_rename(conn, path1, path2);		break;	case I_SYMLINK:		path2 = make_absolute(path2, *pwd);		err = do_symlink(conn, path1, path2);		break;	case I_RM:		path1 = make_absolute(path1, *pwd);		remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g);		for (i = 0; g.gl_pathv[i]; i++) {			printf("Removing %s\n", g.gl_pathv[i]);			if (do_rm(conn, g.gl_pathv[i]) == -1)				err = -1;		}		break;	case I_MKDIR:		path1 = make_absolute(path1, *pwd);		attrib_clear(&a);		a.flags |= SSH2_FILEXFER_ATTR_PERMISSIONS;		a.perm = 0777;		err = do_mkdir(conn, path1, &a);		break;	case I_RMDIR:		path1 = make_absolute(path1, *pwd);		err = do_rmdir(conn, path1);		break;	case I_CHDIR:		path1 = make_absolute(path1, *pwd);		if ((tmp = do_realpath(conn, path1)) == NULL) {			err = 1;			break;		}		if ((aa = do_stat(conn, tmp, 0)) == NULL) {			xfree(tmp);			err = 1;			break;		}		if (!(aa->flags & SSH2_FILEXFER_ATTR_PERMISSIONS)) {			error("Can't change directory: Can't check target");			xfree(tmp);			err = 1;			break;		}		if (!S_ISDIR(aa->perm)) {			error("Can't change directory: \"%s\" is not "			    "a directory", tmp);			xfree(tmp);			err = 1;			break;		}		xfree(*pwd);		*pwd = tmp;		break;	case I_LS:		if (!path1) {			do_ls(conn, *pwd);			break;		}		path1 = make_absolute(path1, *pwd);		if ((tmp = do_realpath(conn, path1)) == NULL)			break;		xfree(path1);		path1 = tmp;		if ((aa = do_stat(conn, path1, 0)) == NULL)			break;		if ((aa->flags & SSH2_FILEXFER_ATTR_PERMISSIONS) &&		    !S_ISDIR(aa->perm)) {			error("Can't ls: \"%s\" is not a directory", path1);			break;		}		do_ls(conn, path1);		break;	case I_LCHDIR:		if (chdir(path1) == -1) {			error("Couldn't change local directory to "			    "\"%s\": %s", path1, strerror(errno));			err = 1;		}		break;	case I_LMKDIR:		if (mkdir(path1, 0777) == -1) {			error("Couldn't create local directory "			    "\"%s\": %s", path1, strerror(errno));			err = 1;		}		break;	case I_LLS:		local_do_ls(cmd);		break;	case I_SHELL:		local_do_shell(cmd);		break;	case I_LUMASK:		umask(n_arg);		printf("Local umask: %03lo\n", n_arg);		break;	case I_CHMOD:		path1 = make_absolute(path1, *pwd);		attrib_clear(&a);		a.flags |= SSH2_FILEXFER_ATTR_PERMISSIONS;		a.perm = n_arg;		remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g);		for (i = 0; g.gl_pathv[i]; i++) {			printf("Changing mode on %s\n", g.gl_pathv[i]);			do_setstat(conn, g.gl_pathv[i], &a);		}		break;	case I_CHOWN:		path1 = make_absolute(path1, *pwd);		remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g);		for (i = 0; g.gl_pathv[i]; i++) {			if (!(aa = do_stat(conn, g.gl_pathv[i], 0)))				continue;			if (!(aa->flags & SSH2_FILEXFER_ATTR_UIDGID)) {				error("Can't get current ownership of "				    "remote file \"%s\"", g.gl_pathv[i]);				continue;			}			printf("Changing owner on %s\n", g.gl_pathv[i]);			aa->flags &= SSH2_FILEXFER_ATTR_UIDGID;			aa->uid = n_arg;			do_setstat(conn, g.gl_pathv[i], aa);		}		break;	case I_CHGRP:		path1 = make_absolute(path1, *pwd);		remote_glob(conn, path1, GLOB_NOCHECK, NULL, &g);		for (i = 0; g.gl_pathv[i]; i++) {			if (!(aa = do_stat(conn, g.gl_pathv[i], 0)))				continue;			if (!(aa->flags & SSH2_FILEXFER_ATTR_UIDGID)) {				error("Can't get current ownership of "				    "remote file \"%s\"", g.gl_pathv[i]);				continue;			}			printf("Changing group on %s\n", g.gl_pathv[i]);			aa->flags &= SSH2_FILEXFER_ATTR_UIDGID;			aa->gid = n_arg;			do_setstat(conn, g.gl_pathv[i], aa);		}		break;	case I_PWD:		printf("Remote working directory: %s\n", *pwd);		break;	case I_LPWD:		if (!getcwd(path_buf, sizeof(path_buf)))			error("Couldn't get local cwd: %s",			    strerror(errno));		else			printf("Local working directory: %s\n",			    path_buf);		break;	case I_QUIT:		return(-1);	case I_HELP:		help();		break;	case I_VERSION:		printf("SFTP protocol version %d\n", sftp_proto_version(conn));		break;	default:		fatal("%d is not implemented", cmdnum);	}	if (g.gl_pathc)		globfree(&g);	if (path1)		xfree(path1);	if (path2)		xfree(path2);	/* If an error occurs in batch mode we should abort. */	if (infile != stdin && err > 0)		return -1;	return(0);}voidinteractive_loop(int fd_in, int fd_out, char *file1, char *file2){	char *pwd;	char *dir = NULL;	char cmd[2048];	struct sftp_conn *conn;	conn = do_init(fd_in, fd_out, copy_buffer_len, num_requests);	if (conn == NULL)		fatal("Couldn't initialise connection to server");	pwd = do_realpath(conn, ".");	if (pwd == NULL)		fatal("Need cwd");	if (file1 != NULL) {		dir = xstrdup(file1);		dir = make_absolute(dir, pwd);		if (remote_is_dir(conn, dir) && file2 == NULL) {			printf("Changing to: %s\n", dir);			snprintf(cmd, sizeof cmd, "cd \"%s\"", dir);			parse_dispatch_command(conn, cmd, &pwd);		} else {			if (file2 == NULL)				snprintf(cmd, sizeof cmd, "get %s", dir);			else				snprintf(cmd, sizeof cmd, "get %s %s", dir,				    file2);			parse_dispatch_command(conn, cmd, &pwd);			xfree(dir);			return;		}		xfree(dir);	}#if HAVE_SETVBUF	setvbuf(stdout, NULL, _IOLBF, 0);	setvbuf(infile, NULL, _IOLBF, 0);#else	setlinebuf(stdout);	setlinebuf(infile);#endif	for (;;) {		char *cp;		printf("sftp> ");		/* XXX: use libedit */		if (fgets(cmd, sizeof(cmd), infile) == NULL) {			printf("\n");			break;		} else if (infile != stdin) /* Bluff typing */			printf("%s", cmd);		cp = strrchr(cmd, '\n');		if (cp)			*cp = '\0';		if (parse_dispatch_command(conn, cmd, &pwd))			break;	}	xfree(pwd);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -