Brandon Fictorie / Mbed 2 deprecated BF_Websocket

Dependencies:   mbed

Committer:
bfictorie
Date:
Sun Mar 25 17:26:30 2012 +0000
Revision:
0:8cdad1c73e8e

        

Who changed what in which revision?

UserRevisionLine numberNew contents of line
bfictorie 0:8cdad1c73e8e 1
bfictorie 0:8cdad1c73e8e 2 /*
bfictorie 0:8cdad1c73e8e 3 Copyright (c) 2010 Donatien Garnier (donatiengar [at] gmail [dot] com)
bfictorie 0:8cdad1c73e8e 4
bfictorie 0:8cdad1c73e8e 5 Permission is hereby granted, free of charge, to any person obtaining a copy
bfictorie 0:8cdad1c73e8e 6 of this software and associated documentation files (the "Software"), to deal
bfictorie 0:8cdad1c73e8e 7 in the Software without restriction, including without limitation the rights
bfictorie 0:8cdad1c73e8e 8 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
bfictorie 0:8cdad1c73e8e 9 copies of the Software, and to permit persons to whom the Software is
bfictorie 0:8cdad1c73e8e 10 furnished to do so, subject to the following conditions:
bfictorie 0:8cdad1c73e8e 11
bfictorie 0:8cdad1c73e8e 12 The above copyright notice and this permission notice shall be included in
bfictorie 0:8cdad1c73e8e 13 all copies or substantial portions of the Software.
bfictorie 0:8cdad1c73e8e 14
bfictorie 0:8cdad1c73e8e 15 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
bfictorie 0:8cdad1c73e8e 16 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
bfictorie 0:8cdad1c73e8e 17 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
bfictorie 0:8cdad1c73e8e 18 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
bfictorie 0:8cdad1c73e8e 19 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
bfictorie 0:8cdad1c73e8e 20 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
bfictorie 0:8cdad1c73e8e 21 THE SOFTWARE.
bfictorie 0:8cdad1c73e8e 22 */
bfictorie 0:8cdad1c73e8e 23
bfictorie 0:8cdad1c73e8e 24 #include "core/netservice.h"
bfictorie 0:8cdad1c73e8e 25 #include "HTTPRequestHandler.h"
bfictorie 0:8cdad1c73e8e 26
bfictorie 0:8cdad1c73e8e 27 #include <string.h>
bfictorie 0:8cdad1c73e8e 28
bfictorie 0:8cdad1c73e8e 29 //#define __DEBUG
bfictorie 0:8cdad1c73e8e 30 #include "dbg/dbg.h"
bfictorie 0:8cdad1c73e8e 31
bfictorie 0:8cdad1c73e8e 32 #define HTTP_REQUEST_TIMEOUT 5000
bfictorie 0:8cdad1c73e8e 33
bfictorie 0:8cdad1c73e8e 34 HTTPRequestHandler::HTTPRequestHandler(const char* rootPath, const char* path, TCPSocket* pTCPSocket) : NetService(),
bfictorie 0:8cdad1c73e8e 35 m_pTCPSocket(pTCPSocket), m_reqHeaders(), m_respHeaders(),
bfictorie 0:8cdad1c73e8e 36 m_rootPath(rootPath), m_path(path), m_errc(200),
bfictorie 0:8cdad1c73e8e 37 m_watchdog(), m_timeout(0), m_closed(false), m_headersSent(false) //OK
bfictorie 0:8cdad1c73e8e 38 {
bfictorie 0:8cdad1c73e8e 39 //Read & parse headers
bfictorie 0:8cdad1c73e8e 40 readHeaders();
bfictorie 0:8cdad1c73e8e 41 m_pTCPSocket->setOnEvent(this, &HTTPRequestHandler::onTCPSocketEvent);
bfictorie 0:8cdad1c73e8e 42 setTimeout(HTTP_REQUEST_TIMEOUT);
bfictorie 0:8cdad1c73e8e 43 }
bfictorie 0:8cdad1c73e8e 44
bfictorie 0:8cdad1c73e8e 45 HTTPRequestHandler::~HTTPRequestHandler()
bfictorie 0:8cdad1c73e8e 46 {
bfictorie 0:8cdad1c73e8e 47 close();
bfictorie 0:8cdad1c73e8e 48 }
bfictorie 0:8cdad1c73e8e 49
bfictorie 0:8cdad1c73e8e 50 void HTTPRequestHandler::onTimeout() //Connection has timed out
bfictorie 0:8cdad1c73e8e 51 {
bfictorie 0:8cdad1c73e8e 52 close();
bfictorie 0:8cdad1c73e8e 53 }
bfictorie 0:8cdad1c73e8e 54
bfictorie 0:8cdad1c73e8e 55 void HTTPRequestHandler::close() //Close socket and destroy data
bfictorie 0:8cdad1c73e8e 56 {
bfictorie 0:8cdad1c73e8e 57 if(m_closed)
bfictorie 0:8cdad1c73e8e 58 return;
bfictorie 0:8cdad1c73e8e 59 m_closed = true; //Prevent recursive calling or calling on an object being destructed by someone else
bfictorie 0:8cdad1c73e8e 60 m_watchdog.detach();
bfictorie 0:8cdad1c73e8e 61 onClose();
bfictorie 0:8cdad1c73e8e 62 m_pTCPSocket->resetOnEvent();
bfictorie 0:8cdad1c73e8e 63 m_pTCPSocket->close();
bfictorie 0:8cdad1c73e8e 64 delete m_pTCPSocket; //Can safely destroy socket
bfictorie 0:8cdad1c73e8e 65 NetService::close();
bfictorie 0:8cdad1c73e8e 66 }
bfictorie 0:8cdad1c73e8e 67
bfictorie 0:8cdad1c73e8e 68 map<string, string>& HTTPRequestHandler::reqHeaders() //const
bfictorie 0:8cdad1c73e8e 69 {
bfictorie 0:8cdad1c73e8e 70 return m_reqHeaders;
bfictorie 0:8cdad1c73e8e 71 }
bfictorie 0:8cdad1c73e8e 72
bfictorie 0:8cdad1c73e8e 73 string& HTTPRequestHandler::path() //const
bfictorie 0:8cdad1c73e8e 74 {
bfictorie 0:8cdad1c73e8e 75 return m_path;
bfictorie 0:8cdad1c73e8e 76 }
bfictorie 0:8cdad1c73e8e 77
bfictorie 0:8cdad1c73e8e 78 int HTTPRequestHandler::dataLen() const
bfictorie 0:8cdad1c73e8e 79 {
bfictorie 0:8cdad1c73e8e 80 map<string,string>::iterator it;
bfictorie 0:8cdad1c73e8e 81 it = m_reqHeaders.find("Content-Length");
bfictorie 0:8cdad1c73e8e 82 if( it == m_reqHeaders.end() )
bfictorie 0:8cdad1c73e8e 83 {
bfictorie 0:8cdad1c73e8e 84 return 0;
bfictorie 0:8cdad1c73e8e 85 }
bfictorie 0:8cdad1c73e8e 86 return atoi((*it).second.c_str()); //return 0 if parse fails, so that's fine
bfictorie 0:8cdad1c73e8e 87 }
bfictorie 0:8cdad1c73e8e 88
bfictorie 0:8cdad1c73e8e 89 int HTTPRequestHandler::readData(char* buf, int len)
bfictorie 0:8cdad1c73e8e 90 {
bfictorie 0:8cdad1c73e8e 91 return m_pTCPSocket->recv(buf, len);
bfictorie 0:8cdad1c73e8e 92 }
bfictorie 0:8cdad1c73e8e 93
bfictorie 0:8cdad1c73e8e 94 string& HTTPRequestHandler::rootPath() //const
bfictorie 0:8cdad1c73e8e 95 {
bfictorie 0:8cdad1c73e8e 96 return m_rootPath;
bfictorie 0:8cdad1c73e8e 97 }
bfictorie 0:8cdad1c73e8e 98
bfictorie 0:8cdad1c73e8e 99 void HTTPRequestHandler::setErrCode(int errc)
bfictorie 0:8cdad1c73e8e 100 {
bfictorie 0:8cdad1c73e8e 101 m_errc = errc;
bfictorie 0:8cdad1c73e8e 102 }
bfictorie 0:8cdad1c73e8e 103
bfictorie 0:8cdad1c73e8e 104 void HTTPRequestHandler::setContentLen(int len)
bfictorie 0:8cdad1c73e8e 105 {
bfictorie 0:8cdad1c73e8e 106 char len_str[6] = {0};
bfictorie 0:8cdad1c73e8e 107 sprintf(len_str, "%d", len);
bfictorie 0:8cdad1c73e8e 108 respHeaders()["Content-Length"] = len_str;
bfictorie 0:8cdad1c73e8e 109 }
bfictorie 0:8cdad1c73e8e 110
bfictorie 0:8cdad1c73e8e 111 map<string, string>& HTTPRequestHandler::respHeaders()
bfictorie 0:8cdad1c73e8e 112 {
bfictorie 0:8cdad1c73e8e 113 return m_respHeaders;
bfictorie 0:8cdad1c73e8e 114 }
bfictorie 0:8cdad1c73e8e 115
bfictorie 0:8cdad1c73e8e 116 int HTTPRequestHandler::writeData(const char* buf, int len)
bfictorie 0:8cdad1c73e8e 117 {
bfictorie 0:8cdad1c73e8e 118 if(!m_headersSent)
bfictorie 0:8cdad1c73e8e 119 {
bfictorie 0:8cdad1c73e8e 120 m_headersSent = true;
bfictorie 0:8cdad1c73e8e 121 writeHeaders();
bfictorie 0:8cdad1c73e8e 122 }
bfictorie 0:8cdad1c73e8e 123
bfictorie 0:8cdad1c73e8e 124 return m_pTCPSocket->send(buf, len);
bfictorie 0:8cdad1c73e8e 125 }
bfictorie 0:8cdad1c73e8e 126
bfictorie 0:8cdad1c73e8e 127 void HTTPRequestHandler::setTimeout(int ms)
bfictorie 0:8cdad1c73e8e 128 {
bfictorie 0:8cdad1c73e8e 129 m_timeout = 1000*ms;
bfictorie 0:8cdad1c73e8e 130 resetTimeout();
bfictorie 0:8cdad1c73e8e 131 }
bfictorie 0:8cdad1c73e8e 132
bfictorie 0:8cdad1c73e8e 133 void HTTPRequestHandler::resetTimeout()
bfictorie 0:8cdad1c73e8e 134 {
bfictorie 0:8cdad1c73e8e 135 m_watchdog.detach();
bfictorie 0:8cdad1c73e8e 136 m_watchdog.attach_us<HTTPRequestHandler>(this, &HTTPRequestHandler::onTimeout, m_timeout);
bfictorie 0:8cdad1c73e8e 137 }
bfictorie 0:8cdad1c73e8e 138
bfictorie 0:8cdad1c73e8e 139
bfictorie 0:8cdad1c73e8e 140 void HTTPRequestHandler::readHeaders()
bfictorie 0:8cdad1c73e8e 141 {
bfictorie 0:8cdad1c73e8e 142 static char line[128];
bfictorie 0:8cdad1c73e8e 143 static char key[128];
bfictorie 0:8cdad1c73e8e 144 static char value[128];
bfictorie 0:8cdad1c73e8e 145 while( readLine(line, 128) > 0) //if == 0, it is an empty line = end of headers
bfictorie 0:8cdad1c73e8e 146 {
bfictorie 0:8cdad1c73e8e 147 int n = sscanf(line, "%[^:]: %[^\n]", key, value);
bfictorie 0:8cdad1c73e8e 148 if ( n == 2 )
bfictorie 0:8cdad1c73e8e 149 {
bfictorie 0:8cdad1c73e8e 150 DBG("\r\nRead header : %s : %s\r\n", key, value);
bfictorie 0:8cdad1c73e8e 151 m_reqHeaders[key] = value;
bfictorie 0:8cdad1c73e8e 152 }
bfictorie 0:8cdad1c73e8e 153 //TODO: Impl n==1 case (part 2 of previous header)
bfictorie 0:8cdad1c73e8e 154 }
bfictorie 0:8cdad1c73e8e 155 }
bfictorie 0:8cdad1c73e8e 156
bfictorie 0:8cdad1c73e8e 157 void HTTPRequestHandler::writeHeaders() //Called at the first writeData call
bfictorie 0:8cdad1c73e8e 158 {
bfictorie 0:8cdad1c73e8e 159 static char line[128];
bfictorie 0:8cdad1c73e8e 160
bfictorie 0:8cdad1c73e8e 161 //Response line
bfictorie 0:8cdad1c73e8e 162 sprintf(line, "HTTP/1.1 %d MbedInfo\r\n", m_errc); //Not a violation of the standard not to include the descriptive text
bfictorie 0:8cdad1c73e8e 163 m_pTCPSocket->send(line, strlen(line));
bfictorie 0:8cdad1c73e8e 164
bfictorie 0:8cdad1c73e8e 165 map<string,string>::iterator it;
bfictorie 0:8cdad1c73e8e 166 while( !m_respHeaders.empty() )
bfictorie 0:8cdad1c73e8e 167 {
bfictorie 0:8cdad1c73e8e 168 it = m_respHeaders.begin();
bfictorie 0:8cdad1c73e8e 169 sprintf(line, "%s: %s\r\n", (*it).first.c_str(), (*it).second.c_str() );
bfictorie 0:8cdad1c73e8e 170 DBG("\r\n%s", line);
bfictorie 0:8cdad1c73e8e 171 m_pTCPSocket->send(line, strlen(line));
bfictorie 0:8cdad1c73e8e 172 m_respHeaders.erase(it);
bfictorie 0:8cdad1c73e8e 173 }
bfictorie 0:8cdad1c73e8e 174 m_pTCPSocket->send("\r\n",2); //End of head
bfictorie 0:8cdad1c73e8e 175 }
bfictorie 0:8cdad1c73e8e 176
bfictorie 0:8cdad1c73e8e 177 int HTTPRequestHandler::readLine(char* str, int maxLen)
bfictorie 0:8cdad1c73e8e 178 {
bfictorie 0:8cdad1c73e8e 179 int ret;
bfictorie 0:8cdad1c73e8e 180 int len = 0;
bfictorie 0:8cdad1c73e8e 181 for(int i = 0; i < maxLen - 1; i++)
bfictorie 0:8cdad1c73e8e 182 {
bfictorie 0:8cdad1c73e8e 183 ret = m_pTCPSocket->recv(str, 1);
bfictorie 0:8cdad1c73e8e 184 if(!ret)
bfictorie 0:8cdad1c73e8e 185 {
bfictorie 0:8cdad1c73e8e 186 break;
bfictorie 0:8cdad1c73e8e 187 }
bfictorie 0:8cdad1c73e8e 188 if( (len > 1) && *(str-1)=='\r' && *str=='\n' )
bfictorie 0:8cdad1c73e8e 189 {
bfictorie 0:8cdad1c73e8e 190 str--;
bfictorie 0:8cdad1c73e8e 191 len-=2;
bfictorie 0:8cdad1c73e8e 192 break;
bfictorie 0:8cdad1c73e8e 193 }
bfictorie 0:8cdad1c73e8e 194 else if( *str=='\n' )
bfictorie 0:8cdad1c73e8e 195 {
bfictorie 0:8cdad1c73e8e 196 len--;
bfictorie 0:8cdad1c73e8e 197 break;
bfictorie 0:8cdad1c73e8e 198 }
bfictorie 0:8cdad1c73e8e 199 str++;
bfictorie 0:8cdad1c73e8e 200 len++;
bfictorie 0:8cdad1c73e8e 201 }
bfictorie 0:8cdad1c73e8e 202 *str = 0;
bfictorie 0:8cdad1c73e8e 203 return len;
bfictorie 0:8cdad1c73e8e 204 }
bfictorie 0:8cdad1c73e8e 205
bfictorie 0:8cdad1c73e8e 206 void HTTPRequestHandler::onTCPSocketEvent(TCPSocketEvent e)
bfictorie 0:8cdad1c73e8e 207 {
bfictorie 0:8cdad1c73e8e 208
bfictorie 0:8cdad1c73e8e 209 DBG("\r\nEvent %d in HTTPRequestHandler\r\n", e);
bfictorie 0:8cdad1c73e8e 210
bfictorie 0:8cdad1c73e8e 211 if(m_closed)
bfictorie 0:8cdad1c73e8e 212 {
bfictorie 0:8cdad1c73e8e 213 DBG("\r\nWARN: Discarded\r\n");
bfictorie 0:8cdad1c73e8e 214 return;
bfictorie 0:8cdad1c73e8e 215 }
bfictorie 0:8cdad1c73e8e 216
bfictorie 0:8cdad1c73e8e 217 switch(e)
bfictorie 0:8cdad1c73e8e 218 {
bfictorie 0:8cdad1c73e8e 219 case TCPSOCKET_READABLE:
bfictorie 0:8cdad1c73e8e 220 resetTimeout();
bfictorie 0:8cdad1c73e8e 221 onReadable();
bfictorie 0:8cdad1c73e8e 222 break;
bfictorie 0:8cdad1c73e8e 223 case TCPSOCKET_WRITEABLE:
bfictorie 0:8cdad1c73e8e 224 resetTimeout();
bfictorie 0:8cdad1c73e8e 225 onWriteable();
bfictorie 0:8cdad1c73e8e 226 break;
bfictorie 0:8cdad1c73e8e 227 case TCPSOCKET_CONTIMEOUT:
bfictorie 0:8cdad1c73e8e 228 case TCPSOCKET_CONRST:
bfictorie 0:8cdad1c73e8e 229 case TCPSOCKET_CONABRT:
bfictorie 0:8cdad1c73e8e 230 case TCPSOCKET_ERROR:
bfictorie 0:8cdad1c73e8e 231 case TCPSOCKET_DISCONNECTED:
bfictorie 0:8cdad1c73e8e 232 DBG("\r\nConnection error in handler\r\n");
bfictorie 0:8cdad1c73e8e 233 close();
bfictorie 0:8cdad1c73e8e 234 break;
bfictorie 0:8cdad1c73e8e 235 }
bfictorie 0:8cdad1c73e8e 236
bfictorie 0:8cdad1c73e8e 237 }