1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Option to log proxy setup diagnostics to the terminal.

It has three settings: on, off, and 'only until session starts'. The
idea of the last one is that if you use something like 'ssh -v' as
your proxy command, you probably wanted to see the initial SSH
connection-setup messages while you were waiting to see if the
connection would be set up successfully at all, but probably _didn't_
want a slew of diagnostics from rekeys disrupting your terminal in
mid-emacs once the session had got properly under way.

Default is off, to avoid startling people used to the old behaviour. I
wonder if I should have set it more aggressively, though.
This commit is contained in:
Simon Tatham 2015-11-22 14:33:28 +00:00
parent 297efff303
commit 7c65b9c57a
10 changed files with 63 additions and 11 deletions

View File

@ -2,12 +2,16 @@
* be_misc.c: helper functions shared between main network backends. * be_misc.c: helper functions shared between main network backends.
*/ */
#include <assert.h>
#include <string.h>
#define DEFINE_PLUG_METHOD_MACROS #define DEFINE_PLUG_METHOD_MACROS
#include "putty.h" #include "putty.h"
#include "network.h" #include "network.h"
void backend_socket_log(void *frontend, int type, SockAddr addr, int port, void backend_socket_log(void *frontend, int type, SockAddr addr, int port,
const char *error_msg, int error_code) const char *error_msg, int error_code, Conf *conf,
int session_started)
{ {
char addrbuf[256], *msg; char addrbuf[256], *msg;
@ -27,7 +31,22 @@ void backend_socket_log(void *frontend, int type, SockAddr addr, int port,
case 2: case 2:
/* Proxy-related log messages have their own identifying /* Proxy-related log messages have their own identifying
* prefix already, put on by our caller. */ * prefix already, put on by our caller. */
msg = dupstr(error_msg); {
int len, log_to_term;
/* Suffix \r\n temporarily, so we can log to the terminal. */
msg = dupprintf("%s\r\n", error_msg);
len = strlen(msg);
assert(len >= 2);
log_to_term = conf_get_int(conf, CONF_proxy_log_to_term);
if (log_to_term == AUTO)
log_to_term = session_started ? FORCE_OFF : FORCE_ON;
if (log_to_term == FORCE_ON)
from_backend(frontend, TRUE, msg, len);
msg[len-2] = '\0'; /* remove the \r\n again */
}
break; break;
default: default:
msg = NULL; /* shouldn't happen, but placate optimiser */ msg = NULL; /* shouldn't happen, but placate optimiser */

View File

@ -2072,6 +2072,15 @@ void setup_config_box(struct controlbox *b, int midsession,
HELPCTX(proxy_command), HELPCTX(proxy_command),
conf_editbox_handler, conf_editbox_handler,
I(CONF_proxy_telnet_command), I(1)); I(CONF_proxy_telnet_command), I(1));
ctrl_radiobuttons(s, "Print proxy diagnostics "
"in the terminal window", 'r', 5,
HELPCTX(proxy_main),
conf_radiobutton_handler,
I(CONF_proxy_log_to_term),
"No", I(FORCE_OFF),
"Yes", I(FORCE_ON),
"Only until session starts", I(AUTO), NULL);
} }
/* /*

View File

@ -227,7 +227,8 @@ Socket new_error_socket(const char *errmsg, Plug plug);
* Exports from be_misc.c. * Exports from be_misc.c.
*/ */
void backend_socket_log(void *frontend, int type, SockAddr addr, int port, void backend_socket_log(void *frontend, int type, SockAddr addr, int port,
const char *error_msg, int error_code); const char *error_msg, int error_code, Conf *conf,
int session_started);
typedef struct bufchain_tag bufchain; /* rest of declaration in misc.c */ typedef struct bufchain_tag bufchain; /* rest of declaration in misc.c */
void log_proxy_stderr(Plug plug, bufchain *buf, const void *vdata, int len); void log_proxy_stderr(Plug plug, bufchain *buf, const void *vdata, int len);

View File

@ -279,9 +279,9 @@ enum {
* three-way settings whose values are `always yes', `always * three-way settings whose values are `always yes', `always
* no', and `decide by some more complex automated means'. This * no', and `decide by some more complex automated means'. This
* is true of line discipline options (local echo and line * is true of line discipline options (local echo and line
* editing), proxy DNS, Close On Exit, and SSH server bug * editing), proxy DNS, proxy terminal logging, Close On Exit, and
* workarounds. Accordingly I supply a single enum here to deal * SSH server bug workarounds. Accordingly I supply a single enum
* with them all. * here to deal with them all.
*/ */
FORCE_ON, FORCE_OFF, AUTO FORCE_ON, FORCE_OFF, AUTO
}; };
@ -681,6 +681,7 @@ void cleanup_exit(int);
X(STR, NONE, proxy_username) \ X(STR, NONE, proxy_username) \
X(STR, NONE, proxy_password) \ X(STR, NONE, proxy_password) \
X(STR, NONE, proxy_telnet_command) \ X(STR, NONE, proxy_telnet_command) \
X(INT, NONE, proxy_log_to_term) \
/* SSH options */ \ /* SSH options */ \
X(STR, NONE, remote_cmd) \ X(STR, NONE, remote_cmd) \
X(STR, NONE, remote_cmd2) /* fallback if remote_cmd fails; never loaded or saved */ \ X(STR, NONE, remote_cmd2) /* fallback if remote_cmd fails; never loaded or saved */ \

