2 Commits

Author SHA1 Message Date
1da824d8cb Add comments 2025-08-14 20:23:45 -04:00
f441b186b5 Enhancements 2025-08-14 17:32:16 -04:00
7 changed files with 118 additions and 49 deletions

View File

@@ -15,7 +15,7 @@ class Utility
static String formatCurrency(double number,int places=2); static String formatCurrency(double number,int places=2);
static bool fmod(double dividend,double divisor); static bool fmod(double dividend,double divisor);
static bool fmod(double dividend, double divisor,double epsilon); static bool fmod(double dividend, double divisor,double epsilon);
static String bytesTransferredToString(double bytesTransferred); static String byteCountToString(double bytesTransferred, bool perSecond = true);
private: private:
static String pad(String theString,char padChar,int toLength); static String pad(String theString,char padChar,int toLength);
static String addCommas(String& theString); static String addCommas(String& theString);
@@ -34,30 +34,35 @@ Utility::~Utility()
} }
inline inline
String Utility::bytesTransferredToString(double bytesTransferred) String Utility::byteCountToString(double bytesTransferred, bool perSecond)
{ {
double roundedValue = std::round(bytesTransferred * 100.00)/100.00; double roundedValue = std::round(bytesTransferred * 100.00)/100.00;
String strBytesTransferrred;
if(roundedValue >= 1E+15) if(roundedValue >= 1E+15)
{ {
return Utility::formatNumber(roundedValue/1E+15,2) + "Pb/s"; strBytesTransferrred = Utility::formatNumber(roundedValue/1E+15,2) + "Pb";
} }
else if(roundedValue >= 1000000000000) else if(roundedValue >= 1000000000000)
{ {
return Utility::formatNumber(roundedValue/1000000000000,2) + "Tb/s"; strBytesTransferrred = Utility::formatNumber(roundedValue/1000000000000,2) + "Tb";
} }
else if(roundedValue >= 1073741824) else if(roundedValue >= 1073741824)
{ {
return Utility::formatNumber(roundedValue/1073741824,2) + "Gbb/s"; strBytesTransferrred = Utility::formatNumber(roundedValue/1073741824,2) + "Gb";
} }
else if(roundedValue >= 1000000) else if(roundedValue >= 1000000)
{ {
return Utility::formatNumber(roundedValue/1000000,2) + "Mb/s"; strBytesTransferrred = Utility::formatNumber(roundedValue/1000000,2) + "Mb";
} }
else if(roundedValue >= 1000) else if(roundedValue >= 1000)
{ {
return Utility::formatNumber(roundedValue/1000,2) + "Kb/s"; strBytesTransferrred = Utility::formatNumber(roundedValue/1000,2) + "Kb";
} }
return Utility::formatNumber(roundedValue,2) + "B/s"; else strBytesTransferrred = Utility::formatNumber(roundedValue,2) + "B";
if(perSecond)strBytesTransferrred+="/s";
return strBytesTransferrred;
} }
inline inline

View File

