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:
Mon Sep 01 21:25:27 2014 +0000
Revision:
36:5ca4a8b8d888
Parent:
35:87d3577800dc
Child:
38:962d71d6dd0e
Stabilized version - tested with WiFlyInterface.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
WiredHome 4:178df829d62b 1 /** @file main.cpp contains the main program that a user would write.
WiredHome 4:178df829d62b 2 * see the documentation above "main"
WiredHome 4:178df829d62b 3 */
WiredHome 35:87d3577800dc 4 #include "mbed.h" // ver 83
WiredHome 35:87d3577800dc 5 #include "RawSerial.h"
WiredHome 30:4717cc1f970e 6
WiredHome 4:178df829d62b 7 // My components
WiredHome 29:14c47d31a9dc 8 #include "Utility.h" // a couple of simple helper functions
WiredHome 33:41ac99847df8 9 #include "WiflyInterface.h" // ver 53, derived from mbed official ver 4
WiredHome 32:34dae3cae1b0 10 #include "SW_HTTPServer.h" // ver 31, derived from nweb
WiredHome 10:b0b6da272a7b 11 #include "DynamicPages.h" // my dynamically generated pages
WiredHome 35:87d3577800dc 12
WiredHome 35:87d3577800dc 13 #define MBED_APP_BOARD 1 /* http://mbed.org/components/mbed-Application-Board/ */
WiredHome 35:87d3577800dc 14 #define SMART_BOARD 2 /* http://mbed.org/users/WiredHome/notebook/SmartBoard-baseboard/ */
WiredHome 35:87d3577800dc 15
WiredHome 35:87d3577800dc 16 #define HW_ADAPTER MBED_APP_BOARD /* Which board are we compiling against? SMART_BOARD or MBED_APP_BOARD */
WiredHome 15:1f2b62130ffb 17
WiredHome 10:b0b6da272a7b 18 #define HTTP_SERVER_PORT 80
WiredHome 4:178df829d62b 19
WiredHome 35:87d3577800dc 20 RawSerial pc(USBTX, USBRX);
WiredHome 11:183b3893eb7d 21
WiredHome 11:183b3893eb7d 22
WiredHome 11:183b3893eb7d 23
WiredHome 10:b0b6da272a7b 24 int main()
WiredHome 10:b0b6da272a7b 25 {
WiredHome 35:87d3577800dc 26 pc.baud(460800); // I like a snappy terminal, so crank it up!
WiredHome 14:85c805890454 27 pc.printf("\r\nSmart WiFly - Build " __DATE__ " " __TIME__ "\r\n");
WiredHome 4:178df829d62b 28
WiredHome 35:87d3577800dc 29 // first signals are tx, rx, reset, status pins.
WiredHome 35:87d3577800dc 30 #if HW_ADAPTER == MBED_APP_BOARD
WiredHome 35:87d3577800dc 31 WiflyInterface wifly( p9, p10, p30, p29, "sf1896", "Sir William Hearth", WPA);
WiredHome 35:87d3577800dc 32 #elif HW_ADAPTER == SMART_BOARD
WiredHome 35:87d3577800dc 33 WiflyInterface wifly(p28, p27, p23, p24, "sf1896", "Sir William Hearth", WPA);
WiredHome 35:87d3577800dc 34 #endif
WiredHome 4:178df829d62b 35
WiredHome 15:1f2b62130ffb 36 // Bring the WiFly interface online
WiredHome 11:183b3893eb7d 37 do {
WiredHome 10:b0b6da272a7b 38 wifly.init(); // start it up as a client of my network using DHCP
WiredHome 36:5ca4a8b8d888 39 wifly.baud(460800); // Only 230K w/o HW flow control
WiredHome 35:87d3577800dc 40 if (0 == wifly.connect())
WiredHome 10:b0b6da272a7b 41 break;
WiredHome 10:b0b6da272a7b 42 pc.printf(" Failed to connect, retrying...\r\n");
WiredHome 10:b0b6da272a7b 43 wait(1.0);
WiredHome 4:178df829d62b 44 wifly.reset();
WiredHome 10:b0b6da272a7b 45 } while (1);
WiredHome 11:183b3893eb7d 46
WiredHome 35:87d3577800dc 47 // Let the human know it is ready - if they are watching
WiredHome 35:87d3577800dc 48 pc.printf("Connect to this server at http://%s:%d\r\n", wifly.getIPAddress(), HTTP_SERVER_PORT);
WiredHome 4:178df829d62b 49
WiredHome 12:479ff89c190b 50 // Now let's instantiate the web server - along with a few settings:
WiredHome 12:479ff89c190b 51 // the Wifly object, the port of interest (typically 80),
WiredHome 35:87d3577800dc 52 // file system path to the static pages, if any.
WiredHome 12:479ff89c190b 53 // the maximum parameters per transaction (in the query string),
WiredHome 12:479ff89c190b 54 // the maximum number of dynamic pages that can be registered,
WiredHome 12:479ff89c190b 55 // the serial port back thru USB (for development/logging)
WiredHome 35:87d3577800dc 56 HTTPServer svr(HTTP_SERVER_PORT, "/", 15, 30, 10, &pc);
WiredHome 10:b0b6da272a7b 57
WiredHome 4:178df829d62b 58 // But for even more fun, I'm registering a few dynamic pages
WiredHome 15:1f2b62130ffb 59 // You see the handlers for in DynamicPages.cpp.
WiredHome 4:178df829d62b 60 // Here you can see the path to place on the URL.
WiredHome 4:178df829d62b 61 // ex. http://192.168.1.140/dyn
WiredHome 35:87d3577800dc 62 svr.RegisterHandler("/", SuperSimpleDynamicPage);
WiredHome 35:87d3577800dc 63 svr.RegisterHandler("/dyn", SimpleDynamicPage);
WiredHome 35:87d3577800dc 64 svr.RegisterHandler("/led", SimpleDynamicForm);
WiredHome 13:14c6ce4ced9c 65
WiredHome 35:87d3577800dc 66
WiredHome 10:b0b6da272a7b 67 while (true) {
WiredHome 33:41ac99847df8 68 svr.Poll(); // non-blocking, but also not deterministic
WiredHome 10:b0b6da272a7b 69 }
WiredHome 4:178df829d62b 70 }
WiredHome 24:849cbe0b36ac 71