12
raw.c
View File

@ -25,7 +25,9 @@ typedef struct raw_backend_data {
int closed_on_socket_error; int closed_on_socket_error;
int bufsize; int bufsize;
void *frontend; void *frontend;
int sent_console_eof, sent_socket_eof; int sent_console_eof, sent_socket_eof, session_started;
Conf *conf;
} *Raw; } *Raw;
static void raw_size(void *handle, int width, int height); static void raw_size(void *handle, int width, int height);
@ -41,7 +43,7 @@ static void raw_log(Plug plug, int type, SockAddr addr, int port,
{ {
Raw raw = (Raw) plug; Raw raw = (Raw) plug;
backend_socket_log(raw->frontend, type, addr, port, backend_socket_log(raw->frontend, type, addr, port,
error_msg, error_code); error_msg, error_code, raw->conf, raw->session_started);
} }
static void raw_check_close(Raw raw) static void raw_check_close(Raw raw)
@ -97,6 +99,9 @@ static int raw_receive(Plug plug, int urgent, char *data, int len)
{ {
Raw raw = (Raw) plug; Raw raw = (Raw) plug;
c_write(raw, data, len); c_write(raw, data, len);
/* We count 'session start', for proxy logging purposes, as being
* when data is received from the network and printed. */
raw->session_started = TRUE;
return 1; return 1;
} }
@ -138,6 +143,8 @@ static const char *raw_init(void *frontend_handle, void **backend_handle,
*backend_handle = raw; *backend_handle = raw;
raw->sent_console_eof = raw->sent_socket_eof = FALSE; raw->sent_console_eof = raw->sent_socket_eof = FALSE;
raw->bufsize = 0; raw->bufsize = 0;
raw->session_started = FALSE;
raw->conf = conf_copy(conf);
raw->frontend = frontend_handle; raw->frontend = frontend_handle;
@ -184,6 +191,7 @@ static void raw_free(void *handle)
if (raw->s) if (raw->s)
sk_close(raw->s); sk_close(raw->s);
conf_free(raw->conf);
sfree(raw); sfree(raw);
} }

View File