@@ -1,5 +1,8 @@
#include <sstp/clientsocketsender.hpp> #include <sstp/clientsocketsender.hpp>
/// @brief Establish a connection to the socket specified by ipAddress and port
/// @param ipAddress The ipAddress
/// @param port The port
ClientSocketSender::ClientSocketSender(String ipAddress, int port) ClientSocketSender::ClientSocketSender(String ipAddress, int port)
: mIPAddress(ipAddress), mPort(port) : mIPAddress(ipAddress), mPort(port)
{ {
@@ -26,11 +29,15 @@ ClientSocketSender::ClientSocketSender(String ipAddress, int port)
} }
} }
/// @brief Close the connection on destruction
ClientSocketSender::~ClientSocketSender() ClientSocketSender::~ClientSocketSender()
{ {
close(); close();
} }
/// @brief Closes the connection
/// @param
void ClientSocketSender::close(void) void ClientSocketSender::close(void)
{ {
if(-1!=mSocket) if(-1!=mSocket)
@@ -40,6 +47,9 @@ void ClientSocketSender::close(void)
} }
} }
/// @brief Send the specified file through the connection
/// @param pathFileName The file to send
/// @return Returns true if successful
bool ClientSocketSender::sendFile(String &pathFileName) bool ClientSocketSender::sendFile(String &pathFileName)
{ {
Profiler profiler; Profiler profiler;

View File

@@ -1,5 +1,5 @@
#ifndef _LISTENER_CLIENTSOCKETSENDER_HPP_ #ifndef _SSTP_CLIENTSOCKETSENDER_HPP_
#define _LISTENER_CLIENTSOCKETSENDER_HPP_ #define _SSTP_CLIENTSOCKETSENDER_HPP_
#include <stdio.h> #include <stdio.h>
#include <iostream> #include <iostream>
#include <sys/types.h> #include <sys/types.h>
@@ -21,8 +21,7 @@ class ClientSocketSender
bool sendFile(String &pathFileName); bool sendFile(String &pathFileName);
bool isOkay(void); bool isOkay(void);
private: private:
// static constexpr int BUFFER_LENGTH=65536; static constexpr int BUFFER_LENGTH=1048576; // This buffer length is used for both reading the file and also the packet transmission length
static constexpr int BUFFER_LENGTH=1048576;
void close(void); void close(void);
bool sendPacketIndicator(DWORD bytesToSend); bool sendPacketIndicator(DWORD bytesToSend);
bool sendPacket(Array<char> &buffer,DWORD bytesToSend); bool sendPacket(Array<char> &buffer,DWORD bytesToSend);

View File

@@ -26,38 +26,55 @@ void handleClient(Block<String> &commands);
/// @return /// @return
int main(int argc, char **argv) int main(int argc, char **argv)
{ {
int returnCode(0);
Profiler profiler; Profiler profiler;
String version = "1.00";
if(argc<2) std::cout << "sstp version " << version.str() << std::endl;
try
{ {
std::cout << "sstp SERVERMODE {port} | CLIENTMODE {serveripaddress} {serverport} {pathfileName}." << std::endl; if(argc<2)
std::cout << "SERVERMODE will listen on the specified port for a connection and receive the specified file." << std::endl; {
std::cout << "CLIENTMODE willl connect to the specified address and port and send the specfied file." << std::endl; std::cout << "sstp SERVERMODE {port} | CLIENTMODE {serveripaddress} {serverport} {pathfileName}." << std::endl;
return 1; std::cout << "SERVERMODE will listen on the specified port for a connection and receive the specified file." << std::endl;
} std::cout << "CLIENTMODE willl connect to the specified address and port and send the specfied file." << std::endl;
Block<String> arguments; return 1;
for(int index=0;index<argc;index++) }
{ Block<String> arguments;
String item(argv[index]); for(int index=0;index<argc;index++)
arguments.insert(item); {
} String item(argv[index]);
arguments.insert(item);
}
std::cout << argv[1] << std::endl; std::cout << argv[1] << std::endl;
if(arguments[1]=="SERVERMODE") if(arguments[1]=="SERVERMODE")
{ {
handleServer(arguments); handleServer(arguments);
}
else if(arguments[1]=="CLIENTMODE")
{
handleClient(arguments);
}
else
{
std::cout << "Unknown command " << arguments[1] << std::endl;
returnCode=-1;
}
} }
else if(arguments[1]=="CLIENTMODE") catch(Exception& exception)
{ {
handleClient(arguments); std::cout << exception.toString() << std::endl;
returnCode=-1;
} }
else catch(...)
{ {
std::cout << "Unknown command " << arguments[1] << std::endl; std::cout << "An unhandled exception was encountered" << std::endl;
returnCode=-1;
} }
return 0;
std::cout << "Done, total took " << Utility::formatNumber(profiler.end()) << "(ms)" << std::endl; std::cout << "Done, total took " << Utility::formatNumber(profiler.end()) << "(ms)" << std::endl;
return returnCode;
} }
/// @brief [0]=program [1]=SERVERMODE [2]={port} /// @brief [0]=program [1]=SERVERMODE [2]={port}

View File

@@ -2,15 +2,22 @@
#include <common/utility.hpp> #include <common/utility.hpp>
#include <common/stringbuffer.hpp> #include <common/stringbuffer.hpp>
/// @brief After a connection has been accepted the SocketConnectionReceiver is initialized with the incoming socket
/// @param socket
/// @param internalSocketAddress
SocketConnectionReceiver::SocketConnectionReceiver(int socket, sockaddr_in internalSocketAddress) SocketConnectionReceiver::SocketConnectionReceiver(int socket, sockaddr_in internalSocketAddress)
: mSocket(socket), mInternalSocketAddress(internalSocketAddress) : mSocket(socket), mInternalSocketAddress(internalSocketAddress)
{ {
} }
/// @brief The destructor will close the socket
SocketConnectionReceiver::~SocketConnectionReceiver() SocketConnectionReceiver::~SocketConnectionReceiver()
{ {
close();
} }
/// @brief Close the socket and set the mSocket variable to -1
/// @param
void SocketConnectionReceiver::close(void) void SocketConnectionReceiver::close(void)
{ {
if(-1==mSocket)return; if(-1==mSocket)return;
@@ -19,6 +26,8 @@ void SocketConnectionReceiver::close(void)
mSocket=-1; mSocket=-1;
} }
/// @brief The threadFunction will handle the lifetime of the socket connection. When a QUIT is received the socket will be closed and the thread method will exit
/// @param data
void SocketConnectionReceiver::threadFunction(int data) void SocketConnectionReceiver::threadFunction(int data)
{ {
String line; String line;
@@ -30,7 +39,11 @@ void SocketConnectionReceiver::threadFunction(int data)
std::cout << "Received an empty command" << std::endl; std::cout << "Received an empty command" << std::endl;
break; break;
} }
if(line=="QUIT" || line=="quit")break; if(line=="QUIT" || line=="quit")
{
handleQuit();
break;
}
Block<String> commands = line.split('|'); Block<String> commands = line.split('|');
if(0==commands.size())continue; if(0==commands.size())continue;
commands[0].upper(); commands[0].upper();
@@ -49,9 +62,9 @@ void SocketConnectionReceiver::threadFunction(int data)
/// @brief The client wants to put a file. put {filename} {lengthbytes} /// @brief The client wants to put a file. put {filename} {lengthbytes}
/// @param commands [0]=PUT, [1]=filename, [2]=total length /// @param commands [0]=PUT, [1]=filename, [2]=total length
/// This will read PCKT {length}. /// This message contains PUT|{filename}|{lengthbytes} followed by as many PCKT|{length} packets as needed to read the entire file contents and save to disk
/// The connection will continue reading and writing until it receives PCKT 0 /// The connection will continue reading packets and saving data to disk until it receives PCKT 0 which signals the end of transmission
/// @return /// @return true or false
bool SocketConnectionReceiver::handlePut(Block<String> &commands) bool SocketConnectionReceiver::handlePut(Block<String> &commands)
{ {
Profiler profiler; Profiler profiler;
@@ -61,6 +74,7 @@ bool SocketConnectionReceiver::handlePut(Block<String> &commands)
String fileName; String fileName;
size_t fileLength; size_t fileLength;
size_t totalBytesRead=0; size_t totalBytesRead=0;
size_t totalPacketsRead=0;
fileName = commands[1]; fileName = commands[1];
fileLength = commands[commands.size()-1].toLong(); fileLength = commands[commands.size()-1].toLong();
@@ -93,6 +107,7 @@ bool SocketConnectionReceiver::handlePut(Block<String> &commands)
receiveBuffer.size(bufferLength); receiveBuffer.size(bufferLength);
size_t bytes_read = read(receiveBuffer); size_t bytes_read = read(receiveBuffer);
totalBytesRead+=bytes_read; totalBytesRead+=bytes_read;
totalPacketsRead++;
if(bufferLength!=bytes_read) if(bufferLength!=bytes_read)
{ {
std::cout << "Send/Receive size mismatch. The client indicated a data length of " << Utility::formatNumber(bufferLength) << " but a data length of " << Utility::formatNumber(bytes_read) << " was received"; std::cout << "Send/Receive size mismatch. The client indicated a data length of " << Utility::formatNumber(bufferLength) << " but a data length of " << Utility::formatNumber(bytes_read) << " was received";
@@ -105,17 +120,19 @@ bool SocketConnectionReceiver::handlePut(Block<String> &commands)
{ {
double elapsedTimeSeconds = profiler.elapsed()/1000.00; double elapsedTimeSeconds = profiler.elapsed()/1000.00;
double bytesPerSecond = totalBytesRead / elapsedTimeSeconds; double bytesPerSecond = totalBytesRead / elapsedTimeSeconds;
String strBytesPerSecond = Utility::bytesTransferredToString(bytesPerSecond); String strBytesPerSecond = Utility::byteCountToString(bytesPerSecond);
std::cout << "Transferred " << Utility::formatNumber(totalBytesRead) << " of " << Utility::formatNumber(fileLength) << " " << percent << " percent " << strBytesPerSecond << std::endl; std::cout << "Transferred " << Utility::byteCountToString(totalBytesRead,false) << " of " << Utility::byteCountToString(fileLength,false) << " " << percent << " percent " << strBytesPerSecond << std::endl;
} }
} }
std::cout << "Transfer complete" << std::endl;
std::cout << "Received " << Utility::formatNumber(totalBytesRead) << " in " << Utility::formatNumber(profiler.end()) << "(ms)" << std::endl;
writeFile.close(); writeFile.close();
std::cout << "Transfer complete. Total packets received " << Utility::formatNumber(totalPacketsRead) << std::endl;
std::cout << "Received " << Utility::byteCountToString(totalBytesRead,false) << " in " << Utility::formatNumber(profiler.end()) << "(ms)" << std::endl;
return totalBytesRead==fileLength; return totalBytesRead==fileLength;
} }
/// @brief Read a PCKT message from the socket and return the length. This message is of the form PCKT|{length} and the method returns the second parameter (length)
/// @param
/// @return
size_t SocketConnectionReceiver::expectPacket(void) size_t SocketConnectionReceiver::expectPacket(void)
{ {
Block<String> subCommands; Block<String> subCommands;
@@ -128,7 +145,7 @@ size_t SocketConnectionReceiver::expectPacket(void)
sb.append("Received an invalid PCKT command"); sb.append("Received an invalid PCKT command");
String str=sb.toString(); String str=sb.toString();
std::cout << str << std::endl; std::cout << str << std::endl;
throw new InvalidOperationException(str); throw InvalidOperationException(str);
} }
if(!(subCommands[0]=="PCKT")) if(!(subCommands[0]=="PCKT"))
{ {
@@ -138,12 +155,15 @@ size_t SocketConnectionReceiver::expectPacket(void)
sb.append("Expected PCKT but received ").append(subCommands[0]); sb.append("Expected PCKT but received ").append(subCommands[0]);
String str = sb.toString(); String str = sb.toString();
std::cout << str << std::endl; std::cout << str << std::endl;
throw new InvalidOperationException(str); throw InvalidOperationException(str);
} }
size_t bufferLength = subCommands[1].toULong(); size_t bufferLength = subCommands[1].toULong();
return bufferLength; return bufferLength;
} }
/// @brief Read a '\r\n' terminated string from the socket. Return the number of bytes read
/// @param line The string to read data into
/// @return
size_t SocketConnectionReceiver::readLine(String &line) size_t SocketConnectionReceiver::readLine(String &line)
{ {
size_t bytes_read = 0; size_t bytes_read = 0;
@@ -169,6 +189,9 @@ size_t SocketConnectionReceiver::readLine(String &line)
return total_bytes; return total_bytes;
} }
/// @brief Read bytes from a socket into the provided buffer.
/// @param buffer The buffer must be pre-allocated to a given length. The read operation will continue until all bytes in the buffer have been read
/// @return
size_t SocketConnectionReceiver::read(Array<char> &buffer) size_t SocketConnectionReceiver::read(Array<char> &buffer)
{ {
size_t bytes_read = 0; size_t bytes_read = 0;
@@ -185,3 +208,10 @@ size_t SocketConnectionReceiver::read(Array<char> &buffer)
} }
return total_bytes_read; return total_bytes_read;
} }
/// @brief Handle a QUIT message... which just displays
/// @param
void SocketConnectionReceiver::handleQuit(void)
{
std::cout << "Received QUIT" << std::endl;
}

