Introduction

These sketches were written during the development of my ESP32 Robot. As part of the troubleshooting process, I developed pairs of generic Server/Client sketches for HTTP communication between the ESP32 and the ESP8266, which I then used as templates for the final robot and remote. Below I have included two pairs of sketches. In the first pair, the ESP8266 acts as a Wireless Access Point and HTTP Server, and the ESP32 acts as a Wireless Station that connects and makes a GET request. In the second pair, the roles are reversed.

Each pair of sketches should work out of the box to connect to each other. They could be used as-is to test a newly purchased/produced ESP8266 or ESP32 hardware piece. Depending on the hardware you have on hand, it should be possible to mix the pairs, as long as you change the ssid and password variables to match each other. They can also be used as templates to start a home automation network. An ESP8266 in AP mode can handle five connected stations simultaneously, even if it can only handle one HTTP connection at a time. I hope the ESP32's hardware supports more stations and more simultaneous connections!

The ESP32 sketches are tested on a HiLetgo ESP32,ESP32 and the ESP8266 files are tested on a HiLetgo NodeMCU and an Adafruit Feather HUZZAH. NodeMCU Feather HUZZAH The AP files use the on-board blue LED to indicate that a station has connected to the access point, and the STA files use the LED to indicate that they have received a positive response from the server.


Customization

Client Customization

By default, the client connects to the AP, requests the root of the server, and prints the received response to the Serial port in the setup function, then has an empty loop function. You can make it more manageable by moving the request code into its own function as shown below. The first argument cannot be a constant, so you will also need to change the definition of the host variable at the top of the sketch. The second argument should be the path to the resource on the server, like "/led/on" or "/servo/45". It's basically the command to be sent to the other microcontroller. You can set your server to search for these strings in the processRequest() function described in the server customization section.

boolean makeRequest(char* host, String path) {
  //  Client
  if (WiFi.status() == WL_CONNECTED) {
    //WiFi.printDiag(Serial);
    WiFiClient client;
    Serial.printf("\n[Connecting to %s ... ", host);
    if (client.connect(host, 80)) {
      Serial.println("connected]");
      Serial.println("[Sending a request for " + path + "]");
      client.print(String("GET ") + path + " HTTP/1.1\r\n" +
                   "Host: " + host + "\r\n" +
                   "Connection: close\r\n" +
                   "\r\n"
                  );
      Serial.println("[Response:]");
      while (client.connected()) {
        if (client.available()) {
          String line = client.readStringUntil('\n');
          Serial.println(line);
        }
      }
      client.stop();
      Serial.println("\n[Disconnected]");
      return true;
    } else {
      Serial.println("connection failed!]");
      client.stop();
      delay(5000);
    }
    return false;
  }
}

Server Customization

By default, the server prints all received data to the Serial port and responds with a "Hello World" message. You can customize how the server reacts to the first line of the GET request by adding these lines

    String req = client.readStringUntil('\r');
    Serial.println(req);
    String response = processRequest(req);

