Chilkat HOME Android™ Classic ASP C C++ C# Mono C# .NET Core C# C# UWP/WinRT DataFlex Delphi ActiveX Delphi DLL Visual FoxPro Java Lianja MFC Objective-C Perl PHP ActiveX PHP Extension PowerBuilder PowerShell PureBasic CkPython Chilkat2-Python Ruby SQL Server Swift 2 Swift 3,4,5... Tcl Unicode C Unicode C++ Visual Basic 6.0 VB.NET VB.NET UWP/WinRT VBScript Xojo Plugin Node.js Excel Go
(MFC) TCP or TLS over Multiple Hop SSH to Remote ServerDemonstrates how to use the Chilkat Socket API to connect to a remote server (using TCP or TLS) tunneled through mulitple-hop SSH. The scheme looks like this: Application => ServerSSH1 => ServerSSH2 => DestinationServer The ConnectThroughSsh and UseSsh methods are added in Chilkat version 9.5.0.55 to accomplish this task.
#include <CkSsh.h> #include <CkSocket.h> void ChilkatSample(void) { CkString strOut; // This example requires the Chilkat API to have been previously unlocked. // See Global Unlock Sample for sample code. CkSsh ssh1; // Connect directly to the 1st SSH server. bool success = ssh1.Connect("serverssh1.com",22); if (success != true) { strOut.append(ssh1.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } // Authenticate using login/password: success = ssh1.AuthenticatePw("ssh1Login","ssh1Password"); if (success != true) { strOut.append(ssh1.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } // Connect through the 1st SSH connection to reach a 2nd SSH server. // Note: Any number of SSH connections may be simultaneously tunneled through a single // existing SSH connection. CkSsh ssh2; success = ssh2.ConnectThroughSsh(ssh1,"serverssh2.com",22); if (success != true) { strOut.append(ssh2.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } // Authenticate with ssh2... success = ssh2.AuthenticatePw("ssh2Login","ssh2Password"); if (success != true) { strOut.append(ssh2.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } CkSocket socket; // Tell the socket object to connect to our destination server though the ssh2 tunnel (which itself is routed through ssh1). // The connection looks like this: ApplicationSocket(TLS) => ServerSSH1 => ServerSSH2 => DestinationServer success = socket.UseSsh(ssh2); if (success != true) { strOut.append(socket.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } // Connect using TLS to www.chilkatsoft.com // We could also tunnel a bare TCP connection by specifying port 80 with useTls = false. bool useTls = true; int maxWaitMillisec = 20000; success = socket.Connect("www.chilkatsoft.com",443,useTls,maxWaitMillisec); if (success != true) { strOut.append(socket.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } // Once the multiple hop SSH tunneled connection is setup, the socket programming // is identical to the normal case where we have a direct connection. // Tell the socket object that all text is to be sent in the utf-8 encoding, // and the text received is assumed to be utf-8. socket.put_StringCharset("utf-8"); // Send an HTTP HEAD request: success = socket.SendString("HEAD / HTTP/1.1\r\nHost: www.chilkatsoft.com\r\n\r\n"); if (success != true) { strOut.append(socket.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } // Wait a maximum of 4 seconds while no data is forthcoming: socket.put_MaxReadIdleMs(4000); // Get the 1st response line, which should be "HTTP/1.1 200 OK" const char *responseStatusLine = socket.receiveToCRLF(); if (socket.get_LastMethodSuccess() != true) { strOut.append(socket.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } strOut.append("StatusLine: "); strOut.append(responseStatusLine); strOut.append("\r\n"); // Now get the 1st line of the response header: const char *responseHeaderLine = socket.receiveToCRLF(); if (socket.get_LastMethodSuccess() != true) { strOut.append(socket.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } strOut.append("HeaderLine: "); strOut.append(responseHeaderLine); strOut.append("\r\n"); // Now read the remainder of the response header by reading until a double CRLF is seen: const char *remainderOfHeader = socket.receiveUntilMatch("\r\n\r\n"); if (socket.get_LastMethodSuccess() != true) { strOut.append(socket.lastErrorText()); strOut.append("\r\n"); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); return; } strOut.append("Remainder: "); strOut.append(remainderOfHeader); strOut.append("\r\n"); // Close the connection with the server. This closes the tunnel through ssh2. // Wait a max of 20 seconds (20000 millsec) success = socket.Close(20000); // Close the connection with ssh2. (This closes the the tunnel through ssh1.) // The connection with ssh1 is still alive, and may be used for more connections. ssh2.Disconnect(); ssh1.Disconnect(); SetDlgItemText(IDC_EDIT1,strOut.getUnicode()); } |
© 2000-2022 Chilkat Software, Inc. All Rights Reserved.