diff --git a/software/inc/webtools.hpp b/software/inc/webtools.hpp new file mode 100644 index 0000000..acccd94 --- /dev/null +++ b/software/inc/webtools.hpp @@ -0,0 +1,27 @@ +/** + * Let There Be Light project + * + * @file remoteRequest.hxx + */ + +/* Includes -------------------------------------------- */ + +/* Defines --------------------------------------------- */ + +/* Type definitions ------------------------------------ */ +typedef enum _restMethods { + REST_UNKNOWN = 0, + REST_INDEX, /* Shows the index page */ + REST_GET, /* GET is used to request data from a specified resource. */ + REST_POST, /* POST is used to send data to a server to create/update a resource. */ + REST_PUT, /* PUT is used to send data to a server to create/update a resource. */ + REST_HEAD, /* HEAD is almost identical to GET, but without the response body. */ + REST_DELETE, /* The DELETE method deletes the specified resource. */ + REST_PATCH, + REST_OPTIONS /* The OPTIONS method describes the communication options for the target resource. */ +} restMethod_t; + +/* Variable declarations ------------------------------- */ + +/* Functions ------------------------------------------- */ +int acceptRequest(const int pSocket, int * const pResult); \ No newline at end of file diff --git a/software/src/webtools.cpp b/software/src/webtools.cpp new file mode 100644 index 0000000..0a29c53 --- /dev/null +++ b/software/src/webtools.cpp @@ -0,0 +1,588 @@ +/** + * Let There Be Light project + * + * @file remoteRequest.cxx + */ + +/* Includes -------------------------------------------- */ +#include "remoteRequest.hxx" + +#include "Relay.hxx" + +#include "HttpRequest.hxx" + +/* C++ System */ +#include + +/* C System */ +#include + +#include + +#include +#include + +#include + +#include /* For errno */ +#include /* For strerror */ + +/* Defines --------------------------------------------- */ +#define IS_SPACE(x) isspace((int)(x)) + +/* Variable declarations ------------------------------- */ +extern elec::Relay sRelay; + +/* Forward declaration of private functions ------------ */ +int htmlSend(const int pClient, const char * const pStr); +int htmlSend(const int pClient, const std::string pStr); + +int getLine(int pSock, char *pBuf, int pSize); +void cat(const int pClient, FILE * const pResource); +void serveFile(const int pClient, const char * const pFileName); +void errorDie(const char * const sc); + +void unimplemented(const int pClient); +void badRequest(const int pClient); +void cannotExecute(const int pClient); +void headers(const int pClient, const char * const pFileName); +void notFound(const int pClient); +void home(const int pClient); +void togglePage(const int pClient); + +/* Functions ------------------------------------------- */ +int htmlSend(const int pClient, const char * const pStr) { + int lResult = 0; + + errno = 0; + lResult = send(pClient, pStr, strlen(pStr), 0); + if(0 > lResult) { + std::cerr << "[ERROR] send failed with error code " << lResult << std::endl; + if(errno) { + std::cerr << " errno = " << errno << ", " << strerror(errno) << std::endl; + } + } + + return lResult; +} + +int htmlSend(const int pClient, const std::string pStr) { + int lResult = 0; + + errno = 0; + lResult = send(pClient, pStr.c_str(), pStr.length(), 0); + if(0 > lResult) { + std::cerr << "[ERROR] send failed with error code " << lResult << std::endl; + if(errno) { + std::cerr << " errno = " << errno << ", " << strerror(errno) << std::endl; + } + } + + return lResult; +} + +/**********************************************************************/ +/* Get a line from a socket, whether the line ends in a newline, + * carriage return, or a CRLF combination. Terminates the string read + * with a null character. If no newline indicator is found before the + * end of the buffer, the string is terminated with a null. If any of + * the above three line terminators is read, the last character of the + * string will be a linefeed and the string will be terminated with a + * null character. + * Parameters: the socket descriptor + * the buffer to save the data in + * the size of the buffer + * Returns: the number of bytes stored (excluding null) */ +/**********************************************************************/ +int getLine(int pSock, char *pBuf, int pSize) +{ + /* Variable declarations */ + char c = '\0'; + char *lBuf = pBuf; + bool lGotCR = false; + int lReadChars = 0, i = 0; + + /* Loop over the line */ + do { + /* Read a character */ + errno = 0; + lReadChars = recv(pSock, &c, 1U, 0); + + /* Check errno */ + bool lErrnoBreak = false; + switch (errno) + { + case 0: + /* Nothing to do, everything is fine ! */ + break; +#if EAGAIN != EWOULDBLOCK + case EAGAIN: +#endif /* EAGAIN != EWOULDBLOCK */ + case EWOULDBLOCK: + /* Socket is O_NONBLOCK and there is no data available */ + std::cerr << "[WARN ] recv got EAGAIN, this is normal, nothing received, errno = " << errno << " (" << std::string(strerror(errno)) << ")" << std::endl; + lErrnoBreak = true; + break; + case EINTR: + /* An interrupt (signal) has been caught */ + /* This should be ingored in most cases */ + std::cerr << "[ERROR] recv got EINTR, an interrupt (signal) has been caught. Should be ignored. errno = " << errno << " (" << std::string(strerror(errno)) << ")" << std::endl; + lErrnoBreak = true; + break; + default: + /* socket has an error, not valid anymore */ + std::cerr << "[ERROR] recv failed, errno = " << errno << " (" << std::string(strerror(errno)) << ")" << std::endl; + //std::cout << "[DEBUG] c = " << c << ", i.e. : " << std::hex << (unsigned int)c << std::dec << std::endl; + return -1; + } + + /** Break the do/while loop if we caught an errno + * that requires a break of loop */ + if(lErrnoBreak) { + break; + } + + /* Check lReadChars */ + if(0 > lReadChars) { + std::cerr << "[ERROR] recv failed !" << std::endl; + return -1; + } else if (0 == lReadChars) { + //std::cout << "[WARN ] Unexpected behavior, read 0 chars from socket but no errors..." << std::endl; + /* The socket buffer is empty, return */ + break; + } + + /* Put the character in the buffer */ + *lBuf = c; + ++lBuf; + + /* Increment the number of chars read */ + ++i; + + /* Check for CR */ + if('\r' == c) { + lGotCR = true; + /* e expect to get a LF afterwards */ + //std::cout << "[DEBUG] Got CR, expecting an LF..." << std::endl; + } + + /** Check the character for LF + * Note : In the HTTP protocol, the CR-LF sequence is always used to terminate a line. + */ + if('\n' == c) { + if(!lGotCR) { + std::cout << "[WARN ] Got LF without CR, doesn't comply with HTTP protocols !" << std::endl; + } else { + //std::cout << "[DEBUG] Got LF after CR !" << std::endl; + } + break; + } + + if('\0' == c) { + //std::cout << "[DEBUG] Got \\0, exiting..." << std::endl; + break; + } + + //std::cout << "[DEBUG] c = " << c << ", i.e. : " << std::hex << (unsigned int)c << std::dec << std::endl; + // { + // std::string pBufString(pBuf); + // std::cout << "[DEBUG] pBufString = " << std::string(pBufString) << std::endl; + // if(pBufString.empty()) { + // std::cout << "[DEBUG] c = " << c << ", i.e. : " << std::hex << (unsigned int)c << std::dec << std::endl; + // } + // } + + } while(true); + + return i; +} + +/**********************************************************************/ +/* Put the entire contents of a file out on a socket. This function + * is named after the UNIX "cat" command, because it might have been + * easier just to do something like pipe, fork, and exec("cat"). + * Parameters: the client socket descriptor + * FILE pointer for the file to cat */ +/**********************************************************************/ +void cat(const int pClient, FILE * const pResource) { + char lBuf[1024U]; + memset(lBuf, 0, 1024U); + + /* Get the contents of the file */ + fgets(lBuf, sizeof(lBuf), pResource); + + /* Send the contents of the file into the socket */ + while (!feof(pResource)) { + errno = 0; + send(pClient, lBuf, strlen(lBuf), 0); + if(errno) { + std::cerr << "[ERROR] send failed : errno = " << errno << ", " << strerror(errno) << std::endl; + } + fgets(lBuf, sizeof(lBuf), pResource); + } +} + +/**********************************************************************/ +/* Send a regular file to the client. Use headers, and report + * errors to client if they occur. + * Parameters: a pointer to a file structure produced from the socket + * file descriptor + * the name of the file to serve */ +/**********************************************************************/ +void serveFile(const int pClient, const char * const pFileName) { + FILE *lResource = nullptr; + unsigned int lNumChars = 1; + char lBuf[1024U]; + memset(lBuf, 0, 1024U); + + lBuf[0U] = 'A'; + lBuf[1U] = '\0'; + + while ((lNumChars > 0) && strcmp("\n", lBuf)) { /* read & discard headers */ + lNumChars = getLine(pClient, lBuf, sizeof(lBuf)); + } + + lResource = fopen(pFileName, "r"); + if (lResource == NULL) { + notFound(pClient); + } else { + headers(pClient, pFileName); + cat(pClient, lResource); + } + + fclose(lResource); +} + +/**********************************************************************/ +/* Print out an error message with perror() (for system errors; based + * on value of errno, which indicates system call errors) and exit the + * program indicating an error. */ +/**********************************************************************/ +void error_die(const char * const sc) +{ + perror(sc); + exit(EXIT_FAILURE); +} + +/**********************************************************************/ +/* Inform the client that the requested web method has not been + * implemented. + * Parameter: the client socket */ +/**********************************************************************/ +void unimplemented(const int pClient) +{ + std::string lUnimplemented = "HTTP/1.0 501 Method Not Implemented\r\n" + "Content-Type: text/html\r\n" + "\r\n" + "\r\n" + " \r\n" + " Method Not Implemented\r\n" + " \r\n" + " \r\n" + " HTTP request method not supported.\r\n" + " \r\n" + "\r\n\r\n"; + + const int lResult = htmlSend(pClient, lUnimplemented); + if(0 > lResult) { + std::cerr << "[ERROR] htmlSend failed with return code " << lResult << std::endl; + } +} + +/**********************************************************************/ +/* Inform the client that a request it has made has a problem. + * Parameters: client socket */ +/**********************************************************************/ +void badRequest(const int pClient) +{ + std::string lBadReq = "HTTP/1.0 400 BAD REQUEST\r\n" + "Content-Type: text/html\r\n" + "\r\n" + "" + " \r\n" + " web-example - 500 Server error\r\n" + " \r\n" + " \r\n" + " Your browser sent a bad request, " + " such as a POST without a Content-Length.\r\n" + " \r\n" + "\r\n\r\n"; + + const int lResult = htmlSend(pClient, lBadReq); + if(0 > lResult) { + std::cerr << "[ERROR] htmlSend failed with return code " << lResult << std::endl; + } +} + +/**********************************************************************/ +/* Inform the client that a CGI script could not be executed. + * Parameter: the client socket descriptor. */ +/**********************************************************************/ +void cannotExecute(const int pClient) +{ + std::string lExeError = "HTTP/1.0 500 Internal Server Error\r\n" + "Content-Type: text/html\r\n" + "\r\n" + "" + " \r\n" + " web-example - 500 Server error\r\n" + " \r\n" + " \r\n" + " Server-side error prohibited execution." + " \r\n" + "\r\n\r\n"; + + const int lResult = htmlSend(pClient, lExeError); + if(0 > lResult) { + std::cerr << "[ERROR] htmlSend failed with return code " << lResult << std::endl; + } +} + +/**********************************************************************/ +/* Return the informational HTTP headers about a file. */ +/* Parameters: the socket to print the headers on + * the name of the file */ +/**********************************************************************/ +void headers(const int pClient, const char * const pFileName) +{ + std::string lHeader = "HTTP/1.0 200 Ok\r\n" + "Content-Type: text/html\r\n" + "\r\n"; + + const int lResult = htmlSend(pClient, lHeader); + if(0 > lResult) { + std::cerr << "[ERROR] htmlSend failed with return code " << lResult << std::endl; + } +} + +/**********************************************************************/ +/* Give a client a 404 not found status message. */ +/**********************************************************************/ +void notFound(const int pClient) +{ + std::string l404 = "HTTP/1.0 404 NOT FOUND\r\n" + "Content-Type: text/html\r\n" + "\r\n" + "" + " \r\n" + " web-example - 404 Not Found\r\n" + " \r\n" + " \r\n" + " The server could not fulfill your request because the resource specified\r\n" + " is unavailable or nonexistent.\r\n" + "
\r\n" + " Please check that you entered the correct URL.\r\n" + " \r\n" + "\r\n\r\n"; + + const int lResult = htmlSend(pClient, l404); + if(0 > lResult) { + std::cerr << "[ERROR] htmlSend failed with return code " << lResult << std::endl; + } +} + +/**********************************************************************/ +/* Give a client a "found" status message. */ +/**********************************************************************/ +void home(const int pClient) { + std::string lHome = "HTTP/1.0 200 Ok\r\n" + "Content-Type: text/html\r\n" + "\r\n" + "" + " \r\n" + " web-example\r\n" + " \r\n" + " \r\n" + " Hello there !
\r\n" + " General Kenobi !\r\n" + " \r\n" + "\r\n\r\n"; + + const int lResult = htmlSend(pClient, lHome); + if(0 > lResult) { + std::cerr << "[ERROR] htmlSend failed with return code " << lResult << std::endl; + } +} + +void togglePage(const int pClient) { + std::string lTogglePage = "HTTP/1.0 200 Ok\r\n" + "Content-Type: text/html\r\n" + "\r\n" + "\r\n" + " \r\n" + " \r\n" + "\r\n" + " \r\n" + " web-example - Toggle switch\r\n" + "\r\n" + "\r\n" + " \r\n" + "