to the server's loop function, just before this line

    while (client.connected()) {

Then implement this function

String processRequest(String req);

that returns a String to be sent to the client as the response. You could return the HTML for an entire webpage. You can use req.indexOf() to search for the path arguments that you specified in the Client Customization section, and react to them accordingly if they're found (if the index is greater than -1). Look for an example in my ESP32 Robot code. Mine controlled servos.


First Pair: ESP8266 as AP, ESP32 as STA

ap_server_8266.ino

/*
 * ap_server_8266.ino
 * Matt Gorr
 * 7/9/2017
 * 
 */
#include <ESP8266WiFi.h>
char * ssid = "NodeMCU";
char * password = "Node8266";
int red = LED_BUILTIN;
int blue = D4;
int button = 0;
WiFiServer server(80);
void setup() {
  Serial.begin(115200);
  //  LED's
  pinMode(blue, OUTPUT);
  pinMode(red, OUTPUT);
  digitalWrite(blue, HIGH);
  digitalWrite(red, HIGH);
  // Wireless
  Serial.print("Creating Access Point");
  WiFi.mode(WIFI_AP);
  while (!WiFi.softAP(ssid, password)) {
    Serial.print(".");
  }
  Serial.println("");
  Serial.println("Network Created!");
  Serial.print("Soft-AP IP address = ");
  Serial.println(WiFi.softAPIP());
  // Server
  Serial.print("Creating HTTP Server");
  server.begin();
}
void loop() {
  digitalWrite(blue, !WiFi.softAPgetStationNum());
  //Serial.println(analogRead(A0));
  //delay(100);
  WiFiClient client = server.available();
  if (client) {
    // we have a new client sending some request
    while (client.connected()) {
      if (client.available()) {
        String line = client.readStringUntil('\r');
        Serial.print(line);
        if (line.length() == 1 && line[0] == '\n') {
          client.println(prepareHtmlPage("Hello World!"));
          break;
        }
      }
    }
    delay(1); // give the web browser time to receive the data
    // close the connection:
    client.stop();
    Serial.println("[Client disonnected]");
  }
}
String prepareHtmlPage(String output) {
  String htmlPage =
    String("HTTP/1.1 200 OK\r\n") +
    "Content-Type: text/html\r\n" +
    "Connection: close\r\n" +  // the connection will be closed after completion of the response
    "\r\n" +
    "<!DOCTYPE HTML>" +
    "<html>" +
    output +
    "</html>" +
    "\r\n";
  return htmlPage;
} 
String prepareHtmlPage() {
  String htmlPage =
    String("HTTP/1.1 200 OK\r\n") +
    "Content-Type: text/html\r\n" +
    "Connection: close\r\n" +  // the connection will be closed after completion of the response
    "Refresh: 5\r\n" +  // refresh the page automatically every 5 sec
    "\r\n" +
    "<!DOCTYPE HTML>" +
    "<html>" +
    "Analog input:  " + String(analogRead(A0)) +
    "</html>" +
    "\r\n";
  return htmlPage;
}

sta_client_32.ino

/*
 * sta_client_32.ino
 * Matt Gorr
 * 7/9/2017
 * 
 */
#include<WiFi.h>
char * ssid = "NodeMCU";
char * password = "Node8266";
const char* host = "192.168.4.1";
int blue = LED_BUILTIN;
int button = 0;
void setup() {
  Serial.begin(115200);
  //  LED's
  pinMode(blue, OUTPUT);
  digitalWrite(blue, LOW);
  //  Wireless
  WiFi.mode(WIFI_OFF);  // Turn off radio completely
  delay(1000);          
  WiFi.mode(WIFI_STA);  // Turn on radio in STAtion mode: don't broadcast my own SSID
  WiFi.begin(ssid, password);
  while (WiFi.status() != WL_CONNECTED) {
    delay(500);
    Serial.print(".");
  }
  Serial.println();
  Serial.print("Connected, IP address: ");
  Serial.println(WiFi.localIP());
  //  Client
  while (WiFi.status() == WL_CONNECTED) {
    WiFiClient client;
    Serial.printf("\n[Connecting to %s ... ", host);
    if (client.connect(host, 80)) {
      Serial.println("connected]");
      Serial.println("[Sending a request]");
      client.print(String("GET /") + " HTTP/1.1\r\n" +
                   "Host: " + host + "\r\n" +
                   "Connection: close\r\n" +
                   "\r\n"
                  );
      Serial.println("[Response:]");
      while (client.connected()) {
        if (client.available()) {
          String line = client.readStringUntil('\n');
          Serial.println(line);
        }
      }
      client.stop();
      Serial.println("\n[Disconnected]");
      break;
    } else {
      Serial.println("connection failed!]");
      client.stop();
      delay(5000);
    }
  }
}
void loop() {
  digitalWrite(blue, HIGH);
}

Second Pair: ESP32 as AP, ESP8266 as STA

ap_server_32.ino

/*
 * ap_server_32.ino
 * Matt Gorr
 * 7/9/2017
 * 
 */
#include<WiFi.h>
char * ssid = "Node32s";
char * password = "Nodemcu32";
const char* host = "192.168.4.1";
int blue = LED_BUILTIN;
int button = 0;
WiFiServer server(80);
void setup() {
  Serial.begin(115200);
  //Serial.setDebugOutput(true);
  //  LED's
  pinMode(blue, OUTPUT);
  digitalWrite(blue, LOW);
  // Wireless
  Serial.print("Creating Access Point");
  WiFi.mode(WIFI_MODE_AP);
  while (!WiFi.softAP(ssid, password)) {
    Serial.print(".");
  }
  Serial.println("");
  Serial.println("Network Created!");
  Serial.print("Soft-AP IP address = ");
  Serial.println(WiFi.softAPIP());
  // Server
  Serial.print("Creating HTTP Server");
  server.begin();
  //WiFi.printDiag(Serial);
}
void loop() {
  //Serial.println(analogRead(A0));
  //delay(100);
  WiFiClient client = server.available();
  digitalWrite(blue, client);
  if (client) {
    // we have a new client sending some request
    while (client.connected()) {
      if (client.available()) {
        String line = client.readStringUntil('\r');
        Serial.print(line);
        if (line.length() == 1 && line[0] == '\n') {
          client.println(prepareHtmlPage("Hello World!"));
          break;
        }
      }
    }
    delay(1); // give the web browser time to receive the data
    // close the connection:
    client.stop();
    Serial.println("[Client disonnected]");
  }
}
String prepareHtmlPage(String output) {
  String htmlPage =
    String("HTTP/1.1 200 OK\r\n") +
    "Content-Type: text/html\r\n" +
    "Connection: close\r\n" +  // the connection will be closed after completion of the response
    //"Refresh: 5\r\n" +  // refresh the page automatically every 5 sec
    "\r\n" +
    "<!DOCTYPE HTML>" +
    "<html>" +
    output +
    "</html>" +
    "\r\n";
  return htmlPage;
}

sta_client_8266.ino

/*
 * sta_client_8266.ino
 * Matt Gorr
 * 7/9/2017
 * 
 */
#include <ESP8266WiFi.h>
char * ssid = "Node32s";
char * password = "Nodemcu32";
const char* host = "192.168.4.1";
int red = LED_BUILTIN;
int blue = 2;
int button = 0;
WiFiServer server(80);
void setup() {
  Serial.begin(115200);
  //Serial.setDebugOutput(true);
  //  LED's
  pinMode(blue, OUTPUT);
  pinMode(red, OUTPUT);
  digitalWrite(blue, HIGH);
  digitalWrite(red, HIGH);
  //  Wireless
  WiFi.mode(WIFI_OFF);  // Turn off radio completely
  delay(1000);          // Wait for '32 to decide '8266 is not connected anymore
  WiFi.mode(WIFI_STA);  // Turn on radio in STAtion mode: don't broadcast my own SSID
  WiFi.begin(ssid, password);
  while (WiFi.status() != WL_CONNECTED) {
    delay(500);
    Serial.print(".");
  }
  Serial.println();
  Serial.print("Connected, IP address: ");
  Serial.println(WiFi.localIP());
  //  Client
  while (WiFi.status() == WL_CONNECTED) {
    //WiFi.printDiag(Serial);
    WiFiClient client;
    Serial.printf("\n[Connecting to %s ... ", host);
    if (client.connect(host, 80)) {
      Serial.println("connected]");
      Serial.println("[Sending a request]");
      client.print(String("GET /") + " HTTP/1.1\r\n" +
                   "Host: " + host + "\r\n" +
                   "Connection: close\r\n" +
                   "\r\n"
                  );
      Serial.println("[Response:]");
      while (client.connected()) {
        if (client.available()) {
          String line = client.readStringUntil('\n');
          Serial.println(line);
        }
      }
      client.stop();
      Serial.println("\n[Disconnected]");
      break;
    } else {
      Serial.println("connection failed!]");
      client.stop();
      delay(5000);
    }
  }
}
void loop() {
  digitalWrite(blue, LOW);
}