mirror of
				https://github.com/tmux/tmux.git
				synced 2025-10-26 04:17:01 +00:00 
			
		
		
		
	Make source-file look for files relative to the client working directory
(like load-buffer and save-buffer), from Chris Pickel. Also break the where-is-this-file code out into its own function for loadb and saveb.
This commit is contained in:
		| @@ -57,11 +57,9 @@ cmd_load_buffer_exec(struct cmd *self, struct cmdq_item *item) | |||||||
| 	struct args			*args = self->args; | 	struct args			*args = self->args; | ||||||
| 	struct cmd_load_buffer_data	*cdata; | 	struct cmd_load_buffer_data	*cdata; | ||||||
| 	struct client			*c = item->client; | 	struct client			*c = item->client; | ||||||
| 	struct session  		*s; |  | ||||||
| 	FILE				*f; | 	FILE				*f; | ||||||
| 	const char			*path, *bufname, *cwd; | 	const char			*path, *bufname; | ||||||
| 	char				*pdata, *new_pdata, *cause, *file; | 	char				*pdata, *new_pdata, *cause, *file; | ||||||
| 	char				 resolved[PATH_MAX]; |  | ||||||
| 	size_t				 psize; | 	size_t				 psize; | ||||||
| 	int				 ch, error; | 	int				 ch, error; | ||||||
|  |  | ||||||
| @@ -87,26 +85,11 @@ cmd_load_buffer_exec(struct cmd *self, struct cmdq_item *item) | |||||||
| 		return (CMD_RETURN_WAIT); | 		return (CMD_RETURN_WAIT); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (c != NULL && c->session == NULL && c->cwd != NULL) | 	file = server_client_get_path(c, path); | ||||||
| 		cwd = c->cwd; | 	f = fopen(file, "rb"); | ||||||
| 	else if (c != NULL && (s = c->session) != NULL && s->cwd != NULL) |  | ||||||
| 		cwd = s->cwd; |  | ||||||
| 	else |  | ||||||
| 		cwd = "."; |  | ||||||
|  |  | ||||||
| 	if (*path == '/') |  | ||||||
| 		file = xstrdup(path); |  | ||||||
| 	else |  | ||||||
| 		xasprintf(&file, "%s/%s", cwd, path); |  | ||||||
| 	if (realpath(file, resolved) == NULL && |  | ||||||
| 	    strlcpy(resolved, file, sizeof resolved) >= sizeof resolved) { |  | ||||||
| 		cmdq_error(item, "%s: %s", file, strerror(ENAMETOOLONG)); |  | ||||||
| 		return (CMD_RETURN_ERROR); |  | ||||||
| 	} |  | ||||||
| 	f = fopen(resolved, "rb"); |  | ||||||
| 	free(file); |  | ||||||
| 	if (f == NULL) { | 	if (f == NULL) { | ||||||
| 		cmdq_error(item, "%s: %s", resolved, strerror(errno)); | 		cmdq_error(item, "%s: %s", file, strerror(errno)); | ||||||
|  | 		free(file); | ||||||
| 		return (CMD_RETURN_ERROR); | 		return (CMD_RETURN_ERROR); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -122,13 +105,14 @@ cmd_load_buffer_exec(struct cmd *self, struct cmdq_item *item) | |||||||
| 		pdata[psize++] = ch; | 		pdata[psize++] = ch; | ||||||
| 	} | 	} | ||||||
| 	if (ferror(f)) { | 	if (ferror(f)) { | ||||||
| 		cmdq_error(item, "%s: read error", resolved); | 		cmdq_error(item, "%s: read error", file); | ||||||
| 		goto error; | 		goto error; | ||||||
| 	} | 	} | ||||||
| 	if (pdata != NULL) | 	if (pdata != NULL) | ||||||
| 		pdata[psize] = '\0'; | 		pdata[psize] = '\0'; | ||||||
|  |  | ||||||
| 	fclose(f); | 	fclose(f); | ||||||
|  | 	free(file); | ||||||
|  |  | ||||||
| 	if (paste_set(pdata, psize, bufname, &cause) != 0) { | 	if (paste_set(pdata, psize, bufname, &cause) != 0) { | ||||||
| 		cmdq_error(item, "%s", cause); | 		cmdq_error(item, "%s", cause); | ||||||
|   | |||||||
| @@ -61,11 +61,10 @@ cmd_save_buffer_exec(struct cmd *self, struct cmdq_item *item) | |||||||
| { | { | ||||||
| 	struct args		*args = self->args; | 	struct args		*args = self->args; | ||||||
| 	struct client		*c = item->client; | 	struct client		*c = item->client; | ||||||
| 	struct session          *s; |  | ||||||
| 	struct paste_buffer	*pb; | 	struct paste_buffer	*pb; | ||||||
| 	const char		*path, *bufname, *bufdata, *start, *end, *cwd; | 	const char		*path, *bufname, *bufdata, *start, *end; | ||||||
| 	const char		*flags; | 	const char		*flags; | ||||||
| 	char			*msg, *file, resolved[PATH_MAX]; | 	char			*msg, *file; | ||||||
| 	size_t			 size, used, msglen, bufsize; | 	size_t			 size, used, msglen, bufsize; | ||||||
| 	FILE			*f; | 	FILE			*f; | ||||||
|  |  | ||||||
| @@ -98,39 +97,26 @@ cmd_save_buffer_exec(struct cmd *self, struct cmdq_item *item) | |||||||
| 		goto do_print; | 		goto do_print; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (c != NULL && c->session == NULL && c->cwd != NULL) |  | ||||||
| 		cwd = c->cwd; |  | ||||||
| 	else if (c != NULL && (s = c->session) != NULL && s->cwd != NULL) |  | ||||||
| 		cwd = s->cwd; |  | ||||||
| 	else |  | ||||||
| 		cwd = "."; |  | ||||||
|  |  | ||||||
| 	flags = "wb"; | 	flags = "wb"; | ||||||
| 	if (args_has(self->args, 'a')) | 	if (args_has(self->args, 'a')) | ||||||
| 		flags = "ab"; | 		flags = "ab"; | ||||||
|  |  | ||||||
| 	if (*path == '/') | 	file = server_client_get_path(c, path); | ||||||
| 		file = xstrdup(path); | 	f = fopen(file, flags); | ||||||
| 	else |  | ||||||
| 		xasprintf(&file, "%s/%s", cwd, path); |  | ||||||
| 	if (realpath(file, resolved) == NULL && |  | ||||||
| 	    strlcpy(resolved, file, sizeof resolved) >= sizeof resolved) { |  | ||||||
| 		cmdq_error(item, "%s: %s", file, strerror(ENAMETOOLONG)); |  | ||||||
| 		return (CMD_RETURN_ERROR); |  | ||||||
| 	} |  | ||||||
| 	f = fopen(resolved, flags); |  | ||||||
| 	free(file); |  | ||||||
| 	if (f == NULL) { | 	if (f == NULL) { | ||||||
| 		cmdq_error(item, "%s: %s", resolved, strerror(errno)); | 		cmdq_error(item, "%s: %s", file, strerror(errno)); | ||||||
|  | 		free(file); | ||||||
| 		return (CMD_RETURN_ERROR); | 		return (CMD_RETURN_ERROR); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (fwrite(bufdata, 1, bufsize, f) != bufsize) { | 	if (fwrite(bufdata, 1, bufsize, f) != bufsize) { | ||||||
| 		cmdq_error(item, "%s: write error", resolved); | 		cmdq_error(item, "%s: write error", file); | ||||||
| 		fclose(f); | 		fclose(f); | ||||||
| 		return (CMD_RETURN_ERROR); | 		return (CMD_RETURN_ERROR); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	fclose(f); | 	fclose(f); | ||||||
|  | 	free(file); | ||||||
|  |  | ||||||
| 	return (CMD_RETURN_NORMAL); | 	return (CMD_RETURN_NORMAL); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ | |||||||
| #include <glob.h> | #include <glob.h> | ||||||
| #include <stdlib.h> | #include <stdlib.h> | ||||||
| #include <string.h> | #include <string.h> | ||||||
|  | #include <vis.h> | ||||||
|  |  | ||||||
| #include "tmux.h" | #include "tmux.h" | ||||||
|  |  | ||||||
| @@ -48,22 +49,35 @@ static enum cmd_retval | |||||||
| cmd_source_file_exec(struct cmd *self, struct cmdq_item *item) | cmd_source_file_exec(struct cmd *self, struct cmdq_item *item) | ||||||
| { | { | ||||||
| 	struct args		*args = self->args; | 	struct args		*args = self->args; | ||||||
|  | 	int			 quiet = args_has(args, 'q'); | ||||||
| 	struct client		*c = item->client; | 	struct client		*c = item->client; | ||||||
| 	int			 quiet; |  | ||||||
| 	struct cmdq_item	*new_item; | 	struct cmdq_item	*new_item; | ||||||
| 	enum cmd_retval		 retval; | 	enum cmd_retval		 retval; | ||||||
|  | 	char			*pattern, *tmp; | ||||||
|  | 	const char		*path = args->argv[0]; | ||||||
| 	glob_t			 g; | 	glob_t			 g; | ||||||
| 	u_int			 i; | 	u_int			 i; | ||||||
|  |  | ||||||
| 	quiet = args_has(args, 'q'); | 	if (*path == '/') | ||||||
| 	if (glob(args->argv[0], 0, NULL, &g) != 0) { | 		pattern = xstrdup(path); | ||||||
| 		if (quiet && errno == ENOENT) | 	else { | ||||||
| 			return (CMD_RETURN_NORMAL); | 		utf8_stravis(&tmp, server_client_get_cwd(c), VIS_GLOB); | ||||||
| 		cmdq_error(item, "%s: %s", args->argv[0], strerror(errno)); | 		xasprintf(&pattern, "%s/%s", tmp, path); | ||||||
| 		return (CMD_RETURN_ERROR); | 		free(tmp); | ||||||
| 	} | 	} | ||||||
|  | 	log_debug("%s: %s", __func__, pattern); | ||||||
|  |  | ||||||
| 	retval = CMD_RETURN_NORMAL; | 	retval = CMD_RETURN_NORMAL; | ||||||
|  | 	if (glob(pattern, 0, NULL, &g) != 0) { | ||||||
|  | 		if (!quiet || errno != ENOENT) { | ||||||
|  | 			cmdq_error(item, "%s: %s", path, strerror(errno)); | ||||||
|  | 			retval = CMD_RETURN_ERROR; | ||||||
|  | 		} | ||||||
|  | 		free(pattern); | ||||||
|  | 		return (retval); | ||||||
|  | 	} | ||||||
|  | 	free(pattern); | ||||||
|  |  | ||||||
| 	for (i = 0; i < (u_int)g.gl_pathc; i++) { | 	for (i = 0; i < (u_int)g.gl_pathc; i++) { | ||||||
| 		if (load_cfg(g.gl_pathv[i], c, item, quiet) != 0) | 		if (load_cfg(g.gl_pathv[i], c, item, quiet) != 0) | ||||||
| 			retval = CMD_RETURN_ERROR; | 			retval = CMD_RETURN_ERROR; | ||||||
|   | |||||||
| @@ -1655,3 +1655,32 @@ server_client_add_message(struct client *c, const char *fmt, ...) | |||||||
| 		free(msg); | 		free(msg); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* Get client working directory. */ | ||||||
|  | const char * | ||||||
|  | server_client_get_cwd(struct client *c) | ||||||
|  | { | ||||||
|  | 	struct session	*s; | ||||||
|  |  | ||||||
|  | 	if (c != NULL && c->session == NULL && c->cwd != NULL) | ||||||
|  | 		return (c->cwd); | ||||||
|  | 	if (c != NULL && (s = c->session) != NULL && s->cwd != NULL) | ||||||
|  | 		return (s->cwd); | ||||||
|  | 	return ("."); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Resolve an absolute path or relative to client working directory. */ | ||||||
|  | char * | ||||||
|  | server_client_get_path(struct client *c, const char *file) | ||||||
|  | { | ||||||
|  | 	char	*path, resolved[PATH_MAX]; | ||||||
|  |  | ||||||
|  | 	if (*file == '/') | ||||||
|  | 		path = xstrdup(file); | ||||||
|  | 	else | ||||||
|  | 		xasprintf(&path, "%s/%s", server_client_get_cwd(c), file); | ||||||
|  | 	if (realpath(path, resolved) == NULL) | ||||||
|  | 		return (path); | ||||||
|  | 	free(path); | ||||||
|  | 	return (xstrdup(resolved)); | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								tmux.h
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								tmux.h
									
									
									
									
									
								
							| @@ -1834,6 +1834,8 @@ void	 server_client_push_stdout(struct client *); | |||||||
| void	 server_client_push_stderr(struct client *); | void	 server_client_push_stderr(struct client *); | ||||||
| void printflike(2, 3) server_client_add_message(struct client *, const char *, | void printflike(2, 3) server_client_add_message(struct client *, const char *, | ||||||
| 	     ...); | 	     ...); | ||||||
|  | char	*server_client_get_path(struct client *, const char *); | ||||||
|  | const char *server_client_get_cwd(struct client *); | ||||||
|  |  | ||||||
| /* server-fn.c */ | /* server-fn.c */ | ||||||
| void	 server_fill_environ(struct session *, struct environ *); | void	 server_fill_environ(struct session *, struct environ *); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 nicm
					nicm