A sample program demonstrating a small but powerful web server using the Wifly module. This uses several libraries from others, but has a custom version of the WiflyInterface library, with numerous improvement to the mbed standard library.

Dependencies:   SW_HTTPServer WiflyInterface mbed C12832 IniManager

Here's the code

But you also might want to check out the SmartBoard-WiFly project page.

Basic Web Server

  • Serves static files from the selected file system. This is a compile-time setting, and a typical configuration supports gif, jpg, jpeg, ico, png, zip, gz, tar, txt, pdf, htm, and html.
  • It is designed to be small, thereby better supporting the limited resources of an embedded environment.

Advanced Web Services

  • Serves dynamically generated pages, where your software registers for a path, and then everything to that path activates your handler. Your handler then defines the header and body response.
  • Dynamic handlers can process GET query parameters (e.g. /dyn1?sky=blue&grass=green).
  • Dynamic handlers can process POST query parameters, as delivered from submission of a form.
  • Dynamic handlers can protect a resource with user:password access.

Run-Time Configurations

  • File System Support - using either the "local" file system supported by the magic chip, or from either an SD-Card or a USB flash drive.
  • Configurable to the maximum number of dynamic handlers (minimize memory requirements).
  • Configurable to the maximum number of name=value pairs for dynamic handlers (minimize memory requirements).

Compile-Time Configurations

  • Default filename for URL ending in '/' - default is 'index.htm'.
  • Configurable buffer sizes permit minimizing RAM requirements.
  • Configurable header response information.
  • Configurable for which serial port is used to communicate to the WiFly module.
  • Improved security option - to disable telnet access.

Diagnostics

  • API to determine the largest header (to more efficiently size the buffers).
  • API to gather the run-time characteristics - header processing time and content delivery time.

Limitations / Constraints

Known Issues

These are known issues, not yet resolved.

  1. Occasionally fails to serve a page - one test will constantly reload a web page every 30 seconds. It may run for hours, or minutes, then fail to load. Behaviors then are:
    • Hit the reload button in the browser and away it goes.
    • Hit the reload and you'll see the Wifly LEDs energize from the request, but no response by the web server. It appears that the embedded code does not "accept()" the connection in the TCP Socket Server.
      • In this case, the Wifly module has gone through an internal watchdog reset and the configuration parameters are such that it does not gracefully recover. Microchip is aware of this issue, but has not solved it.

Wifly Limitations

  • Single thread - it doesn't respond to overlapping requests (e.g. an embedded image may be requested before the main page completes transfer - the request is lost and the image not shown).
  • Single client - goes along with the single thread, but it doesn't support more than one client at a time.

Smart-Wifly-WebServer

  • Dynamic memory allocation - it does use dynamic memory allocation, which would be discouraged/avoided in many embedded systems. Here it uses it in parsing a request and it releases those resources upon completion of that request. If there is no other dynamic allocation that persists beyond a transaction, it should not cause memory fragmentation. Note that with multi-threading (if this is implemented with an OS), you then have race conditions that could cause fragmentation.

Web Server

Here's the web server in action. A combination of static pages served from the file system and dynamically generated pages.

/media/uploads/WiredHome/swsvr_1.pngPart of the main demo page,
which basically has all the
specifications, configurations, and limitations.
/media/uploads/WiredHome/swsvr_2.pngA zoomed out view of the same page.
/media/uploads/WiredHome/swsvr_3.pngIt would be possible to configure
the server via the web.
/media/uploads/WiredHome/swsvr_4.pngOne of the dynamically generated pages.
This one has parsed the query parameters.
/media/uploads/WiredHome/swsvr_5.pngA simple form which has a dynamic handler on the back end.
Here it takes the value associated with "leds"
and uses that to set the 4 LEDs on the mbed module.
/media/uploads/WiredHome/swsvr_6.pngA dynamic handler can require authentication.
/media/uploads/WiredHome/swsvr_7.pngSuccess!

