Example program with HTTPServer and sensor data streaming over TCPSockets, using Donatien Garnier's Net APIs and services code on top of LWIP. Files StreamServer.h and .cpp encapsulate streaming over TCPSockets. Broadcast is done by sendToAll(), and all incoming data is echoed back to the client. Echo code can be replaced with some remote control of the streaming interface. See main() that shows how to periodically send some data to all subscribed clients. To subscribe, a client should open a socket at <mbed_ip> port 123. I used few lines in TCL code to set up a quick sink for the data. HTTP files are served on port 80 concurrently to the streaming.

Dependencies:   mbed

Committer:
iva2k
Date:
Sat Jun 12 06:01:50 2010 +0000
Revision:
0:e614f7875b60
Child:
1:3ee499525aa5

        

Who changed what in which revision?

UserRevisionLine numberNew contents of line
iva2k 0:e614f7875b60 1
iva2k 0:e614f7875b60 2 /*
iva2k 0:e614f7875b60 3 Copyright (c) 2010 IVA2K
iva2k 0:e614f7875b60 4
iva2k 0:e614f7875b60 5 Permission is hereby granted, free of charge, to any person obtaining a copy
iva2k 0:e614f7875b60 6 of this software and associated documentation files (the "Software"), to deal
iva2k 0:e614f7875b60 7 in the Software without restriction, including without limitation the rights
iva2k 0:e614f7875b60 8 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
iva2k 0:e614f7875b60 9 copies of the Software, and to permit persons to whom the Software is
iva2k 0:e614f7875b60 10 furnished to do so, subject to the following conditions:
iva2k 0:e614f7875b60 11
iva2k 0:e614f7875b60 12 The above copyright notice and this permission notice shall be included in
iva2k 0:e614f7875b60 13 all copies or substantial portions of the Software.
iva2k 0:e614f7875b60 14
iva2k 0:e614f7875b60 15 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
iva2k 0:e614f7875b60 16 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
iva2k 0:e614f7875b60 17 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
iva2k 0:e614f7875b60 18 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
iva2k 0:e614f7875b60 19 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
iva2k 0:e614f7875b60 20 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
iva2k 0:e614f7875b60 21 THE SOFTWARE.
iva2k 0:e614f7875b60 22 */
iva2k 0:e614f7875b60 23
iva2k 0:e614f7875b60 24 #include "StreamServer.h"
iva2k 0:e614f7875b60 25
iva2k 0:e614f7875b60 26 #include "dbg/dbg.h"
iva2k 0:e614f7875b60 27
iva2k 0:e614f7875b60 28
iva2k 0:e614f7875b60 29 //BEGIN REQUEST DISPATCHER======================================================
iva2k 0:e614f7875b60 30 StreamRequestDispatcher::StreamRequestDispatcher(StreamServer* pSvr, TCPSocket* pTcpSocket)
iva2k 0:e614f7875b60 31 : NetService(), m_pSvr(pSvr), m_pTcpSocket(pTcpSocket), m_closed(false)
iva2k 0:e614f7875b60 32 {
iva2k 0:e614f7875b60 33 m_pTcpSocket->setOnEvent(this, &StreamRequestDispatcher::onTcpSocketEvent);
iva2k 0:e614f7875b60 34 }
iva2k 0:e614f7875b60 35
iva2k 0:e614f7875b60 36 StreamRequestDispatcher::~StreamRequestDispatcher()
iva2k 0:e614f7875b60 37 {
iva2k 0:e614f7875b60 38 close();
iva2k 0:e614f7875b60 39 }
iva2k 0:e614f7875b60 40
iva2k 0:e614f7875b60 41 int StreamRequestDispatcher::writeData(const char* buf, int len)
iva2k 0:e614f7875b60 42 {
iva2k 0:e614f7875b60 43 if(m_closed) {
iva2k 0:e614f7875b60 44 DBG("\r\nIN StreamRequestDispatcher::writeData - m_closed\r\n");
iva2k 0:e614f7875b60 45 return TCPSOCKET_RST;
iva2k 0:e614f7875b60 46 }
iva2k 0:e614f7875b60 47 int ret = m_pTcpSocket->send(buf, len);
iva2k 0:e614f7875b60 48 if (ret != len) DBG("\r\nStreamRequestDispatcher::writeData - ret=%d\r\n", ret);
iva2k 0:e614f7875b60 49 return ret;
iva2k 0:e614f7875b60 50 }
iva2k 0:e614f7875b60 51
iva2k 0:e614f7875b60 52 //FIXME: Need implementation for StreamServer
iva2k 0:e614f7875b60 53 void StreamRequestDispatcher::dispatchRequest()
iva2k 0:e614f7875b60 54 {
iva2k 0:e614f7875b60 55 string request;
iva2k 0:e614f7875b60 56 const char* buf;
iva2k 0:e614f7875b60 57 int len, ret;
iva2k 0:e614f7875b60 58 DBG("\r\nIN StreamRequestDispatcher::dispatchRequest()\r\n");
iva2k 0:e614f7875b60 59
iva2k 0:e614f7875b60 60 while ( getRequest(&request) )
iva2k 0:e614f7875b60 61 {
iva2k 0:e614f7875b60 62 // Just echo it back
iva2k 0:e614f7875b60 63 buf = request.c_str();
iva2k 0:e614f7875b60 64 len = strlen(buf);
iva2k 0:e614f7875b60 65 ret = writeData(buf,len);
iva2k 0:e614f7875b60 66 writeData("\r\n",2);
iva2k 0:e614f7875b60 67 DBG("\r\nReceived req (%s), write ret=%d\r\n", buf, ret);
iva2k 0:e614f7875b60 68 }
iva2k 0:e614f7875b60 69 }
iva2k 0:e614f7875b60 70
iva2k 0:e614f7875b60 71 void StreamRequestDispatcher::close() //Close socket and destroy data
iva2k 0:e614f7875b60 72 {
iva2k 0:e614f7875b60 73 if(m_closed)
iva2k 0:e614f7875b60 74 return;
iva2k 0:e614f7875b60 75 m_closed = true; //Prevent recursive calling or calling on an object being destructed by someone else
iva2k 0:e614f7875b60 76 m_pSvr->m_lpClients.remove(this);
iva2k 0:e614f7875b60 77 if(m_pTcpSocket)
iva2k 0:e614f7875b60 78 {
iva2k 0:e614f7875b60 79 m_pTcpSocket->resetOnEvent();
iva2k 0:e614f7875b60 80 m_pTcpSocket->close();
iva2k 0:e614f7875b60 81 delete m_pTcpSocket;
iva2k 0:e614f7875b60 82 }
iva2k 0:e614f7875b60 83 NetService::close();
iva2k 0:e614f7875b60 84 }
iva2k 0:e614f7875b60 85
iva2k 0:e614f7875b60 86 //FIXME: Need implementation for StreamServer
iva2k 0:e614f7875b60 87 bool StreamRequestDispatcher::getRequest(string* request)
iva2k 0:e614f7875b60 88 {
iva2k 0:e614f7875b60 89 const int maxLen = 64;
iva2k 0:e614f7875b60 90 char req[maxLen];
iva2k 0:e614f7875b60 91 //Read Line
iva2k 0:e614f7875b60 92 int ret;
iva2k 0:e614f7875b60 93 int len = 0;
iva2k 0:e614f7875b60 94 char* p = req;
iva2k 0:e614f7875b60 95 for(int i = 0; i < maxLen - 1; i++)
iva2k 0:e614f7875b60 96 {
iva2k 0:e614f7875b60 97 ret = m_pTcpSocket->recv(p, 1);
iva2k 0:e614f7875b60 98 //FIXME: handle errors here (ret < 0)?
iva2k 0:e614f7875b60 99 if(!ret)
iva2k 0:e614f7875b60 100 {
iva2k 0:e614f7875b60 101 break;
iva2k 0:e614f7875b60 102 }
iva2k 0:e614f7875b60 103 #if 1
iva2k 0:e614f7875b60 104 // Consider incoming symbols - line ends etc.
iva2k 0:e614f7875b60 105 if( (len > 1) && *(p-1)=='\r' && *p=='\n' )
iva2k 0:e614f7875b60 106 {
iva2k 0:e614f7875b60 107 p--;
iva2k 0:e614f7875b60 108 len-=2;
iva2k 0:e614f7875b60 109 break;
iva2k 0:e614f7875b60 110 }
iva2k 0:e614f7875b60 111 else if( *p=='\n' )
iva2k 0:e614f7875b60 112 {
iva2k 0:e614f7875b60 113 len--;
iva2k 0:e614f7875b60 114 break;
iva2k 0:e614f7875b60 115 }
iva2k 0:e614f7875b60 116 #endif
iva2k 0:e614f7875b60 117 p++;
iva2k 0:e614f7875b60 118 len++;
iva2k 0:e614f7875b60 119 }
iva2k 0:e614f7875b60 120 *p = 0;
iva2k 0:e614f7875b60 121
iva2k 0:e614f7875b60 122 DBG("\r\nParsing request (%s) ret=%d\r\n", req, ret);
iva2k 0:e614f7875b60 123
iva2k 0:e614f7875b60 124 *request = string(req);
iva2k 0:e614f7875b60 125
iva2k 0:e614f7875b60 126 return (len > 0);
iva2k 0:e614f7875b60 127 }
iva2k 0:e614f7875b60 128
iva2k 0:e614f7875b60 129
iva2k 0:e614f7875b60 130 void StreamRequestDispatcher::onTcpSocketEvent(TCPSocketEvent e)
iva2k 0:e614f7875b60 131 {
iva2k 0:e614f7875b60 132 DBG("\r\nStreamRequestDispatcher Event %d\r\n", e);
iva2k 0:e614f7875b60 133
iva2k 0:e614f7875b60 134 if(m_closed)
iva2k 0:e614f7875b60 135 {
iva2k 0:e614f7875b60 136 DBG("\r\nWARN: Discarded\r\n");
iva2k 0:e614f7875b60 137 return;
iva2k 0:e614f7875b60 138 }
iva2k 0:e614f7875b60 139
iva2k 0:e614f7875b60 140 switch(e)
iva2k 0:e614f7875b60 141 {
iva2k 0:e614f7875b60 142 case TCPSOCKET_READABLE:
iva2k 0:e614f7875b60 143 //Req arrived, dispatch :
iva2k 0:e614f7875b60 144 dispatchRequest();
iva2k 0:e614f7875b60 145 break;
iva2k 0:e614f7875b60 146 case TCPSOCKET_CONTIMEOUT:
iva2k 0:e614f7875b60 147 case TCPSOCKET_CONRST:
iva2k 0:e614f7875b60 148 case TCPSOCKET_CONABRT:
iva2k 0:e614f7875b60 149 case TCPSOCKET_ERROR:
iva2k 0:e614f7875b60 150 case TCPSOCKET_DISCONNECTED:
iva2k 0:e614f7875b60 151 close();
iva2k 0:e614f7875b60 152 break;
iva2k 0:e614f7875b60 153 }
iva2k 0:e614f7875b60 154
iva2k 0:e614f7875b60 155 }
iva2k 0:e614f7875b60 156 //END REQUEST DISPATCHER========================================================
iva2k 0:e614f7875b60 157
iva2k 0:e614f7875b60 158
iva2k 0:e614f7875b60 159 StreamServer::StreamServer()
iva2k 0:e614f7875b60 160 {
iva2k 0:e614f7875b60 161 m_pTcpSocket = new TCPSocket;
iva2k 0:e614f7875b60 162 m_pTcpSocket->setOnEvent(this, &StreamServer::onTcpSocketEvent);
iva2k 0:e614f7875b60 163 }
iva2k 0:e614f7875b60 164
iva2k 0:e614f7875b60 165 StreamServer::~StreamServer()
iva2k 0:e614f7875b60 166 {
iva2k 0:e614f7875b60 167 delete m_pTcpSocket;
iva2k 0:e614f7875b60 168 }
iva2k 0:e614f7875b60 169
iva2k 0:e614f7875b60 170 void StreamServer::bind(int port /*= 123*/)
iva2k 0:e614f7875b60 171 {
iva2k 0:e614f7875b60 172 Host h(IpAddr(127,0,0,1), port, "localhost");
iva2k 0:e614f7875b60 173 m_pTcpSocket->bind(h);
iva2k 0:e614f7875b60 174 m_pTcpSocket->listen(); //Listen
iva2k 0:e614f7875b60 175 }
iva2k 0:e614f7875b60 176
iva2k 0:e614f7875b60 177 #if 0 //Just for clarity
iva2k 0:e614f7875b60 178 template<typename T>
iva2k 0:e614f7875b60 179 void StreamServer::addHandler(const char* path)
iva2k 0:e614f7875b60 180 {
iva2k 0:e614f7875b60 181 // m_lpHandlers[path] = &T::inst;
iva2k 0:e614f7875b60 182 }
iva2k 0:e614f7875b60 183 #endif
iva2k 0:e614f7875b60 184
iva2k 0:e614f7875b60 185 void StreamServer::sendToAll(const char* buf, int len)
iva2k 0:e614f7875b60 186 {
iva2k 0:e614f7875b60 187 int ret;
iva2k 0:e614f7875b60 188 int i = 0;
iva2k 0:e614f7875b60 189 for(tClients::iterator it = m_lpClients.begin(); it != m_lpClients.end(); ++it)
iva2k 0:e614f7875b60 190 {
iva2k 0:e614f7875b60 191 ret = (*it)->writeData(buf, len);
iva2k 0:e614f7875b60 192 if (
iva2k 0:e614f7875b60 193 ret == TCPSOCKET_RST // This is a safety valve. We should have self-removed from m_lpClients upon socket closure and not get here.
iva2k 0:e614f7875b60 194 || ret == TCPSOCKET_MEM // This probably means that network is not delivering packets, and we're backed up.
iva2k 0:e614f7875b60 195 ) {
iva2k 0:e614f7875b60 196 DBG("\r\nStreamServer::sendToAll(%s) - erasing socket %d\r\n", buf, i);
iva2k 0:e614f7875b60 197 delete (*it);
iva2k 0:e614f7875b60 198 //? (*it) = NULL;
iva2k 0:e614f7875b60 199 //DBG("\r\nStreamServer::sendToAll() - erasing socket\r\n");
iva2k 0:e614f7875b60 200 //? it = m_lpClients.erase(it);
iva2k 0:e614f7875b60 201 continue;
iva2k 0:e614f7875b60 202 }
iva2k 0:e614f7875b60 203 i++;
iva2k 0:e614f7875b60 204 }
iva2k 0:e614f7875b60 205 DBG("\r\nStreamServer::sendToAll(%s) - %d sockets\r\n", buf, i);
iva2k 0:e614f7875b60 206 }
iva2k 0:e614f7875b60 207
iva2k 0:e614f7875b60 208 void StreamServer::onTcpSocketEvent(TCPSocketEvent e)
iva2k 0:e614f7875b60 209 {
iva2k 0:e614f7875b60 210
iva2k 0:e614f7875b60 211 DBG("\r\nStreamServer::onTcpSocketEvent : Event %d\r\n", e);
iva2k 0:e614f7875b60 212
iva2k 0:e614f7875b60 213 if(e==TCPSOCKET_ACCEPT)
iva2k 0:e614f7875b60 214 {
iva2k 0:e614f7875b60 215 TCPSocket* pTcpSocket;
iva2k 0:e614f7875b60 216 Host client;
iva2k 0:e614f7875b60 217
iva2k 0:e614f7875b60 218 if( !!m_pTcpSocket->accept(&client, &pTcpSocket) )
iva2k 0:e614f7875b60 219 {
iva2k 0:e614f7875b60 220 DBG("\r\nStreamServer::onTcpSocketEvent : Could not accept connection.\r\n");
iva2k 0:e614f7875b60 221 return; //Error in accept, discard connection
iva2k 0:e614f7875b60 222 }
iva2k 0:e614f7875b60 223
iva2k 0:e614f7875b60 224 StreamRequestDispatcher* pDispatcher = new StreamRequestDispatcher(this, pTcpSocket); //TcpSocket ownership is passed to dispatcher
iva2k 0:e614f7875b60 225 //The dispatcher object will destroy itself when its socket closes, or will be destroyed on Server destruction
iva2k 0:e614f7875b60 226 m_lpClients.push_back(pDispatcher);
iva2k 0:e614f7875b60 227 }
iva2k 0:e614f7875b60 228 }