mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 01:02:24 +00:00
SOCKS proxy support added (next instalment of Justin Bradford's
proxy work). SOCKS 5 username/password authentication still unsupported. [originally from svn r1622]
This commit is contained in:
parent
6a95a15aaa
commit
5331956c07
@ -76,6 +76,8 @@ void sk_cleanup(void); /* called just before program exit */
|
|||||||
|
|
||||||
SockAddr sk_namelookup(char *host, char **canonicalname);
|
SockAddr sk_namelookup(char *host, char **canonicalname);
|
||||||
void sk_getaddr(SockAddr addr, char *buf, int buflen);
|
void sk_getaddr(SockAddr addr, char *buf, int buflen);
|
||||||
|
int sk_addrtype(SockAddr addr);
|
||||||
|
void sk_addrcopy(SockAddr addr, char *buf);
|
||||||
void sk_addr_free(SockAddr addr);
|
void sk_addr_free(SockAddr addr);
|
||||||
|
|
||||||
Socket sk_new(SockAddr addr, int port, int privport, int oobinline,
|
Socket sk_new(SockAddr addr, int port, int privport, int oobinline,
|
||||||
|
383
proxy.c
383
proxy.c
@ -322,7 +322,10 @@ Socket new_connection(SockAddr addr, char *hostname,
|
|||||||
if (cfg.proxy_type == PROXY_HTTP) {
|
if (cfg.proxy_type == PROXY_HTTP) {
|
||||||
ret->negotiate = proxy_http_negotiate;
|
ret->negotiate = proxy_http_negotiate;
|
||||||
} else if (cfg.proxy_type == PROXY_SOCKS) {
|
} else if (cfg.proxy_type == PROXY_SOCKS) {
|
||||||
ret->negotiate = proxy_socks_negotiate;
|
if (cfg.proxy_socks_version == 4)
|
||||||
|
ret->negotiate = proxy_socks4_negotiate;
|
||||||
|
else
|
||||||
|
ret->negotiate = proxy_socks5_negotiate;
|
||||||
} else if (cfg.proxy_type == PROXY_TELNET) {
|
} else if (cfg.proxy_type == PROXY_TELNET) {
|
||||||
ret->negotiate = proxy_telnet_negotiate;
|
ret->negotiate = proxy_telnet_negotiate;
|
||||||
} else {
|
} else {
|
||||||
@ -427,7 +430,6 @@ int proxy_http_negotiate (Proxy_Socket p, int change)
|
|||||||
sk_write(p->sub_socket, buf, strlen(buf));
|
sk_write(p->sub_socket, buf, strlen(buf));
|
||||||
|
|
||||||
p->state = 1;
|
p->state = 1;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -532,19 +534,385 @@ int proxy_http_negotiate (Proxy_Socket p, int change)
|
|||||||
|
|
||||||
plug_closing(p->plug, "Network error: Unexpected proxy error",
|
plug_closing(p->plug, "Network error: Unexpected proxy error",
|
||||||
PROXY_ERROR_UNEXPECTED, 0);
|
PROXY_ERROR_UNEXPECTED, 0);
|
||||||
return 0;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ----------------------------------------------------------------------
|
/* ----------------------------------------------------------------------
|
||||||
* SOCKS proxy type (as yet unimplemented).
|
* SOCKS proxy type.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
int proxy_socks_negotiate (Proxy_Socket p, int change)
|
/* SOCKS version 4 */
|
||||||
|
int proxy_socks4_negotiate (Proxy_Socket p, int change)
|
||||||
{
|
{
|
||||||
p->error = "Network error: SOCKS proxy implementation is incomplete";
|
if (p->state == PROXY_CHANGE_NEW) {
|
||||||
|
|
||||||
|
/* request format:
|
||||||
|
* version number (1 byte) = 4
|
||||||
|
* command code (1 byte)
|
||||||
|
* 1 = CONNECT
|
||||||
|
* 2 = BIND
|
||||||
|
* dest. port (2 bytes) [network order]
|
||||||
|
* dest. address (4 bytes)
|
||||||
|
* user ID (variable length, null terminated string)
|
||||||
|
*/
|
||||||
|
|
||||||
|
int length;
|
||||||
|
char * command;
|
||||||
|
|
||||||
|
if (sk_addrtype(p->remote_addr) != AF_INET) {
|
||||||
|
plug_closing(p->plug, "Network error: SOCKS version 4 does not support IPv6",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
length = strlen(cfg.proxy_username) + 9;
|
||||||
|
command = (char*) malloc(length);
|
||||||
|
strcpy(command + 8, cfg.proxy_username);
|
||||||
|
|
||||||
|
command[0] = 4; /* version 4 */
|
||||||
|
command[1] = 1; /* CONNECT command */
|
||||||
|
|
||||||
|
/* port */
|
||||||
|
command[2] = (char) (p->remote_port >> 8) & 0xff;
|
||||||
|
command[3] = (char) p->remote_port & 0xff;
|
||||||
|
|
||||||
|
/* address */
|
||||||
|
sk_addrcopy(p->remote_addr, command + 4);
|
||||||
|
|
||||||
|
sk_write(p->sub_socket, command, length);
|
||||||
|
free(command);
|
||||||
|
|
||||||
|
p->state = 1;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_CLOSING) {
|
||||||
|
/* if our proxy negotiation process involves closing and opening
|
||||||
|
* new sockets, then we would want to intercept this closing
|
||||||
|
* callback when we were expecting it. if we aren't anticipating
|
||||||
|
* a socket close, then some error must have occurred. we'll
|
||||||
|
* just pass those errors up to the backend.
|
||||||
|
*/
|
||||||
|
return plug_closing(p->plug, p->closing_error_msg,
|
||||||
|
p->closing_error_code,
|
||||||
|
p->closing_calling_back);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_SENT) {
|
||||||
|
/* some (or all) of what we wrote to the proxy was sent.
|
||||||
|
* we don't do anything new, however, until we receive the
|
||||||
|
* proxy's response. we might want to set a timer so we can
|
||||||
|
* timeout the proxy negotiation after a while...
|
||||||
|
*/
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_ACCEPTING) {
|
||||||
|
/* we should _never_ see this, as we are using our socket to
|
||||||
|
* connect to a proxy, not accepting inbound connections.
|
||||||
|
* what should we do? close the socket with an appropriate
|
||||||
|
* error message?
|
||||||
|
*/
|
||||||
|
return plug_accepting(p->plug, p->accepting_sock);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_RECEIVE) {
|
||||||
|
/* we have received data from the underlying socket, which
|
||||||
|
* we'll need to parse, process, and respond to appropriately.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (p->state == 1) {
|
||||||
|
/* response format:
|
||||||
|
* version number (1 byte) = 4
|
||||||
|
* reply code (1 byte)
|
||||||
|
* 90 = request granted
|
||||||
|
* 91 = request rejected or failed
|
||||||
|
* 92 = request rejected due to lack of IDENTD on client
|
||||||
|
* 93 = request rejected due to difference in user ID
|
||||||
|
* (what we sent vs. what IDENTD said)
|
||||||
|
* dest. port (2 bytes)
|
||||||
|
* dest. address (4 bytes)
|
||||||
|
*/
|
||||||
|
|
||||||
|
char *data;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
/* get the response */
|
||||||
|
bufchain_prefix(&p->pending_input_data, &data, &len);
|
||||||
|
|
||||||
|
if (data[0] != 0) {
|
||||||
|
plug_closing(p->plug, "Network error: SOCKS proxy responded with "
|
||||||
|
"unexpected reply code version",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data[1] != 90) {
|
||||||
|
|
||||||
|
switch (data[1]) {
|
||||||
|
case 92:
|
||||||
|
plug_closing(p->plug, "Network error: SOCKS server wanted IDENTD on client",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
break;
|
||||||
|
case 93:
|
||||||
|
plug_closing(p->plug, "Network error: Username and IDENTD on client don't agree",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
break;
|
||||||
|
case 91:
|
||||||
|
default:
|
||||||
|
plug_closing(p->plug, "Network error: Error while communicating with proxy",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* we're done */
|
||||||
|
proxy_activate(p);
|
||||||
|
/* proxy activate will have dealt with
|
||||||
|
* whatever is left of the buffer */
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
plug_closing(p->plug, "Network error: Unexpected proxy error",
|
||||||
|
PROXY_ERROR_UNEXPECTED, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* SOCKS version 5 */
|
||||||
|
int proxy_socks5_negotiate (Proxy_Socket p, int change)
|
||||||
|
{
|
||||||
|
if (p->state == PROXY_CHANGE_NEW) {
|
||||||
|
|
||||||
|
/* initial command:
|
||||||
|
* version number (1 byte) = 5
|
||||||
|
* number of available authentication methods (1 byte)
|
||||||
|
* available authentication methods (1 byte * previous value)
|
||||||
|
* authentication methods:
|
||||||
|
* 0x00 = no authentication
|
||||||
|
* 0x01 = GSSAPI
|
||||||
|
* 0x02 = username/password
|
||||||
|
* 0x03 = CHAP
|
||||||
|
*/
|
||||||
|
|
||||||
|
char command[3];
|
||||||
|
|
||||||
|
command[0] = 5; /* version 5 */
|
||||||
|
command[1] = 1; /* TODO: we don't currently support any auth methods */
|
||||||
|
command[2] = 0x00; /* no authentication */
|
||||||
|
|
||||||
|
sk_write(p->sub_socket, command, 3);
|
||||||
|
|
||||||
|
p->state = 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_CLOSING) {
|
||||||
|
/* if our proxy negotiation process involves closing and opening
|
||||||
|
* new sockets, then we would want to intercept this closing
|
||||||
|
* callback when we were expecting it. if we aren't anticipating
|
||||||
|
* a socket close, then some error must have occurred. we'll
|
||||||
|
* just pass those errors up to the backend.
|
||||||
|
*/
|
||||||
|
return plug_closing(p->plug, p->closing_error_msg,
|
||||||
|
p->closing_error_code,
|
||||||
|
p->closing_calling_back);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_SENT) {
|
||||||
|
/* some (or all) of what we wrote to the proxy was sent.
|
||||||
|
* we don't do anything new, however, until we receive the
|
||||||
|
* proxy's response. we might want to set a timer so we can
|
||||||
|
* timeout the proxy negotiation after a while...
|
||||||
|
*/
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_ACCEPTING) {
|
||||||
|
/* we should _never_ see this, as we are using our socket to
|
||||||
|
* connect to a proxy, not accepting inbound connections.
|
||||||
|
* what should we do? close the socket with an appropriate
|
||||||
|
* error message?
|
||||||
|
*/
|
||||||
|
return plug_accepting(p->plug, p->accepting_sock);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (change == PROXY_CHANGE_RECEIVE) {
|
||||||
|
/* we have received data from the underlying socket, which
|
||||||
|
* we'll need to parse, process, and respond to appropriately.
|
||||||
|
*/
|
||||||
|
|
||||||
|
char *data;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
if (p->state == 1) {
|
||||||
|
|
||||||
|
/* initial response:
|
||||||
|
* version number (1 byte) = 5
|
||||||
|
* authentication method (1 byte)
|
||||||
|
* authentication methods:
|
||||||
|
* 0x00 = no authentication
|
||||||
|
* 0x01 = GSSAPI
|
||||||
|
* 0x02 = username/password
|
||||||
|
* 0x03 = CHAP
|
||||||
|
* 0xff = no acceptable methods
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* get the response */
|
||||||
|
bufchain_prefix(&p->pending_input_data, &data, &len);
|
||||||
|
|
||||||
|
if (data[0] != 5) {
|
||||||
|
plug_closing(p->plug, "Network error: Error while communicating with proxy",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data[1] == 0x00) p->state = 2; /* no authentication needed */
|
||||||
|
else if (data[1] == 0x01) p->state = 4; /* GSSAPI authentication */
|
||||||
|
else if (data[1] == 0x02) p->state = 5; /* username/password authentication */
|
||||||
|
else if (data[1] == 0x03) p->state = 6; /* CHAP authentication */
|
||||||
|
else {
|
||||||
|
plug_closing(p->plug, "Network error: We don't support any of the SOCKS "
|
||||||
|
"server's authentication methods",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->state == 2) {
|
||||||
|
|
||||||
|
/* request format:
|
||||||
|
* version number (1 byte) = 5
|
||||||
|
* command code (1 byte)
|
||||||
|
* 1 = CONNECT
|
||||||
|
* 2 = BIND
|
||||||
|
* 3 = UDP ASSOCIATE
|
||||||
|
* reserved (1 byte) = 0x00
|
||||||
|
* address type (1 byte)
|
||||||
|
* 1 = IPv4
|
||||||
|
* 3 = domainname (first byte has length, no terminating null)
|
||||||
|
* 4 = IPv6
|
||||||
|
* dest. address (variable)
|
||||||
|
* dest. port (2 bytes) [network order]
|
||||||
|
*/
|
||||||
|
|
||||||
|
char command[22];
|
||||||
|
int len;
|
||||||
|
|
||||||
|
if (sk_addrtype(p->remote_addr) == AF_INET) {
|
||||||
|
len = 10;
|
||||||
|
command[3] = 1; /* IPv4 */
|
||||||
|
} else {
|
||||||
|
len = 22;
|
||||||
|
command[3] = 4; /* IPv6 */
|
||||||
|
}
|
||||||
|
|
||||||
|
command[0] = 5; /* version 5 */
|
||||||
|
command[1] = 1; /* CONNECT command */
|
||||||
|
command[2] = 0x00;
|
||||||
|
|
||||||
|
/* address */
|
||||||
|
sk_addrcopy(p->remote_addr, command+4);
|
||||||
|
|
||||||
|
/* port */
|
||||||
|
command[len-2] = (char) (p->remote_port >> 8) & 0xff;
|
||||||
|
command[len-1] = (char) p->remote_port & 0xff;
|
||||||
|
|
||||||
|
sk_write(p->sub_socket, command, len);
|
||||||
|
|
||||||
|
p->state = 3;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->state == 3) {
|
||||||
|
|
||||||
|
/* reply format:
|
||||||
|
* version number (1 bytes) = 5
|
||||||
|
* reply code (1 byte)
|
||||||
|
* 0 = succeeded
|
||||||
|
* 1 = general SOCKS server failure
|
||||||
|
* 2 = connection not allowed by ruleset
|
||||||
|
* 3 = network unreachable
|
||||||
|
* 4 = host unreachable
|
||||||
|
* 5 = connection refused
|
||||||
|
* 6 = TTL expired
|
||||||
|
* 7 = command not supported
|
||||||
|
* 8 = address type not supported
|
||||||
|
* reserved (1 byte) = x00
|
||||||
|
* address type (1 byte)
|
||||||
|
* 1 = IPv4
|
||||||
|
* 3 = domainname (first byte has length, no terminating null)
|
||||||
|
* 4 = IPv6
|
||||||
|
* server bound address (variable)
|
||||||
|
* server bound port (2 bytes) [network order]
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* get the response */
|
||||||
|
bufchain_prefix(&p->pending_input_data, &data, &len);
|
||||||
|
|
||||||
|
if (data[0] != 5) {
|
||||||
|
plug_closing(p->plug, "Network error: Error while communicating with proxy",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data[1] != 0) {
|
||||||
|
|
||||||
|
switch (data[1]) {
|
||||||
|
case 1:
|
||||||
|
case 2:
|
||||||
|
case 3:
|
||||||
|
case 4:
|
||||||
|
case 5:
|
||||||
|
case 6:
|
||||||
|
case 7:
|
||||||
|
case 8:
|
||||||
|
default:
|
||||||
|
plug_closing(p->plug, "Network error: Error while communicating with proxy",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* we're done */
|
||||||
|
proxy_activate(p);
|
||||||
|
/* proxy activate will have dealt with
|
||||||
|
* whatever is left of the buffer */
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->state == 4) {
|
||||||
|
/* TODO: Handle GSSAPI authentication */
|
||||||
|
plug_closing(p->plug, "Network error: We don't support GSSAPI authentication",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->state == 5) {
|
||||||
|
/* TODO: Handle username/password authentication */
|
||||||
|
plug_closing(p->plug, "Network error: We don't support username/password "
|
||||||
|
"authentication",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->state == 6) {
|
||||||
|
/* TODO: Handle CHAP authentication */
|
||||||
|
plug_closing(p->plug, "Network error: We don't support CHAP authentication",
|
||||||
|
PROXY_ERROR_GENERAL, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
plug_closing(p->plug, "Network error: Unexpected proxy error",
|
||||||
|
PROXY_ERROR_UNEXPECTED, 0);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
/* ----------------------------------------------------------------------
|
/* ----------------------------------------------------------------------
|
||||||
* `Telnet' proxy type.
|
* `Telnet' proxy type.
|
||||||
*
|
*
|
||||||
@ -709,7 +1077,6 @@ int proxy_telnet_negotiate (Proxy_Socket p, int change)
|
|||||||
}
|
}
|
||||||
|
|
||||||
p->state = 1;
|
p->state = 1;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -757,5 +1124,5 @@ int proxy_telnet_negotiate (Proxy_Socket p, int change)
|
|||||||
|
|
||||||
plug_closing(p->plug, "Network error: Unexpected proxy error",
|
plug_closing(p->plug, "Network error: Unexpected proxy error",
|
||||||
PROXY_ERROR_UNEXPECTED, 0);
|
PROXY_ERROR_UNEXPECTED, 0);
|
||||||
return 0;
|
return 1;
|
||||||
}
|
}
|
||||||
|
3
proxy.h
3
proxy.h
@ -96,6 +96,7 @@ extern void proxy_activate (Proxy_Socket);
|
|||||||
|
|
||||||
extern int proxy_http_negotiate (Proxy_Socket, int);
|
extern int proxy_http_negotiate (Proxy_Socket, int);
|
||||||
extern int proxy_telnet_negotiate (Proxy_Socket, int);
|
extern int proxy_telnet_negotiate (Proxy_Socket, int);
|
||||||
extern int proxy_socks_negotiate (Proxy_Socket, int);
|
extern int proxy_socks4_negotiate (Proxy_Socket, int);
|
||||||
|
extern int proxy_socks5_negotiate (Proxy_Socket, int);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
20
winnet.c
20
winnet.c
@ -368,6 +368,26 @@ void sk_getaddr(SockAddr addr, char *buf, int buflen)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int sk_addrtype(SockAddr addr)
|
||||||
|
{
|
||||||
|
return addr->family;
|
||||||
|
}
|
||||||
|
|
||||||
|
void sk_addrcopy(SockAddr addr, char *buf)
|
||||||
|
{
|
||||||
|
#ifdef IPV6
|
||||||
|
if (addr->family == AF_INET) {
|
||||||
|
#endif
|
||||||
|
struct in_addr a;
|
||||||
|
a.s_addr = htonl(addr->address);
|
||||||
|
strncpy(buf, (char*) &a.s_addr, 4);
|
||||||
|
#ifdef IPV6
|
||||||
|
} else {
|
||||||
|
strncpy(buf, (char*) addr->ai, 16);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
void sk_addr_free(SockAddr addr)
|
void sk_addr_free(SockAddr addr)
|
||||||
{
|
{
|
||||||
sfree(addr);
|
sfree(addr);
|
||||||
|
Loading…
Reference in New Issue
Block a user