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

📄 sftp.c

📁 OpenSSH 是 SSH (Secure SHell) 协议的免费开源实现。它用安全、加密的网络连接工具代替了 telnet、ftp、 rlogin、rsh 和 rcp 工具。OpenSSH 支持
💻 C
📖 第 1 页 / 共 3 页
字号:
		    dst);		err = -1;		goto out;	}	for (i = 0; g.gl_pathv[i] && !interrupted; i++) {		if (infer_path(g.gl_pathv[i], &tmp)) {			err = -1;			goto out;		}		if (g.gl_matchc == 1 && dst) {			/* If directory specified, append filename */			if (is_dir(dst)) {				if (infer_path(g.gl_pathv[0], &tmp)) {					err = 1;					goto out;				}				abs_dst = path_append(dst, tmp);				xfree(tmp);			} else				abs_dst = xstrdup(dst);		} else if (dst) {			abs_dst = path_append(dst, tmp);			xfree(tmp);		} else			abs_dst = tmp;		printf("Fetching %s to %s\n", g.gl_pathv[i], abs_dst);		if (do_download(conn, g.gl_pathv[i], abs_dst, pflag) == -1)			err = -1;		xfree(abs_dst);		abs_dst = NULL;	}out:	xfree(abs_src);	if (abs_dst)		xfree(abs_dst);	globfree(&g);	return(err);}static intprocess_put(struct sftp_conn *conn, char *src, char *dst, char *pwd, int pflag){	char *tmp_dst = NULL;	char *abs_dst = NULL;	char *tmp;	glob_t g;	int err = 0;	int i;	if (dst) {		tmp_dst = xstrdup(dst);		tmp_dst = make_absolute(tmp_dst, pwd);	}	memset(&g, 0, sizeof(g));	debug3("Looking up %s", src);	if (glob(src, 0, NULL, &g)) {		error("File \"%s\" not found.", src);		err = -1;		goto out;	}	/* If multiple matches, dst may be directory or unspecified */	if (g.gl_matchc > 1 && 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] && !interrupted; i++) {		if (!is_reg(g.gl_pathv[i])) {			error("skipping non-regular file %s",			    g.gl_pathv[i]);			continue;		}		if (infer_path(g.gl_pathv[i], &tmp)) {			err = -1;			goto out;		}		if (g.gl_matchc == 1 && tmp_dst) {			/* If directory specified, append filename */			if (remote_is_dir(conn, tmp_dst)) {				if (infer_path(g.gl_pathv[0], &tmp)) {					err = 1;					goto out;				}				abs_dst = path_append(tmp_dst, tmp);				xfree(tmp);			} else				abs_dst = xstrdup(tmp_dst);		} else 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);	globfree(&g);	return(err);}static intsdirent_comp(const void *aa, const void *bb){	SFTP_DIRENT *a = *(SFTP_DIRENT **)aa;	SFTP_DIRENT *b = *(SFTP_DIRENT **)bb;	int rmul = sort_flag & LS_REVERSE_SORT ? -1 : 1;#define NCMP(a,b) (a == b ? 0 : (a < b ? 1 : -1))	if (sort_flag & LS_NAME_SORT)		return (rmul * strcmp(a->filename, b->filename));	else if (sort_flag & LS_TIME_SORT)		return (rmul * NCMP(a->a.mtime, b->a.mtime));	else if (sort_flag & LS_SIZE_SORT)		return (rmul * NCMP(a->a.size, b->a.size));	fatal("Unknown ls sort type");}/* sftp ls.1 replacement for directories */static intdo_ls_dir(struct sftp_conn *conn, char *path, char *strip_path, int lflag){	int n, c = 1, colspace = 0, columns = 1;	SFTP_DIRENT **d;	if ((n = do_readdir(conn, path, &d)) != 0)		return (n);	if (!(lflag & LS_SHORT_VIEW)) {		int m = 0, width = 80;		struct winsize ws;		char *tmp;		/* Count entries for sort and find longest filename */		for (n = 0; d[n] != NULL; n++) {			if (d[n]->filename[0] != '.' || (lflag & LS_SHOW_ALL))				m = MAX(m, strlen(d[n]->filename));		}		/* Add any subpath that also needs to be counted */		tmp = path_strip(path, strip_path);		m += strlen(tmp);		xfree(tmp);		if (ioctl(fileno(stdin), TIOCGWINSZ, &ws) != -1)			width = ws.ws_col;		columns = width / (m + 2);		columns = MAX(columns, 1);		colspace = width / columns;		colspace = MIN(colspace, width);	}	if (lflag & SORT_FLAGS) {		sort_flag = lflag & (SORT_FLAGS|LS_REVERSE_SORT);		qsort(d, n, sizeof(*d), sdirent_comp);	}	for (n = 0; d[n] != NULL && !interrupted; n++) {		char *tmp, *fname;		if (d[n]->filename[0] == '.' && !(lflag & LS_SHOW_ALL))			continue;		tmp = path_append(path, d[n]->filename);		fname = path_strip(tmp, strip_path);		xfree(tmp);		if (lflag & LS_LONG_VIEW) {			if (lflag & LS_NUMERIC_VIEW) {				char *lname;				struct stat sb;				memset(&sb, 0, sizeof(sb));				attrib_to_stat(&d[n]->a, &sb);				lname = ls_file(fname, &sb, 1);				printf("%s\n", lname);				xfree(lname);			} else				printf("%s\n", d[n]->longname);		} else {			printf("%-*s", colspace, fname);			if (c >= columns) {				printf("\n");				c = 1;			} else				c++;		}		xfree(fname);	}	if (!(lflag & LS_LONG_VIEW) && (c != 1))		printf("\n");	free_sftp_dirents(d);	return (0);}/* sftp ls.1 replacement which handles path globs */static intdo_globbed_ls(struct sftp_conn *conn, char *path, char *strip_path,    int lflag){	glob_t g;	int i, c = 1, colspace = 0, columns = 1;	Attrib *a = NULL;	memset(&g, 0, sizeof(g));	if (remote_glob(conn, path, GLOB_MARK|GLOB_NOCHECK|GLOB_BRACE,	    NULL, &g) || (g.gl_pathc && !g.gl_matchc)) {		if (g.gl_pathc)			globfree(&g);		error("Can't ls: \"%s\" not found", path);		return (-1);	}	if (interrupted)		goto out;	/*	 * If the glob returns a single match and it is a directory,	 * then just list its contents.	 */	if (g.gl_matchc == 1) {		if ((a = do_lstat(conn, g.gl_pathv[0], 1)) == NULL) {			globfree(&g);			return (-1);		}		if ((a->flags & SSH2_FILEXFER_ATTR_PERMISSIONS) &&		    S_ISDIR(a->perm)) {			int err;			err = do_ls_dir(conn, g.gl_pathv[0], strip_path, lflag);			globfree(&g);			return (err);		}	}	if (!(lflag & LS_SHORT_VIEW)) {		int m = 0, width = 80;		struct winsize ws;		/* Count entries for sort and find longest filename */		for (i = 0; g.gl_pathv[i]; i++)			m = MAX(m, strlen(g.gl_pathv[i]));		if (ioctl(fileno(stdin), TIOCGWINSZ, &ws) != -1)			width = ws.ws_col;		columns = width / (m + 2);		columns = MAX(columns, 1);		colspace = width / columns;	}	for (i = 0; g.gl_pathv[i] && !interrupted; i++, a = NULL) {		char *fname;		fname = path_strip(g.gl_pathv[i], strip_path);		if (lflag & LS_LONG_VIEW) {			char *lname;			struct stat sb;			/*			 * XXX: this is slow - 1 roundtrip per path			 * A solution to this is to fork glob() and			 * build a sftp specific version which keeps the			 * attribs (which currently get thrown away)			 * that the server returns as well as the filenames.			 */			memset(&sb, 0, sizeof(sb));			if (a == NULL)				a = do_lstat(conn, g.gl_pathv[i], 1);			if (a != NULL)				attrib_to_stat(a, &sb);			lname = ls_file(fname, &sb, 1);			printf("%s\n", lname);			xfree(lname);		} else {			printf("%-*s", colspace, fname);			if (c >= columns) {				printf("\n");				c = 1;			} else				c++;		}		xfree(fname);	}	if (!(lflag & LS_LONG_VIEW) && (c != 1))		printf("\n"); out:	if (g.gl_pathc)		globfree(&g);	return (0);}static intparse_args(const char **cpp, int *pflag, int *lflag, int *iflag,    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 and lines which begin with comment '#' char */	if (*cp == '\0' || *cp == '#')		return (0);	/* Check for leading '-' (disable error processing) */	*iflag = 0;	if (*cp == '-') {		*iflag = 1;		cp++;	}	/* 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 */	*lflag = *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:		if (parse_ls_flags(&cp, lflag))			return(-1);		/* 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:	case I_PROGRESS:		break;	default:		fatal("Command not implemented");	}	*cpp = cp;	return(cmdnum);}static intparse_dispatch_command(struct sftp_conn *conn, const char *cmd, char **pwd,    int err_abort){	char *path1, *path2, *tmp;	int pflag, lflag, iflag, 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, &lflag, &iflag, &n_arg,	    &path1, &path2);	if (iflag != 0)		err_abort = 0;	memset(&g, 0, sizeof(g));	/* Perform command */	switch (cmdnum) {	case 0:		/* Blank line */		break;	case -1:		/* Unrecognized command */		err = -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] && !interrupted; i++) {			printf("Removing %s\n", g.gl_pathv[i]);			err = do_rm(conn, g.gl_pathv[i]);			if (err != 0 && err_abort)				break;		}

⌨️ 快捷键说明

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