@ -49,7 +49,8 @@ static void rlogin_log(Plug plug, int type, SockAddr addr, int port,
{ {
Rlogin rlogin = (Rlogin) plug; Rlogin rlogin = (Rlogin) plug;
backend_socket_log(rlogin->frontend, type, addr, port, backend_socket_log(rlogin->frontend, type, addr, port,
error_msg, error_code); error_msg, error_code,
rlogin->conf, !rlogin->firstbyte);
} }
static int rlogin_closing(Plug plug, const char *error_msg, int error_code, static int rlogin_closing(Plug plug, const char *error_msg, int error_code,

View File

@ -480,6 +480,7 @@ void save_open_settings(void *sesskey, Conf *conf)
write_setting_s(sesskey, "ProxyUsername", conf_get_str(conf, CONF_proxy_username)); write_setting_s(sesskey, "ProxyUsername", conf_get_str(conf, CONF_proxy_username));
write_setting_s(sesskey, "ProxyPassword", conf_get_str(conf, CONF_proxy_password)); write_setting_s(sesskey, "ProxyPassword", conf_get_str(conf, CONF_proxy_password));
write_setting_s(sesskey, "ProxyTelnetCommand", conf_get_str(conf, CONF_proxy_telnet_command)); write_setting_s(sesskey, "ProxyTelnetCommand", conf_get_str(conf, CONF_proxy_telnet_command));
write_setting_i(sesskey, "ProxyLogToTerm", conf_get_int(conf, CONF_proxy_log_to_term));
wmap(sesskey, "Environment", conf, CONF_environmt, TRUE); wmap(sesskey, "Environment", conf, CONF_environmt, TRUE);
write_setting_s(sesskey, "UserName", conf_get_str(conf, CONF_username)); write_setting_s(sesskey, "UserName", conf_get_str(conf, CONF_username));
write_setting_i(sesskey, "UserNameFromEnvironment", conf_get_int(conf, CONF_username_from_env)); write_setting_i(sesskey, "UserNameFromEnvironment", conf_get_int(conf, CONF_username_from_env));
@ -759,6 +760,7 @@ void load_open_settings(void *sesskey, Conf *conf)
gpps(sesskey, "ProxyPassword", "", conf, CONF_proxy_password); gpps(sesskey, "ProxyPassword", "", conf, CONF_proxy_password);
gpps(sesskey, "ProxyTelnetCommand", "connect %host %port\\n", gpps(sesskey, "ProxyTelnetCommand", "connect %host %port\\n",
conf, CONF_proxy_telnet_command); conf, CONF_proxy_telnet_command);
gppi(sesskey, "ProxyLogToTerm", FORCE_OFF, conf, CONF_proxy_log_to_term);
gppmap(sesskey, "Environment", conf, CONF_environmt); gppmap(sesskey, "Environment", conf, CONF_environmt);
gpps(sesskey, "UserName", "", conf, CONF_username); gpps(sesskey, "UserName", "", conf, CONF_username);
gppi(sesskey, "UserNameFromEnvironment", 0, conf, CONF_username_from_env); gppi(sesskey, "UserNameFromEnvironment", 0, conf, CONF_username_from_env);

7
ssh.c
View File

@ -792,6 +792,7 @@ struct ssh_tag {
int send_ok; int send_ok;
int echoing, editing; int echoing, editing;
int session_started;
void *frontend; void *frontend;
int ospeed, ispeed; /* temporaries */ int ospeed, ispeed; /* temporaries */
@ -3070,6 +3071,8 @@ static int do_ssh_init(Ssh ssh, unsigned char c)
crReturn(1); crReturn(1);
} }
ssh->session_started = TRUE;
s->vstrsize = sizeof(protoname) + 16; s->vstrsize = sizeof(protoname) + 16;
s->vstring = snewn(s->vstrsize, char); s->vstring = snewn(s->vstrsize, char);
strcpy(s->vstring, protoname); strcpy(s->vstring, protoname);
@ -3464,7 +3467,8 @@ static void ssh_socket_log(Plug plug, int type, SockAddr addr, int port,
if (!ssh->attempting_connshare) if (!ssh->attempting_connshare)
backend_socket_log(ssh->frontend, type, addr, port, backend_socket_log(ssh->frontend, type, addr, port,
error_msg, error_code); error_msg, error_code, ssh->conf,
ssh->session_started);
} }
void ssh_connshare_log(Ssh ssh, int event, const char *logtext, void ssh_connshare_log(Ssh ssh, int event, const char *logtext,
@ -10998,6 +11002,7 @@ static const char *ssh_init(void *frontend_handle, void **backend_handle,
ssh->X11_fwd_enabled = FALSE; ssh->X11_fwd_enabled = FALSE;
ssh->connshare = NULL; ssh->connshare = NULL;
ssh->attempting_connshare = FALSE; ssh->attempting_connshare = FALSE;
ssh->session_started = FALSE;
*backend_handle = ssh; *backend_handle = ssh;

View File

@ -197,6 +197,7 @@ typedef struct telnet_tag {
int sb_opt, sb_len; int sb_opt, sb_len;
unsigned char *sb_buf; unsigned char *sb_buf;
int sb_size; int sb_size;
int session_started;
enum { enum {
TOP_LEVEL, SEENIAC, SEENWILL, SEENWONT, SEENDO, SEENDONT, TOP_LEVEL, SEENIAC, SEENWILL, SEENWONT, SEENDO, SEENDONT,
@ -653,7 +654,8 @@ static void telnet_log(Plug plug, int type, SockAddr addr, int port,
{ {
Telnet telnet = (Telnet) plug; Telnet telnet = (Telnet) plug;
backend_socket_log(telnet->frontend, type, addr, port, backend_socket_log(telnet->frontend, type, addr, port,
error_msg, error_code); error_msg, error_code, telnet->conf,
telnet->session_started);
} }
static int telnet_closing(Plug plug, const char *error_msg, int error_code, static int telnet_closing(Plug plug, const char *error_msg, int error_code,
@ -687,6 +689,7 @@ static int telnet_receive(Plug plug, int urgent, char *data, int len)
Telnet telnet = (Telnet) plug; Telnet telnet = (Telnet) plug;
if (urgent) if (urgent)
telnet->in_synch = TRUE; telnet->in_synch = TRUE;
telnet->session_started = TRUE;
do_telnet_read(telnet, data, len); do_telnet_read(telnet, data, len);
return 1; return 1;
} }
@ -737,6 +740,7 @@ static const char *telnet_init(void *frontend_handle, void **backend_handle,
telnet->state = TOP_LEVEL; telnet->state = TOP_LEVEL;
telnet->ldisc = NULL; telnet->ldisc = NULL;
telnet->pinger = NULL; telnet->pinger = NULL;
telnet->session_started = TRUE;
*backend_handle = telnet; *backend_handle = telnet;
/* /*

View File

@ -103,6 +103,8 @@ FontSpec *platform_default_fontspec(const char *name) { return fontspec_new("");
Filename *platform_default_filename(const char *name) { return filename_from_str(""); } Filename *platform_default_filename(const char *name) { return filename_from_str(""); }
char *x_get_default(const char *key) { return NULL; } char *x_get_default(const char *key) { return NULL; }
void log_eventlog(void *handle, const char *event) {} void log_eventlog(void *handle, const char *event) {}
int from_backend(void *frontend, int is_stderr, const char *data, int datalen)
{ assert(!"only here to satisfy notional call from backend_socket_log"); }
/* /*
* Short description of parameters. * Short description of parameters.