But I've now gone so far beyond that in the current version. Here's what this one can do:

  1. It serves static web pages from a file system. I've only tested with the local file system and an SD card, but should work for any, so long as you remember that the local file system can't read subdirectories.
  2. It can serve dynamically generated web pages. This lets you accept name=value pairs using the URL (using either a GET or POST method). It can also accept them from forms. The demo lets you control the 4 LEDs from a form.
  3. As safely as possible it retrieves your credentials to the Wi-Fi Access Point. After using them, it overwrites that ram so they can't be as easily extracted.
  4. I made a large number of changes to the Wifly driver. It had too short of a timeout and I found quite a number of optimizations for performance and robustness.
  5. I have the start on a security feature - you can configure a resource to require user credentials to access it. The browser typically provides a username and password dialog. Take care however, as it does not support a secure (https) connection, so the credentials are not as securely transferred as I would like.

Optimizations I'd like to do:

  1. speed it up - I'm running the mbed to wifly module interface at 230K, which is about the top speed w/o flow control. There are other places where some time delays remain - I have eliminated a number of them.
  2. make it non-blocking, so other work can happen.
  3. integrate it with the rtos
  4. When a web page has referenced resources (e.g. an image tag), it rarely loads the image on the first try. I think the request for the resource comes in while it is still in the WiflyInterface cleaning up from the last connection. The Wifly module supports only a single connection at a time. I worked around this with a small bit of javascript to load the images after the web page.

But all in all I think it is a good start.

Program prerequisite

Here's the link to the program, but when you open it up, note a few very important items.

  1. Port Numbers listed in the constructor match the SmartBoard Baseboard.
  2. I sped up the communication baud rate to the mbed from the default 9600. Match your terminal program accordingly.
  3. Download this zip. Place it and an unzipped copy into the mbed local file system. These are the demo files.
  4. The typical ssid and password are not shown. See below to set yours.

ssid and password

You need to create a simple text file on your mbed root folder named "config.ini". The easiest way perhaps is to create "config.txt", add the information shown below and then rename it. This will be read at startup to connect you to the network. Something quite simple, like this:

[Wifi]
ssid=your_ssid
pass=your_pass_code

The program

And the program.

Import programSmart-WiFly-WebServer

A sample program demonstrating a small but powerful web server using the Wifly module. This uses several libraries from others, but has a custom version of the WiflyInterface library, with numerous improvement to the mbed standard library.