Toggle Switch

\r\n" + "

Switch is currently at value : " + std::string((sRelay.isOn() ? "ON" : "OFF")) + "

\r\n" + "
\r\n" + " \r\n" + "
\r\n" + " \r\n" + "\r\n\r\n"; + + std::cout << "[DEBUG] Sending the toggle page code !" << std::endl; + + const int lResult = htmlSend(pClient, lTogglePage); + if(0 > lResult) { + std::cerr << "[ERROR] htmlSend failed with return code " << lResult << std::endl; + } else { + //std::cout << "[DEBUG] Toggle page code sent :" << std::endl << std::endl << lTogglePage << std::endl; + } +} + +/**********************************************************************/ +/* A request has caused a call to accept() on the server port to + * return. Process the request appropriately. + * Parameters: the socket connected to the client */ +/**********************************************************************/ +int acceptRequest(const int pClient, int * const pResult) { + if(nullptr == pResult) { + std::cerr << "[ERROR] Result is a nullptr !" << std::endl; + return -1; + } + + int lNumChars = 0; + + HttpRequest lRequest; + + char lBuf[1024U]; + memset(lBuf, 0, 1024U); + + std::string lRequestStr; + do { + lNumChars = 0; + /* Reset the buffer to receive the line */ + memset(lBuf, 0, 1024U); + + /* Get the line */ + lNumChars = getLine(pClient, lBuf, sizeof(lBuf)); + //std::cout << "[DEBUG] Got \"" << std::string(lBuf) << "\" from the client. " << std::endl; + + if(0 > lNumChars) { + std::cerr << "[ERROR] getLine failed, returned " << lNumChars << std::endl; + return -1; + } else if(0 == lNumChars) { + std::cout << "[WARN ] Read 0 characters !" << std::endl; + } else if(std::string(lBuf).empty()) { + std::cout << "[WARN ] Got an empty line !" << std::endl; + if(lNumChars) { + std::cerr << "[ERROR] Got an empty line, but lNumChars = " << lNumChars << std::endl; + return -1; + } + } + + /* Append the line to the request */ + lRequestStr += std::string(lBuf); + } while(0 < lNumChars); + + /* Print out what we received */ + //std::cout << "[DEBUG] lRequestStr = \"" << lRequestStr << "\"" << std::endl; + if(lRequestStr.empty()) { + std::cout << "[WARN ] Go an empty request, exiting..." << std::endl; + close(pClient); + return -1; + } + lRequest.parseRequest(lRequestStr); + + /* Get the method from the client's request */ + std::string lMethod = lRequest.method(); + std::cout << "[DEBUG] The method is " << lMethod << std::endl; + + /* Get the URL of the request from the client's request */ + std::cout << "[DEBUG] The complete URL is : " << lRequest.URL() << std::endl; + std::cout << "[DEBUG] The short URL is : " << lRequest.shortURL() << std::endl; + std::string lURL = lRequest.shortURL(); + + /* The real query is located after the "?" symbol */ + std::string lQuery = lRequest.query(); + if(lQuery.empty()) { + std::cout << "[DEBUG] No query found. " << std::endl; + } else { + std::cout << "[DEBUG] The query is " << lQuery << std::endl; + } + + /* What method is it ? */ + restMethod_t lREST = REST_UNKNOWN; + if(0 == strcasecmp("GET", lMethod.c_str())) { + lREST = REST_GET; + } else if (0 == strcasecmp("POST", lMethod.c_str())) { + lREST = REST_POST; + } else if (0 == strcasecmp("PUT", lMethod.c_str())) { + lREST = REST_PUT; + } else if (0 == strcasecmp("HEAD", lMethod.c_str())) { + lREST = REST_HEAD; + } else if (0 == strcasecmp("DELETE", lMethod.c_str())) { + lREST = REST_DELETE; + } else if (0 == strcasecmp("PATCH", lMethod.c_str())) { + lREST = REST_PATCH; + } else if (0 == strcasecmp("OPTIONS", lMethod.c_str())) { + lREST = REST_OPTIONS; + } else { + unimplemented(pClient); + std::cout << "[ERROR] Unknown REST method requested ! " << std::endl; + close(pClient); + return -1; + } + + switch(lREST) { + case REST_GET: + std::cout << "[DEBUG] Request is GET" << std::endl; + + /* What is the URL ? */ + if("/" == lURL || "/index.html" == lURL) { + /* Home page */ + } else if ("/toggle" == lURL) { + /* Switch the relays state */ + sRelay.switchState(); + + /* Check the relay's state, print it out */ + if(sRelay.isOn()) { + std::cout << "[DEBUG] Toggled relay, is now ON" << std::endl; + } else { + std::cout << "[DEBUG] Toggled relay, is now OFF" << std::endl; + } + } else { + /* Unimplemented */ + notFound(pClient); + std::cout << "[ERROR] Unknown URL requested !" << std::endl; + close(pClient); + return -1; + } + + /* Send back the toggle page */ + togglePage(pClient); + break; + case REST_POST: + std::cout << "[DEBUG] Request is POST" << std::endl; + unimplemented(pClient); + close(pClient); + return -1; + break; + case REST_INDEX: + std::cout << "[DEBUG] Request is INDEX" << std::endl; + home(pClient); + break; + case REST_PUT: + std::cout << "[DEBUG] Request is PUT" << std::endl; + unimplemented(pClient); + close(pClient); + return -1; + case REST_UNKNOWN: + default: + std::cout << "[ERROR] Request is unknown" << std::endl; + unimplemented(pClient); + close(pClient); + return -1; + } + + /* Close the client */ + usleep(10000); + close(pClient); + + return 0; +} \ No newline at end of file