View File

@@ -1,5 +1,5 @@
#ifndef _LISTENER_SOCKETCONNECTIONRECEIVER_HPP_ #ifndef _SSTP_SOCKETCONNECTIONRECEIVER_HPP_
#define _LISTENER_SOCKETCONNECTIONRECEIVER_HPP_ #define _SSTP_SOCKETCONNECTIONRECEIVER_HPP_
#include <stdio.h> #include <stdio.h>
#include <iostream> #include <iostream>
#include <sys/types.h> #include <sys/types.h>
@@ -23,6 +23,7 @@ class SocketConnectionReceiver
static constexpr size_t BUFFER_LENGTH=65536; // this is the buffer length for the socket static constexpr size_t BUFFER_LENGTH=65536; // this is the buffer length for the socket
void threadFunction(int data); void threadFunction(int data);
bool handlePut(Block<String> &commands); bool handlePut(Block<String> &commands);
void handleQuit(void);
size_t readLine(String &line); size_t readLine(String &line);
size_t expectPacket(void); size_t expectPacket(void);
size_t read(Array<char> &buffer); size_t read(Array<char> &buffer);

View File

@@ -37,17 +37,22 @@ SocketServer::SocketServer(int port)
mIsOkay=true; mIsOkay=true;
} }
/// @brief Close down the listener
SocketServer::~SocketServer() SocketServer::~SocketServer()
{ {
std::cout << "~SocketServer()" << std::endl;
close(); close();
} }
/// @brief returns true if we have a valid socket connection otherwise false
/// @return
bool SocketServer::isOkay(void) bool SocketServer::isOkay(void)
{ {
return mIsOkay; return mIsOkay;
} }
/// @brief This is the listener. It will wait for incoming connections.
/// When an incoming connection is received it will create a SocketConnectionReceiver object and call it's threadFunction to handle further interactions
/// such as transmitting a file.
void SocketServer::listen(void) void SocketServer::listen(void)
{ {
while(isOkay()) while(isOkay())
@@ -75,6 +80,8 @@ void SocketServer::listen(void)
} }
} }
/// @brief Shut down the listener socket
/// @param
void SocketServer::close(void) void SocketServer::close(void)
{ {
if(-1!=mSocketFileDescriptor) if(-1!=mSocketFileDescriptor)