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

📄 sftp-int.c

📁 C++编写
💻 C
📖 第 1 页 / 共 2 页
字号:
static intsdirent_comp(const void *aa, const void *bb){	SFTP_DIRENT *a = *(SFTP_DIRENT **)aa;	SFTP_DIRENT *b = *(SFTP_DIRENT **)bb;	return (strcmp(a->filename, b->filename));}/* sftp ls.1 replacement for directories */static intdo_ls_dir(struct sftp_conn *conn, char *path, char *strip_path, int lflag){	int n;	SFTP_DIRENT **d;	if ((n = do_readdir(conn, path, &d)) != 0)		return (n);	/* Count entries for sort */	for (n = 0; d[n] != NULL; n++)		;	qsort(d, n, sizeof(*d), sdirent_comp);	for (n = 0; d[n] != NULL; n++) {		char *tmp, *fname;		tmp = path_append(path, d[n]->filename);		fname = path_strip(tmp, strip_path);		xfree(tmp);		if (lflag) {			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 {			/* XXX - multicolumn display would be nice here */			printf("%s\n", fname);		}		xfree(fname);	}	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;	Attrib *a;	struct stat sb;	memset(&g, 0, sizeof(g));	if (remote_glob(conn, path, GLOB_MARK|GLOB_NOCHECK|GLOB_BRACE,	    NULL, &g)) {		error("Can't ls: \"%s\" not found", path);		return (-1);	}	/*	 * If the glob returns a single match, which is the same as the	 * input glob, and it is a directory, then just list its contents	 */	if (g.gl_pathc == 1 &&	    strncmp(path, g.gl_pathv[0], strlen(g.gl_pathv[0]) - 1) == 0) {		if ((a = do_lstat(conn, path, 1)) == NULL) {			globfree(&g);	    		return (-1);		}		if ((a->flags & SSH2_FILEXFER_ATTR_PERMISSIONS) &&		    S_ISDIR(a->perm)) {			globfree(&g);			return (do_ls_dir(conn, path, strip_path, lflag));		}	}	for (i = 0; g.gl_pathv[i]; i++) {		char *fname, *lname;		fname = path_strip(g.gl_pathv[i], strip_path);		if (lflag) {			/*			 * 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));			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 {			/* XXX - multicolumn display would be nice here */			printf("%s\n", fname);		}		xfree(fname);	}	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]; i++) {			printf("Removing %s\n", g.gl_pathv[i]);			err = do_rm(conn, g.gl_pathv[i]);			if (err != 0 && err_abort)				break;		}		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_globbed_ls(conn, *pwd, *pwd, lflag);			break;		}		/* Strip pwd off beginning of non-absolute paths */		tmp = NULL;		if (*path1 != '/')			tmp = *pwd;		path1 = make_absolute(path1, *pwd);		err = do_globbed_ls(conn, path1, tmp, lflag);		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]);			err = do_setstat(conn, g.gl_pathv[i], &a);			if (err != 0 && err_abort)				break;		}		break;	case I_CHOWN:	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))) {				if (err != 0 && err_abort)					break;				else					continue;			}			if (!(aa->flags & SSH2_FILEXFER_ATTR_UIDGID)) {				error("Can't get current ownership of "				    "remote file \"%s\"", g.gl_pathv[i]);				if (err != 0 && err_abort)					break;				else					continue;			}			aa->flags &= SSH2_FILEXFER_ATTR_UIDGID;			if (cmdnum == I_CHOWN) {				printf("Changing owner on %s\n", g.gl_pathv[i]);				aa->uid = n_arg;			} else {				printf("Changing group on %s\n", g.gl_pathv[i]);				aa->gid = n_arg;			}			err = do_setstat(conn, g.gl_pathv[i], aa);			if (err != 0 && err_abort)				break;		}		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));			err = -1;			break;		}		printf("Local working directory: %s\n", path_buf);		break;	case I_QUIT:		/* Processed below */		break;	case I_HELP:		help();		break;	case I_VERSION:		printf("SFTP protocol version %u\n", sftp_proto_version(conn));		break;	case I_PROGRESS:		showprogress = !showprogress;		if (showprogress)			printf("Progress meter enabled\n");		else			printf("Progress meter disabled\n");		break;	default:		fatal("%d is not implemented", cmdnum);	}	if (g.gl_pathc)		globfree(&g);	if (path1)		xfree(path1);	if (path2)		xfree(path2);	/* If an unignored error occurs in batch mode we should abort. */	if (err_abort && err != 0)		return (-1);	else if (cmdnum == I_QUIT)		return (1);	return (0);}intinteractive_loop(int fd_in, int fd_out, char *file1, char *file2){	char *pwd;	char *dir = NULL;	char cmd[2048];	struct sftp_conn *conn;	int err;	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);			if (parse_dispatch_command(conn, cmd, &pwd, 1) != 0)				return (-1);		} else {			if (file2 == NULL)				snprintf(cmd, sizeof cmd, "get %s", dir);			else				snprintf(cmd, sizeof cmd, "get %s %s", dir,				    file2);			err = parse_dispatch_command(conn, cmd, &pwd, 1);			xfree(dir);			xfree(pwd);			return (err);		}		xfree(dir);	}	setvbuf(stdout, NULL, _IOLBF, 0);	setvbuf(infile, NULL, _IOLBF, 0);	err = 0;	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';		err = parse_dispatch_command(conn, cmd, &pwd, infile != stdin);		if (err != 0)			break;	}	xfree(pwd);	/* err == 1 signifies normal "quit" exit */	return (err >= 0 ? 0 : -1);}

⌨️ 快捷键说明

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