mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 01:48:00 +00:00
New Seat query, has_mixed_input_stream().
(TL;DR: to suppress redundant 'Press Return to begin session' prompts in between hops of a jump-host configuration, in Plink.) This new query method directly asks the Seat the question: is the same stream of input used to provide responses to interactive login prompts, and the session input provided after login concludes? It's used to suppress the last-ditch anti-spoofing defence in Plink of interactively asking 'Access granted. Press Return to begin session', on the basis that any such spoofing attack works by confusing the user about what's a legit login prompt before the session begins and what's sent by the server after the main session begins - so if those two things take input from different places, the user can't be confused. This doesn't change the existing behaviour of Plink, which was already suppressing the antispoof prompt in cases where its standard input was redirected from something other than a terminal. But previously it was doing it within the can_set_trust_status() seat query, and I've now moved it out into a separate query function. The reason why these need to be separate is for SshProxy, which needs to give an unusual combination of answers when run inside Plink. For can_set_trust_status(), it needs to return whatever the parent Seat returns, so that all the login prompts for a string of proxy connections in session will be antispoofed the same way. But you only want that final 'Access granted' prompt to happen _once_, after all the proxy connection setup phases are done, because up until then you're still in the safe hands of PuTTY itself presenting an unbroken sequence of legit login prompts (even if they come from a succession of different servers). Hence, SshProxy unconditionally returns 'no' to the query of whether it has a single mixed input stream, because indeed, it never does - for purposes of session input it behaves like an always-redirected Plink, no matter what kind of real Seat it ends up sending its pre-session login prompts to.
This commit is contained in:
parent
028714d02a
commit
7eb7d5e2e9
@ -502,6 +502,7 @@ static const SeatVtable SshProxy_seat_vt = {
|
|||||||
.stripctrl_new = sshproxy_stripctrl_new,
|
.stripctrl_new = sshproxy_stripctrl_new,
|
||||||
.set_trust_status = sshproxy_set_trust_status,
|
.set_trust_status = sshproxy_set_trust_status,
|
||||||
.can_set_trust_status = sshproxy_can_set_trust_status,
|
.can_set_trust_status = sshproxy_can_set_trust_status,
|
||||||
|
.has_mixed_input_stream = nullseat_has_mixed_input_stream_no,
|
||||||
.verbose = sshproxy_verbose,
|
.verbose = sshproxy_verbose,
|
||||||
.interactive = sshproxy_interactive,
|
.interactive = sshproxy_interactive,
|
||||||
.get_cursor_position = nullseat_get_cursor_position,
|
.get_cursor_position = nullseat_get_cursor_position,
|
||||||
|
1
pscp.c
1
pscp.c
@ -87,6 +87,7 @@ static const SeatVtable pscp_seat_vt = {
|
|||||||
.stripctrl_new = console_stripctrl_new,
|
.stripctrl_new = console_stripctrl_new,
|
||||||
.set_trust_status = nullseat_set_trust_status,
|
.set_trust_status = nullseat_set_trust_status,
|
||||||
.can_set_trust_status = nullseat_can_set_trust_status_yes,
|
.can_set_trust_status = nullseat_can_set_trust_status_yes,
|
||||||
|
.has_mixed_input_stream = nullseat_has_mixed_input_stream_no,
|
||||||
.verbose = cmdline_seat_verbose,
|
.verbose = cmdline_seat_verbose,
|
||||||
.interactive = nullseat_interactive_no,
|
.interactive = nullseat_interactive_no,
|
||||||
.get_cursor_position = nullseat_get_cursor_position,
|
.get_cursor_position = nullseat_get_cursor_position,
|
||||||
|
1
psftp.c
1
psftp.c
@ -68,6 +68,7 @@ static const SeatVtable psftp_seat_vt = {
|
|||||||
.stripctrl_new = console_stripctrl_new,
|
.stripctrl_new = console_stripctrl_new,
|
||||||
.set_trust_status = nullseat_set_trust_status,
|
.set_trust_status = nullseat_set_trust_status,
|
||||||
.can_set_trust_status = nullseat_can_set_trust_status_yes,
|
.can_set_trust_status = nullseat_can_set_trust_status_yes,
|
||||||
|
.has_mixed_input_stream = nullseat_has_mixed_input_stream_no,
|
||||||
.verbose = cmdline_seat_verbose,
|
.verbose = cmdline_seat_verbose,
|
||||||
.interactive = nullseat_interactive_yes,
|
.interactive = nullseat_interactive_yes,
|
||||||
.get_cursor_position = nullseat_get_cursor_position,
|
.get_cursor_position = nullseat_get_cursor_position,
|
||||||
|
17
putty.h
17
putty.h
@ -1289,6 +1289,18 @@ struct SeatVtable {
|
|||||||
*/
|
*/
|
||||||
bool (*can_set_trust_status)(Seat *seat);
|
bool (*can_set_trust_status)(Seat *seat);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Query whether this Seat's interactive prompt responses and its
|
||||||
|
* session input come from the same place.
|
||||||
|
*
|
||||||
|
* If false, this is used to suppress the final 'Press Return to
|
||||||
|
* begin session' anti-spoofing prompt in Plink. For example,
|
||||||
|
* Plink itself sets this flag if its standard input is redirected
|
||||||
|
* (and therefore not coming from the same place as the console
|
||||||
|
* it's sending its prompts to).
|
||||||
|
*/
|
||||||
|
bool (*has_mixed_input_stream)(Seat *seat);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Ask the seat whether it would like verbose messages.
|
* Ask the seat whether it would like verbose messages.
|
||||||
*/
|
*/
|
||||||
@ -1365,6 +1377,8 @@ static inline void seat_set_trust_status(Seat *seat, bool trusted)
|
|||||||
{ seat->vt->set_trust_status(seat, trusted); }
|
{ seat->vt->set_trust_status(seat, trusted); }
|
||||||
static inline bool seat_can_set_trust_status(Seat *seat)
|
static inline bool seat_can_set_trust_status(Seat *seat)
|
||||||
{ return seat->vt->can_set_trust_status(seat); }
|
{ return seat->vt->can_set_trust_status(seat); }
|
||||||
|
static inline bool seat_has_mixed_input_stream(Seat *seat)
|
||||||
|
{ return seat->vt->has_mixed_input_stream(seat); }
|
||||||
static inline bool seat_verbose(Seat *seat)
|
static inline bool seat_verbose(Seat *seat)
|
||||||
{ return seat->vt->verbose(seat); }
|
{ return seat->vt->verbose(seat); }
|
||||||
static inline bool seat_interactive(Seat *seat)
|
static inline bool seat_interactive(Seat *seat)
|
||||||
@ -1437,6 +1451,8 @@ StripCtrlChars *nullseat_stripctrl_new(
|
|||||||
void nullseat_set_trust_status(Seat *seat, bool trusted);
|
void nullseat_set_trust_status(Seat *seat, bool trusted);
|
||||||
bool nullseat_can_set_trust_status_yes(Seat *seat);
|
bool nullseat_can_set_trust_status_yes(Seat *seat);
|
||||||
bool nullseat_can_set_trust_status_no(Seat *seat);
|
bool nullseat_can_set_trust_status_no(Seat *seat);
|
||||||
|
bool nullseat_has_mixed_input_stream_yes(Seat *seat);
|
||||||
|
bool nullseat_has_mixed_input_stream_no(Seat *seat);
|
||||||
bool nullseat_verbose_no(Seat *seat);
|
bool nullseat_verbose_no(Seat *seat);
|
||||||
bool nullseat_verbose_yes(Seat *seat);
|
bool nullseat_verbose_yes(Seat *seat);
|
||||||
bool nullseat_interactive_no(Seat *seat);
|
bool nullseat_interactive_no(Seat *seat);
|
||||||
@ -1463,6 +1479,7 @@ StripCtrlChars *console_stripctrl_new(
|
|||||||
Seat *seat, BinarySink *bs_out, SeatInteractionContext sic);
|
Seat *seat, BinarySink *bs_out, SeatInteractionContext sic);
|
||||||
void console_set_trust_status(Seat *seat, bool trusted);
|
void console_set_trust_status(Seat *seat, bool trusted);
|
||||||
bool console_can_set_trust_status(Seat *seat);
|
bool console_can_set_trust_status(Seat *seat);
|
||||||
|
bool console_has_mixed_input_stream(Seat *seat);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Other centralised seat functions.
|
* Other centralised seat functions.
|
||||||
|
@ -544,5 +544,9 @@ SshChannel *ssh1_serverside_agent_open(ConnectionLayer *cl, Channel *chan)
|
|||||||
bool ssh1_connection_need_antispoof_prompt(struct ssh1_connection_state *s)
|
bool ssh1_connection_need_antispoof_prompt(struct ssh1_connection_state *s)
|
||||||
{
|
{
|
||||||
seat_set_trust_status(s->ppl.seat, false);
|
seat_set_trust_status(s->ppl.seat, false);
|
||||||
return !seat_can_set_trust_status(s->ppl.seat);
|
if (!seat_has_mixed_input_stream(s->ppl.seat))
|
||||||
|
return false;
|
||||||
|
if (seat_can_set_trust_status(s->ppl.seat))
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -501,6 +501,11 @@ void ssh2channel_send_terminal_size_change(SshChannel *sc, int w, int h)
|
|||||||
bool ssh2_connection_need_antispoof_prompt(struct ssh2_connection_state *s)
|
bool ssh2_connection_need_antispoof_prompt(struct ssh2_connection_state *s)
|
||||||
{
|
{
|
||||||
seat_set_trust_status(s->ppl.seat, false);
|
seat_set_trust_status(s->ppl.seat, false);
|
||||||
bool success = seat_can_set_trust_status(s->ppl.seat);
|
if (!seat_has_mixed_input_stream(s->ppl.seat))
|
||||||
return (!success && !ssh_is_bare(s->ppl.ssh));
|
return false;
|
||||||
|
if (seat_can_set_trust_status(s->ppl.seat))
|
||||||
|
return false;
|
||||||
|
if (ssh_is_bare(s->ppl.ssh))
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -128,6 +128,7 @@ static const SeatVtable server_seat_vt = {
|
|||||||
.stripctrl_new = nullseat_stripctrl_new,
|
.stripctrl_new = nullseat_stripctrl_new,
|
||||||
.set_trust_status = nullseat_set_trust_status,
|
.set_trust_status = nullseat_set_trust_status,
|
||||||
.can_set_trust_status = nullseat_can_set_trust_status_no,
|
.can_set_trust_status = nullseat_can_set_trust_status_no,
|
||||||
|
.has_mixed_input_stream = nullseat_has_mixed_input_stream_no,
|
||||||
.verbose = nullseat_verbose_no,
|
.verbose = nullseat_verbose_no,
|
||||||
.interactive = nullseat_interactive_no,
|
.interactive = nullseat_interactive_no,
|
||||||
.get_cursor_position = nullseat_get_cursor_position,
|
.get_cursor_position = nullseat_get_cursor_position,
|
||||||
|
@ -207,6 +207,7 @@ static const SeatVtable sesschan_seat_vt = {
|
|||||||
.stripctrl_new = nullseat_stripctrl_new,
|
.stripctrl_new = nullseat_stripctrl_new,
|
||||||
.set_trust_status = nullseat_set_trust_status,
|
.set_trust_status = nullseat_set_trust_status,
|
||||||
.can_set_trust_status = nullseat_can_set_trust_status_no,
|
.can_set_trust_status = nullseat_can_set_trust_status_no,
|
||||||
|
.has_mixed_input_stream = nullseat_has_mixed_input_stream_no,
|
||||||
.verbose = nullseat_verbose_no,
|
.verbose = nullseat_verbose_no,
|
||||||
.interactive = nullseat_interactive_no,
|
.interactive = nullseat_interactive_no,
|
||||||
.get_cursor_position = nullseat_get_cursor_position,
|
.get_cursor_position = nullseat_get_cursor_position,
|
||||||
|
@ -328,19 +328,11 @@ void console_set_trust_status(Seat *seat, bool trusted)
|
|||||||
|
|
||||||
bool console_can_set_trust_status(Seat *seat)
|
bool console_can_set_trust_status(Seat *seat)
|
||||||
{
|
{
|
||||||
if (console_batch_mode || !is_interactive() || !console_antispoof_prompt) {
|
if (console_batch_mode) {
|
||||||
/*
|
/*
|
||||||
* In batch mode, we don't need to worry about the server
|
* In batch mode, we don't need to worry about the server
|
||||||
* mimicking our interactive authentication, because the user
|
* mimicking our interactive authentication, because the user
|
||||||
* already knows not to expect any.
|
* already knows not to expect any.
|
||||||
*
|
|
||||||
* If standard input isn't connected to a terminal, likewise,
|
|
||||||
* because even if the server did send a spoof authentication
|
|
||||||
* prompt, the user couldn't respond to it via the terminal
|
|
||||||
* anyway.
|
|
||||||
*
|
|
||||||
* We also return true without enabling any defences if the
|
|
||||||
* user has purposely disabled the antispoof prompt.
|
|
||||||
*/
|
*/
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -348,6 +340,23 @@ bool console_can_set_trust_status(Seat *seat)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool console_has_mixed_input_stream(Seat *seat)
|
||||||
|
{
|
||||||
|
if (!is_interactive() || !console_antispoof_prompt) {
|
||||||
|
/*
|
||||||
|
* If standard input isn't connected to a terminal, then even
|
||||||
|
* if the server did send a spoof authentication prompt, the
|
||||||
|
* user couldn't respond to it via the terminal anyway.
|
||||||
|
*
|
||||||
|
* We also pretend this is true if the user has purposely
|
||||||
|
* disabled the antispoof prompt.
|
||||||
|
*/
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Warn about the obsolescent key file format.
|
* Warn about the obsolescent key file format.
|
||||||
*
|
*
|
||||||
|
@ -411,6 +411,7 @@ static const SeatVtable plink_seat_vt = {
|
|||||||
.stripctrl_new = console_stripctrl_new,
|
.stripctrl_new = console_stripctrl_new,
|
||||||
.set_trust_status = console_set_trust_status,
|
.set_trust_status = console_set_trust_status,
|
||||||
.can_set_trust_status = console_can_set_trust_status,
|
.can_set_trust_status = console_can_set_trust_status,
|
||||||
|
.has_mixed_input_stream = console_has_mixed_input_stream,
|
||||||
.verbose = cmdline_seat_verbose,
|
.verbose = cmdline_seat_verbose,
|
||||||
.interactive = plink_seat_interactive,
|
.interactive = plink_seat_interactive,
|
||||||
.get_cursor_position = nullseat_get_cursor_position,
|
.get_cursor_position = nullseat_get_cursor_position,
|
||||||
|
@ -414,6 +414,7 @@ static const SeatVtable gtk_seat_vt = {
|
|||||||
.stripctrl_new = gtk_seat_stripctrl_new,
|
.stripctrl_new = gtk_seat_stripctrl_new,
|
||||||
.set_trust_status = gtk_seat_set_trust_status,
|
.set_trust_status = gtk_seat_set_trust_status,
|
||||||
.can_set_trust_status = gtk_seat_can_set_trust_status,
|
.can_set_trust_status = gtk_seat_can_set_trust_status,
|
||||||
|
.has_mixed_input_stream = nullseat_has_mixed_input_stream_yes,
|
||||||
.verbose = nullseat_verbose_yes,
|
.verbose = nullseat_verbose_yes,
|
||||||
.interactive = nullseat_interactive_yes,
|
.interactive = nullseat_interactive_yes,
|
||||||
.get_cursor_position = gtk_seat_get_cursor_position,
|
.get_cursor_position = gtk_seat_get_cursor_position,
|
||||||
|
@ -41,6 +41,8 @@ StripCtrlChars *nullseat_stripctrl_new(
|
|||||||
void nullseat_set_trust_status(Seat *seat, bool trusted) {}
|
void nullseat_set_trust_status(Seat *seat, bool trusted) {}
|
||||||
bool nullseat_can_set_trust_status_yes(Seat *seat) { return true; }
|
bool nullseat_can_set_trust_status_yes(Seat *seat) { return true; }
|
||||||
bool nullseat_can_set_trust_status_no(Seat *seat) { return false; }
|
bool nullseat_can_set_trust_status_no(Seat *seat) { return false; }
|
||||||
|
bool nullseat_has_mixed_input_stream_yes(Seat *seat) { return true; }
|
||||||
|
bool nullseat_has_mixed_input_stream_no(Seat *seat) { return false; }
|
||||||
bool nullseat_verbose_no(Seat *seat) { return false; }
|
bool nullseat_verbose_no(Seat *seat) { return false; }
|
||||||
bool nullseat_verbose_yes(Seat *seat) { return true; }
|
bool nullseat_verbose_yes(Seat *seat) { return true; }
|
||||||
bool nullseat_interactive_no(Seat *seat) { return false; }
|
bool nullseat_interactive_no(Seat *seat) { return false; }
|
||||||
|
@ -198,6 +198,12 @@ static bool tempseat_can_set_trust_status(Seat *seat)
|
|||||||
return seat_can_set_trust_status(ts->realseat);
|
return seat_can_set_trust_status(ts->realseat);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool tempseat_has_mixed_input_stream(Seat *seat)
|
||||||
|
{
|
||||||
|
TempSeat *ts = container_of(seat, TempSeat, seat);
|
||||||
|
return seat_has_mixed_input_stream(ts->realseat);
|
||||||
|
}
|
||||||
|
|
||||||
/* ----------------------------------------------------------------------
|
/* ----------------------------------------------------------------------
|
||||||
* Methods that should never be called on a TempSeat, so we can put an
|
* Methods that should never be called on a TempSeat, so we can put an
|
||||||
* unreachable() in them.
|
* unreachable() in them.
|
||||||
@ -324,6 +330,7 @@ static const struct SeatVtable tempseat_vt = {
|
|||||||
.stripctrl_new = tempseat_stripctrl_new,
|
.stripctrl_new = tempseat_stripctrl_new,
|
||||||
.set_trust_status = tempseat_set_trust_status,
|
.set_trust_status = tempseat_set_trust_status,
|
||||||
.can_set_trust_status = tempseat_can_set_trust_status,
|
.can_set_trust_status = tempseat_can_set_trust_status,
|
||||||
|
.has_mixed_input_stream = tempseat_has_mixed_input_stream,
|
||||||
.verbose = tempseat_verbose,
|
.verbose = tempseat_verbose,
|
||||||
.interactive = tempseat_interactive,
|
.interactive = tempseat_interactive,
|
||||||
.get_cursor_position = tempseat_get_cursor_position,
|
.get_cursor_position = tempseat_get_cursor_position,
|
||||||
|
@ -194,19 +194,11 @@ void console_set_trust_status(Seat *seat, bool trusted)
|
|||||||
|
|
||||||
bool console_can_set_trust_status(Seat *seat)
|
bool console_can_set_trust_status(Seat *seat)
|
||||||
{
|
{
|
||||||
if (console_batch_mode || !is_interactive() || !console_antispoof_prompt) {
|
if (console_batch_mode) {
|
||||||
/*
|
/*
|
||||||
* In batch mode, we don't need to worry about the server
|
* In batch mode, we don't need to worry about the server
|
||||||
* mimicking our interactive authentication, because the user
|
* mimicking our interactive authentication, because the user
|
||||||
* already knows not to expect any.
|
* already knows not to expect any.
|
||||||
*
|
|
||||||
* If standard input isn't connected to a terminal, likewise,
|
|
||||||
* because even if the server did send a spoof authentication
|
|
||||||
* prompt, the user couldn't respond to it via the terminal
|
|
||||||
* anyway.
|
|
||||||
*
|
|
||||||
* We also return true without enabling any defences if the
|
|
||||||
* user has purposely disabled the antispoof prompt.
|
|
||||||
*/
|
*/
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -214,6 +206,23 @@ bool console_can_set_trust_status(Seat *seat)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool console_has_mixed_input_stream(Seat *seat)
|
||||||
|
{
|
||||||
|
if (!is_interactive() || !console_antispoof_prompt) {
|
||||||
|
/*
|
||||||
|
* If standard input isn't connected to a terminal, then even
|
||||||
|
* if the server did send a spoof authentication prompt, the
|
||||||
|
* user couldn't respond to it via the terminal anyway.
|
||||||
|
*
|
||||||
|
* We also pretend this is true if the user has purposely
|
||||||
|
* disabled the antispoof prompt.
|
||||||
|
*/
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Ask whether to wipe a session log file before writing to it.
|
* Ask whether to wipe a session log file before writing to it.
|
||||||
* Returns 2 for wipe, 1 for append, 0 for cancel (don't log).
|
* Returns 2 for wipe, 1 for append, 0 for cancel (don't log).
|
||||||
|
@ -105,6 +105,7 @@ static const SeatVtable plink_seat_vt = {
|
|||||||
.stripctrl_new = console_stripctrl_new,
|
.stripctrl_new = console_stripctrl_new,
|
||||||
.set_trust_status = console_set_trust_status,
|
.set_trust_status = console_set_trust_status,
|
||||||
.can_set_trust_status = console_can_set_trust_status,
|
.can_set_trust_status = console_can_set_trust_status,
|
||||||
|
.has_mixed_input_stream = console_has_mixed_input_stream,
|
||||||
.verbose = cmdline_seat_verbose,
|
.verbose = cmdline_seat_verbose,
|
||||||
.interactive = plink_seat_interactive,
|
.interactive = plink_seat_interactive,
|
||||||
.get_cursor_position = nullseat_get_cursor_position,
|
.get_cursor_position = nullseat_get_cursor_position,
|
||||||
|
@ -352,6 +352,7 @@ static const SeatVtable win_seat_vt = {
|
|||||||
.stripctrl_new = win_seat_stripctrl_new,
|
.stripctrl_new = win_seat_stripctrl_new,
|
||||||
.set_trust_status = win_seat_set_trust_status,
|
.set_trust_status = win_seat_set_trust_status,
|
||||||
.can_set_trust_status = win_seat_can_set_trust_status,
|
.can_set_trust_status = win_seat_can_set_trust_status,
|
||||||
|
.has_mixed_input_stream = nullseat_has_mixed_input_stream_yes,
|
||||||
.verbose = nullseat_verbose_yes,
|
.verbose = nullseat_verbose_yes,
|
||||||
.interactive = nullseat_interactive_yes,
|
.interactive = nullseat_interactive_yes,
|
||||||
.get_cursor_position = win_seat_get_cursor_position,
|
.get_cursor_position = win_seat_get_cursor_position,
|
||||||
|
Loading…
Reference in New Issue
Block a user