Committer:
WiredHome
Date:
Sun Sep 01 19:54:42 2013 +0000
Revision:
15:1f2b62130ffb
Parent:
DynamicPages.cpp@14:85c805890454
Child:
16:ab1867ffcf94
Cleaned up the code, removed some of the debug work.; Hopefully it is easier to follow by others.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
WiredHome 7:72b5df2fac93 1 /// Demonstration of dynamic page creation using the Smartware Web Server
WiredHome 11:183b3893eb7d 2 ///
WiredHome 11:183b3893eb7d 3 /// There are a few samples here of the dynamically generated web pages.
WiredHome 11:183b3893eb7d 4 /// Read the header above each one for more details.
WiredHome 11:183b3893eb7d 5 ///
WiredHome 3:b3e21f3306a1 6 #include "mbed.h"
WiredHome 3:b3e21f3306a1 7
WiredHome 3:b3e21f3306a1 8 #include "SW_HTTPServer.h"
WiredHome 3:b3e21f3306a1 9 #include "DynamicPages.h"
WiredHome 3:b3e21f3306a1 10 #include "Utility.h"
WiredHome 15:1f2b62130ffb 11
WiredHome 3:b3e21f3306a1 12
WiredHome 11:183b3893eb7d 13 BusOut leds(LED4,LED3,LED2,LED1); // dynamic page "/dyn2" lets you control these
WiredHome 3:b3e21f3306a1 14
WiredHome 3:b3e21f3306a1 15
WiredHome 4:178df829d62b 16 /// SimplyDynamicPage1
WiredHome 4:178df829d62b 17 ///
WiredHome 14:85c805890454 18 /// This web page is generated dynamically as a kind of "bare minimum".
WiredHome 11:183b3893eb7d 19 /// It doesn't do much.
WiredHome 11:183b3893eb7d 20 ///
WiredHome 11:183b3893eb7d 21 /// You can see in main how this page was registered.
WiredHome 4:178df829d62b 22 ///
WiredHome 14:85c805890454 23 bool SuperSimpleDynamicPage(HTTPServer *svr, HTTPServer::CallBackType type, const char * path, const HTTPServer::namevalue *params, int paramcount)
WiredHome 14:85c805890454 24 {
WiredHome 4:178df829d62b 25 bool ret = false;
WiredHome 14:85c805890454 26
WiredHome 4:178df829d62b 27 switch (type) {
WiredHome 4:178df829d62b 28 case HTTPServer::SEND_PAGE:
WiredHome 4:178df829d62b 29 svr->header(200, "OK", "Content-Type: text/html\r\n");
WiredHome 4:178df829d62b 30 svr->send("<html><head><title>Dynamic Page</title></head>\r\n");
WiredHome 4:178df829d62b 31 svr->send("<body>\r\n");
WiredHome 11:183b3893eb7d 32 svr->send("<h1>Smart WiFly Web Server</h1>\r\n");
WiredHome 7:72b5df2fac93 33 svr->send("This page was generated dynamically.<br/>\r\n");
WiredHome 7:72b5df2fac93 34 svr->send("<a href='/'>back to main</a></body></html>\r\n");
WiredHome 4:178df829d62b 35 ret = true;
WiredHome 4:178df829d62b 36 break;
WiredHome 4:178df829d62b 37 case HTTPServer::CONTENT_LENGTH_REQUEST:
WiredHome 4:178df829d62b 38 ret = true;
WiredHome 4:178df829d62b 39 break;
WiredHome 4:178df829d62b 40 case HTTPServer::DATA_TRANSFER:
WiredHome 4:178df829d62b 41 ret = true;
WiredHome 4:178df829d62b 42 break;
WiredHome 4:178df829d62b 43 default:
WiredHome 4:178df829d62b 44 ret = false;
WiredHome 4:178df829d62b 45 break;
WiredHome 4:178df829d62b 46 }
WiredHome 4:178df829d62b 47 return ret;
WiredHome 4:178df829d62b 48 }
WiredHome 4:178df829d62b 49
WiredHome 4:178df829d62b 50
WiredHome 3:b3e21f3306a1 51 /// SimplyDynamicPage
WiredHome 3:b3e21f3306a1 52 ///
WiredHome 14:85c805890454 53 /// This web page is generated dynamically and fills in part of the information from
WiredHome 11:183b3893eb7d 54 /// data extracted from the URL as a query string. It also shows some additional
WiredHome 11:183b3893eb7d 55 /// information - memory stats, server stats and so forth.
WiredHome 3:b3e21f3306a1 56 ///
WiredHome 3:b3e21f3306a1 57 /// You can see in main how this page was registered.
WiredHome 3:b3e21f3306a1 58 ///
WiredHome 14:85c805890454 59 bool SimpleDynamicPage(HTTPServer *svr, HTTPServer::CallBackType type, const char * path, const HTTPServer::namevalue *params, int paramcount)
WiredHome 14:85c805890454 60 {
WiredHome 3:b3e21f3306a1 61 char buf[100];
WiredHome 11:183b3893eb7d 62 char remoteIP[36];
WiredHome 3:b3e21f3306a1 63 bool ret = false;
WiredHome 4:178df829d62b 64 HTTPServer::SW_PerformanceData perfData;
WiredHome 14:85c805890454 65
WiredHome 3:b3e21f3306a1 66 switch (type) {
WiredHome 3:b3e21f3306a1 67 case HTTPServer::SEND_PAGE:
WiredHome 4:178df829d62b 68 if (strcmp("true", svr->GetParameter("Reset")) == 0)
WiredHome 4:178df829d62b 69 svr->ResetPerformanceData();
WiredHome 4:178df829d62b 70 // Send the header
WiredHome 3:b3e21f3306a1 71 svr->header(200, "OK", "Content-Type: text/html\r\n");
WiredHome 14:85c805890454 72
WiredHome 4:178df829d62b 73 // Send top of the page
WiredHome 3:b3e21f3306a1 74 svr->send("<html><head><title>Dynamic Page</title></head>\r\n");
WiredHome 3:b3e21f3306a1 75 svr->send("<body>\r\n");
WiredHome 11:183b3893eb7d 76 svr->send("<h1>Smart WiFly Web Server</h1>\r\n");
WiredHome 3:b3e21f3306a1 77 svr->send("This page was generated dynamically. Create your own name=value pairs on the URL "
WiredHome 3:b3e21f3306a1 78 "which uses the GET method.<br/>\r\n");
WiredHome 4:178df829d62b 79
WiredHome 4:178df829d62b 80 // Show Passed-in parameters
WiredHome 4:178df829d62b 81 svr->send("<h2>Query Parameters:</h2><blockquote>\r\n");
WiredHome 3:b3e21f3306a1 82 sprintf(buf, "%d parameters passed to {%s}:<br/>\r\n", paramcount, path);
WiredHome 3:b3e21f3306a1 83 svr->send(buf);
WiredHome 3:b3e21f3306a1 84 // show each of the parameters passed on the URL
WiredHome 3:b3e21f3306a1 85 for (int i=0; i<paramcount; i++) {
WiredHome 3:b3e21f3306a1 86 sprintf(buf, "%d: %s = %s<br/>\r\n", i, params[i].name, params[i].value);
WiredHome 3:b3e21f3306a1 87 svr->send(buf);
WiredHome 3:b3e21f3306a1 88 }
WiredHome 4:178df829d62b 89 svr->send("</blockquote>\r\n");
WiredHome 4:178df829d62b 90
WiredHome 4:178df829d62b 91 // Show Memory Stats
WiredHome 7:72b5df2fac93 92 svr->send("<h2>System Statistics:</h2><blockquote>\r\n");
WiredHome 7:72b5df2fac93 93 svr->send("<table border='1'><tr><td>Parameter</td><td>Description</td></tr>\r\n");
WiredHome 7:72b5df2fac93 94 sprintf(buf,"<tr><td align='right'>%d</td><td>Free memory</td></tr>\r\n", Free());
WiredHome 7:72b5df2fac93 95 svr->send(buf);
WiredHome 7:72b5df2fac93 96 sprintf(buf,"<tr><td align='right'>%d</td><td>Max Header size</td></tr>\r\n", svr->GetMaxHeaderSize());
WiredHome 4:178df829d62b 97 svr->send(buf);
WiredHome 14:85c805890454 98 //TODO sprintf(buf,"<tr><td align='right'>%3.2f</td><td>Wifly SW version</td></tr>\r\n", svr->GetWifly()->getWiflyVersion());
WiredHome 14:85c805890454 99 // svr->send(buf);
WiredHome 14:85c805890454 100 // sprintf(buf,"<tr><td align='right'>%s</td><td>Wifly Version Information</td></tr>\r\n", svr->GetWifly()->getWiflyVersionString());
WiredHome 14:85c805890454 101 // svr->send(buf);
WiredHome 7:72b5df2fac93 102 svr->GetRemoteAddr(remoteIP, sizeof(remoteIP));
WiredHome 7:72b5df2fac93 103 sprintf(buf,"<tr><td align='right'>%s</td><td>Client IP Address</td></tr>\r\n", remoteIP);
WiredHome 7:72b5df2fac93 104 svr->send(buf);
WiredHome 7:72b5df2fac93 105 svr->send("</table>\r\n");
WiredHome 4:178df829d62b 106 svr->send("</blockquote>\r\n");
WiredHome 4:178df829d62b 107
WiredHome 7:72b5df2fac93 108 // Show Web Server Performance metrics
WiredHome 4:178df829d62b 109 svr->GetPerformanceData(&perfData);
WiredHome 7:72b5df2fac93 110 svr->send("<h2>Web Server Performance Metrics</h2><blockquote>\r\n");
WiredHome 4:178df829d62b 111 svr->send("<table border='1'><tr><td>avg time (uS)</td><td>samples</td><td>max time (uS)</td><td>description</td></tr>\r\n");
WiredHome 4:178df829d62b 112 sprintf(buf, "<tr><td align='right'>%d</td><td align='right'>%d</td><td align='right'>%d</td><td>%s</td></tr>\r\n",
WiredHome 14:85c805890454 113 (unsigned long)(perfData.Header.TotalTime_us / perfData.Header.Samples),
WiredHome 14:85c805890454 114 perfData.Header.Samples,
WiredHome 14:85c805890454 115 (unsigned long)(perfData.Header.MaxTime_us),
WiredHome 14:85c805890454 116 "Header Response");
WiredHome 4:178df829d62b 117 svr->send(buf);
WiredHome 4:178df829d62b 118 sprintf(buf, "<tr><td align='right'>%d</td><td align='right'>%d</td><td align='right'>%d</td><td>%s</td></tr>\r\n",
WiredHome 14:85c805890454 119 (unsigned long)(perfData.SendData.TotalTime_us / perfData.SendData.Samples),
WiredHome 14:85c805890454 120 perfData.SendData.Samples,
WiredHome 14:85c805890454 121 (unsigned long)(perfData.SendData.MaxTime_us),
WiredHome 14:85c805890454 122 "SendData Response");
WiredHome 4:178df829d62b 123 svr->send(buf);
WiredHome 7:72b5df2fac93 124 svr->send("<tr><td colspan='3'>&nbsp;</td><td><a href='?Reset=false'>Reload</a> <a href='?Reset=true'>Reset Metrics</a></td></tr>\r\n");
WiredHome 4:178df829d62b 125 svr->send("</table>\r\n");
WiredHome 4:178df829d62b 126 svr->send("</blockquote>\r\n");
WiredHome 4:178df829d62b 127
WiredHome 4:178df829d62b 128 // Send bottom of the page
WiredHome 4:178df829d62b 129 svr->send("<br/><a href='/'>back to main</a></body></html>\r\n");
WiredHome 3:b3e21f3306a1 130 ret = true;
WiredHome 3:b3e21f3306a1 131 break;
WiredHome 3:b3e21f3306a1 132 case HTTPServer::CONTENT_LENGTH_REQUEST:
WiredHome 3:b3e21f3306a1 133 ret = true;
WiredHome 3:b3e21f3306a1 134 break;
WiredHome 3:b3e21f3306a1 135 case HTTPServer::DATA_TRANSFER:
WiredHome 3:b3e21f3306a1 136 ret = true;
WiredHome 3:b3e21f3306a1 137 break;
WiredHome 3:b3e21f3306a1 138 default:
WiredHome 3:b3e21f3306a1 139 ret = false;
WiredHome 3:b3e21f3306a1 140 break;
WiredHome 3:b3e21f3306a1 141 }
WiredHome 3:b3e21f3306a1 142 return ret;
WiredHome 3:b3e21f3306a1 143 }
WiredHome 3:b3e21f3306a1 144
WiredHome 3:b3e21f3306a1 145 /// SimplyDynamicForm
WiredHome 3:b3e21f3306a1 146 ///
WiredHome 14:85c805890454 147 /// This web page is generated dynamically and fills in part of the information from
WiredHome 3:b3e21f3306a1 148 /// data extracted from the URL. It also generates a web form, and allows you to
WiredHome 14:85c805890454 149 /// update the data in the form using the POST method. This one also updates the
WiredHome 11:183b3893eb7d 150 /// leds on the mbed module based on the form submission, so demonstrates a very
WiredHome 11:183b3893eb7d 151 /// simple interaction with the hardware.
WiredHome 3:b3e21f3306a1 152 ///
WiredHome 3:b3e21f3306a1 153 /// You can see in main how this page was registered.
WiredHome 3:b3e21f3306a1 154 ///
WiredHome 14:85c805890454 155 bool SimpleDynamicForm(HTTPServer *svr, HTTPServer::CallBackType type, const char * path, const HTTPServer::namevalue *params, int paramcount)
WiredHome 14:85c805890454 156 {
WiredHome 3:b3e21f3306a1 157 char buf[100];
WiredHome 3:b3e21f3306a1 158 bool ret = false;
WiredHome 14:85c805890454 159
WiredHome 3:b3e21f3306a1 160 switch (type) {
WiredHome 3:b3e21f3306a1 161 case HTTPServer::SEND_PAGE:
WiredHome 3:b3e21f3306a1 162 // set the LEDs based on a passed in parameter.
WiredHome 3:b3e21f3306a1 163 leds = atoi(svr->GetParameter("leds"));
WiredHome 3:b3e21f3306a1 164 // send the header
WiredHome 3:b3e21f3306a1 165 svr->header(200, "OK", svr->GetSupportedType(".htm")); //svr->header(200, "OK", "Content-Type: text/html\r\n");
WiredHome 3:b3e21f3306a1 166 // send some data
WiredHome 3:b3e21f3306a1 167 svr->send("<html><head><title>Dynamic Page</title></head>\r\n");
WiredHome 3:b3e21f3306a1 168 svr->send("<body>\r\n");
WiredHome 11:183b3893eb7d 169 svr->send("<h1>Smart WiFly Web Server</h1>\r\n");
WiredHome 3:b3e21f3306a1 170 svr->send("This form was generated dynamically. You can add name=value pairs on the URL "
WiredHome 3:b3e21f3306a1 171 "which will show up in the form, but the form is submitted using POST method.<br/>\r\n");
WiredHome 3:b3e21f3306a1 172 svr->send("Hint: leds=7 turns on 3 of the 4 blue leds on the mbed<br/>\r\n");
WiredHome 3:b3e21f3306a1 173 sprintf(buf, "%d parameters passed to {%s}:<br/>\r\n", paramcount, path);
WiredHome 3:b3e21f3306a1 174 svr->send(buf);
WiredHome 3:b3e21f3306a1 175 // Create a user form for which they can post changes
WiredHome 3:b3e21f3306a1 176 sprintf(buf, "<form method='post' action='%s'>\r\n", path);
WiredHome 3:b3e21f3306a1 177 //sprintf(buf, "<form method='post' enctype='multipart/form-data' action='%s'>\r\n", path); // not supported
WiredHome 3:b3e21f3306a1 178 svr->send(buf);
WiredHome 3:b3e21f3306a1 179 // show the parameters in a nice format
WiredHome 3:b3e21f3306a1 180 svr->send("<table>\r\n");
WiredHome 3:b3e21f3306a1 181 for (int i=0; i<paramcount; i++) {
WiredHome 3:b3e21f3306a1 182 if (strcmp(params[i].name, "InFile") == 0)
WiredHome 3:b3e21f3306a1 183 continue;
WiredHome 3:b3e21f3306a1 184 sprintf(buf, "<tr><td>%d</td><td>%s</td><td><input type='text' name='%s' value='%s'></td></tr>\r\n", i, params[i].name, params[i].name, params[i].value);
WiredHome 3:b3e21f3306a1 185 svr->send(buf);
WiredHome 3:b3e21f3306a1 186 }
WiredHome 3:b3e21f3306a1 187 //svr->send("<tr><td>&nbsp;</td><td>File</td><td><input type='file' name='InFile' size='40'></td></tr>\r\n");
WiredHome 3:b3e21f3306a1 188 svr->send("<tr><td>&nbsp;</td><td colspan='2'><input type='submit' value='submit'><input type='reset' value='clear'></td></tr>\r\n");
WiredHome 3:b3e21f3306a1 189 svr->send("</table>\r\n");
WiredHome 3:b3e21f3306a1 190 svr->send("</form>\r\n");
WiredHome 3:b3e21f3306a1 191 // see how we're doing with free memory
WiredHome 4:178df829d62b 192 svr->send("<br/><a href='/'>Back to main</a></body></html>\r\n");
WiredHome 3:b3e21f3306a1 193 ret = true;
WiredHome 3:b3e21f3306a1 194 break;
WiredHome 3:b3e21f3306a1 195 case HTTPServer::CONTENT_LENGTH_REQUEST:
WiredHome 3:b3e21f3306a1 196 ret = true;
WiredHome 3:b3e21f3306a1 197 break;
WiredHome 3:b3e21f3306a1 198 case HTTPServer::DATA_TRANSFER:
WiredHome 3:b3e21f3306a1 199 ret = true;
WiredHome 3:b3e21f3306a1 200 break;
WiredHome 3:b3e21f3306a1 201 default:
WiredHome 3:b3e21f3306a1 202 ret = false;
WiredHome 3:b3e21f3306a1 203 break;
WiredHome 3:b3e21f3306a1 204 }
WiredHome 3:b3e21f3306a1 205 return ret;
WiredHome 3:b3e21f3306a1 206 }
WiredHome 7:72b5df2fac93 207