diff --git a/01OS/01OS/clients/esp32/README.md b/01OS/01OS/clients/esp32/README.md index 1f3f01b..85e7fb3 100644 --- a/01OS/01OS/clients/esp32/README.md +++ b/01OS/01OS/clients/esp32/README.md @@ -2,11 +2,10 @@ To set up audio recording + playback on the ESP32 (M5 Atom), do the following: -1. Open Arduino IDE, and open the `playback/playback.ino` file +1. Open Arduino IDE, and open the `client/client.ino` file 2. Go to Tools -> Board -> Boards Manager, search "esp32", then install the boards by Arduino and Espressif 3. Go to Tools -> Manage Libraries, then install the following: - M5Atom by M5Stack - WebSockets by Markus Sattler - - HttpClient by Adrian McEwan -4. The board needs to connect to WiFi. Go to the playback.ino code, then add your IP to COMPUTER_IP (line 15) and add your WiFi details (line 180). To find IP on macOS run `ipconfig getifaddr en0` in a terminal window. +4. The board needs to connect to WiFi. Once you flash, connect to ESP32 wifi "captive" which will get wifi details. Once it connects, it will ask you to enter 01OS server address in the format "domain.com:port" or "ip:port". Once its able to connect you can use the device. 5. To flash the .ino to the board, connect the board to the USB port, select the port from the dropdown on the IDE, then select the M5Atom board (or M5Stack-ATOM if you have that). Click on upload to flash the board. \ No newline at end of file diff --git a/01OS/01OS/clients/esp32/playback/playback.ino b/01OS/01OS/clients/esp32/playback/playback.ino deleted file mode 100644 index 883a1d7..0000000 --- a/01OS/01OS/clients/esp32/playback/playback.ino +++ /dev/null @@ -1,243 +0,0 @@ -/*Press button to record,released button to playback*/ - -#include -#include - -#include - -#include -#include -#include - -#include - -//ipconfig getifaddr en0 -#define COMPUTER_IP "192.168.68.63" - -#define CONFIG_I2S_BCK_PIN 19 -#define CONFIG_I2S_LRCK_PIN 33 -#define CONFIG_I2S_DATA_PIN 22 -#define CONFIG_I2S_DATA_IN_PIN 23 - -#define SPEAKER_I2S_NUMBER I2S_NUM_0 - -#define MODE_MIC 0 -#define MODE_SPK 1 -#define DATA_SIZE 1024 - -uint8_t microphonedata0[1024 * 10]; -uint8_t speakerdata0[1024 * 1]; -int speaker_offset = 0; -int data_offset = 0; - -WebSocketsClient webSocket; - -class ButtonChecker { - public: - void loop() { - lastTickState = thisTickState; - thisTickState = M5.Btn.isPressed() != 0; - } - - bool justPressed() { - return thisTickState && !lastTickState; - } - - bool justReleased() { - return !thisTickState && lastTickState; - } - - private: - bool lastTickState = false; - bool thisTickState = false; -}; - -ButtonChecker button = ButtonChecker(); - - - -void hexdump(const void *mem, uint32_t len, uint8_t cols = 16) { - const uint8_t* src = (const uint8_t*) mem; - Serial.printf("\n[HEXDUMP] Address: 0x%08X len: 0x%X (%d)", (ptrdiff_t)src, len, len); - for (uint32_t i = 0; i < len; i++) { - if (i % cols == 0) { - Serial.printf("\n[0x%08X] 0x%08X: ", (ptrdiff_t)src, i); - } - Serial.printf("%02X ", *src); - src++; - } - Serial.printf("\n"); -} - -void InitI2SSpeakerOrMic(int mode) { - Serial.printf("InitI2sSpeakerOrMic %d\n", mode); - esp_err_t err = ESP_OK; - - i2s_driver_uninstall(SPEAKER_I2S_NUMBER); - i2s_config_t i2s_config = { - .mode = (i2s_mode_t)(I2S_MODE_MASTER), - .sample_rate = 16000, - .bits_per_sample = - I2S_BITS_PER_SAMPLE_16BIT, // is fixed at 12bit, stereo, MSB - .channel_format = I2S_CHANNEL_FMT_ALL_RIGHT, -#if ESP_IDF_VERSION > ESP_IDF_VERSION_VAL(4, 1, 0) - .communication_format = - I2S_COMM_FORMAT_STAND_I2S, // Set the format of the communication. -#else // 设置通讯格式 - .communication_format = I2S_COMM_FORMAT_I2S, -#endif - .intr_alloc_flags = ESP_INTR_FLAG_LEVEL1, - .dma_buf_count = 6, - .dma_buf_len = 60, - }; - if (mode == MODE_MIC) { - i2s_config.mode = - (i2s_mode_t)(I2S_MODE_MASTER | I2S_MODE_RX | I2S_MODE_PDM); - } else { - i2s_config.mode = (i2s_mode_t)(I2S_MODE_MASTER | I2S_MODE_TX); - i2s_config.use_apll = false; - i2s_config.tx_desc_auto_clear = true; - } - - err += i2s_driver_install(SPEAKER_I2S_NUMBER, &i2s_config, 0, NULL); - i2s_pin_config_t tx_pin_config; - -#if (ESP_IDF_VERSION > ESP_IDF_VERSION_VAL(4, 3, 0)) - tx_pin_config.mck_io_num = I2S_PIN_NO_CHANGE; -#endif - tx_pin_config.bck_io_num = CONFIG_I2S_BCK_PIN; - tx_pin_config.ws_io_num = CONFIG_I2S_LRCK_PIN; - tx_pin_config.data_out_num = CONFIG_I2S_DATA_PIN; - tx_pin_config.data_in_num = CONFIG_I2S_DATA_IN_PIN; - - // Serial.println("Init i2s_set_pin"); - err += i2s_set_pin(SPEAKER_I2S_NUMBER, &tx_pin_config); - // Serial.println("Init i2s_set_clk"); - err += i2s_set_clk(SPEAKER_I2S_NUMBER, 16000, I2S_BITS_PER_SAMPLE_16BIT, - I2S_CHANNEL_MONO); -} - -void speaker_play(uint8_t *payload, uint32_t len){ - Serial.printf("received %lu bytes", len); - size_t bytes_written; - InitI2SSpeakerOrMic(MODE_SPK); - i2s_write(SPEAKER_I2S_NUMBER, payload, len, - &bytes_written, portMAX_DELAY); -} - -void webSocketEvent(WStype_t type, uint8_t * payload, size_t length) { - switch (type) { - case WStype_DISCONNECTED: - Serial.printf("[WSc] Disconnected!\n"); - break; - case WStype_CONNECTED: - Serial.printf("[WSc] Connected to url: %s\n", payload); - - // send message to server when Connected - break; - case WStype_TEXT: - Serial.printf("[WSc] get text: %s\n", payload); - { - std::string str(payload, payload + length); - bool isAudio = str.find("\"audio\"") != std::string::npos; - if (isAudio && str.find("\"start\"") != std::string::npos) { - Serial.println("start playback"); - speaker_offset = 0; - InitI2SSpeakerOrMic(MODE_SPK); - } else if (isAudio && str.find("\"end\"") != std::string::npos) { - Serial.println("end playback"); - // speaker_play(speakerdata0, speaker_offset); - // speaker_offset = 0; - } - } - // send message to server - // webSocket.sendTXT("message here"); - break; - case WStype_BIN: - Serial.printf("[WSc] get binary length: %u\n", length); - memcpy(speakerdata0 + speaker_offset, payload, length); - speaker_offset += length; - size_t bytes_written; - i2s_write(SPEAKER_I2S_NUMBER, speakerdata0, speaker_offset, &bytes_written, portMAX_DELAY); - speaker_offset = 0; - - - // send data to server - // webSocket.sendBIN(payload, length); - break; - case WStype_ERROR: - case WStype_FRAGMENT_TEXT_START: - case WStype_FRAGMENT_BIN_START: - case WStype_FRAGMENT: - case WStype_FRAGMENT_FIN: - break; - } - -} - -void websocket_setup() { - Serial.begin(115200); - WiFi.begin("Soundview_Guest", ""); - while (WiFi.status() != WL_CONNECTED){ - delay(500); - Serial.println("connecting to WiFi"); - } - Serial.println("connected to WiFi"); - webSocket.begin(COMPUTER_IP, 8000, "/"); - webSocket.onEvent(webSocketEvent); - // webSocket.setAuthorization("user", "Password"); - webSocket.setReconnectInterval(5000); -} - -void setup() { - M5.begin(true, false, true); - M5.dis.drawpix(0, CRGB(128, 128, 0)); - websocket_setup(); - InitI2SSpeakerOrMic(MODE_SPK); - - delay(2000); -} - -bool recording = false; - -void flush_microphone() { - Serial.printf("[microphone] flushing %d bytes of data\n", data_offset); - if (data_offset == 0) return; - webSocket.sendBIN(microphonedata0, data_offset); - data_offset = 0; -} - -void loop() { - button.loop(); - if (button.justPressed()) { - Serial.println("Recording..."); - webSocket.sendTXT("{\"role\": \"user\", \"type\": \"audio\", \"format\": \"bytes.raw\", \"start\": true}"); - InitI2SSpeakerOrMic(MODE_MIC); - recording = true; - data_offset = 0; - Serial.println("Recording ready."); - } else if (button.justReleased()) { - Serial.println("Stopped recording."); - webSocket.sendTXT("{\"role\": \"user\", \"type\": \"audio\", \"format\": \"bytes.raw\", \"end\": true}"); - flush_microphone(); - recording = false; - data_offset = 0; - } else if (recording) { - Serial.printf("Reading chunk at %d...\n", data_offset); - size_t bytes_read; - i2s_read( - SPEAKER_I2S_NUMBER, - (char *)(microphonedata0 + data_offset), - DATA_SIZE, &bytes_read, (100 / portTICK_RATE_MS) - ); - data_offset += bytes_read; - Serial.printf("Read %d bytes in chunk.\n", bytes_read); - - if (data_offset > 1024*9) { - flush_microphone(); - } - } - - M5.update(); - webSocket.loop(); -} diff --git a/01OS/01OS/clients/esp32/src/client/client.ino b/01OS/01OS/clients/esp32/src/client/client.ino new file mode 100644 index 0000000..3111011 --- /dev/null +++ b/01OS/01OS/clients/esp32/src/client/client.ino @@ -0,0 +1,651 @@ +#include +#include +#include //not needed in the arduino ide +#include //https://github.com/me-no-dev/AsyncTCP using the latest dev version from @me-no-dev +#include +#include //https://github.com/me-no-dev/ESPAsyncWebServer using the latest dev version from @me-no-dev +#include //Used for mpdu_rx_disable android workaround +#include +#include +#include +#include +#include +#include + +String server_domain = ""; +int server_port = 8000; + +// ----------------------- START OF WIFI CAPTIVE PORTAL ------------------- + +// Pre reading on the fundamentals of captive portals https://textslashplain.com/2022/06/24/captive-portals/ + +const char *ssid = "captive"; // FYI The SSID can't have a space in it. +// const char * password = "12345678"; //Atleast 8 chars +const char *password = NULL; // no password + +#define MAX_CLIENTS 4 // ESP32 supports up to 10 but I have not tested it yet +#define WIFI_CHANNEL 6 // 2.4ghz channel 6 https://en.wikipedia.org/wiki/List_of_WLAN_channels#2.4_GHz_(802.11b/g/n/ax) + +const IPAddress localIP(4, 3, 2, 1); // the IP address the web server, Samsung requires the IP to be in public space +const IPAddress gatewayIP(4, 3, 2, 1); // IP address of the network should be the same as the local IP for captive portals +const IPAddress subnetMask(255, 255, 255, 0); // no need to change: https://avinetworks.com/glossary/subnet-mask/ +const String localIPURL = "http://4.3.2.1"; // a string version of the local IP with http, used for redirecting clients to your webpage + +// Number of milliseconds to wait without receiving any data before we give up +const int kNetworkTimeout = 30 * 1000; +// Number of milliseconds to wait if no data is available before trying again +const int kNetworkDelay = 1000; + +String generateHTMLWithSSIDs() +{ + String html = "

Select Wi-Fi Network



"; + + return html; +} + +const char index_html[] PROGMEM = R"=====( + + + + WiFi Setup + + + + +

WiFi Setup

+
+
+
+
+

+ +
+ + +)====="; + +const char post_connected_html[] PROGMEM = R"=====( + + + + 01OS Setup + + + + +

01OS Setup

+
+
+
+ +
+ + +)====="; + +DNSServer dnsServer; +AsyncWebServer server(80); + +void setUpDNSServer(DNSServer &dnsServer, const IPAddress &localIP) +{ +// Define the DNS interval in milliseconds between processing DNS requests +#define DNS_INTERVAL 30 + + // Set the TTL for DNS response and start the DNS server + dnsServer.setTTL(3600); + dnsServer.start(53, "*", localIP); +} + +void startSoftAccessPoint(const char *ssid, const char *password, const IPAddress &localIP, const IPAddress &gatewayIP) +{ +// Define the maximum number of clients that can connect to the server +#define MAX_CLIENTS 4 +// Define the WiFi channel to be used (channel 6 in this case) +#define WIFI_CHANNEL 6 + + // Set the WiFi mode to access point and station + // WiFi.mode(WIFI_MODE_AP); + + // Define the subnet mask for the WiFi network + const IPAddress subnetMask(255, 255, 255, 0); + + // Configure the soft access point with a specific IP and subnet mask + WiFi.softAPConfig(localIP, gatewayIP, subnetMask); + + // Start the soft access point with the given ssid, password, channel, max number of clients + WiFi.softAP(ssid, password, WIFI_CHANNEL, 0, MAX_CLIENTS); + + // Disable AMPDU RX on the ESP32 WiFi to fix a bug on Android + esp_wifi_stop(); + esp_wifi_deinit(); + wifi_init_config_t my_config = WIFI_INIT_CONFIG_DEFAULT(); + my_config.ampdu_rx_enable = false; + esp_wifi_init(&my_config); + esp_wifi_start(); + vTaskDelay(100 / portTICK_PERIOD_MS); // Add a small delay +} + +void connectToWifi(String ssid, String password) +{ + WiFi.begin(ssid.c_str(), password.c_str()); + + // Wait for connection to establish + int attempts = 0; + while (WiFi.status() != WL_CONNECTED && attempts < 20) + { + delay(1000); + Serial.print("."); + attempts++; + } + + if (WiFi.status() == WL_CONNECTED) + { + Serial.println("Connected to Wi-Fi"); + } + else + { + Serial.println("Failed to connect to Wi-Fi. Check credentials."); + } +} + +void connectTo01OS(String server_address) +{ + int err = 0; + + String domain = server_address.substring(0, server_address.indexOf(':')); + String portStr = server_address.substring(server_address.indexOf(':') + 1); + int port = portStr.toInt(); + + WiFiClient c; + HttpClient http(c, domain.c_str(), port); + + Serial.println("Connecting to 01OS at " + domain + ":" + port + "/ping"); + err = http.get("/ping"); + // err = http.get("arduino.cc", "/"); + + if (err == 0) + { + Serial.println("Started the ping request"); + + err = http.responseStatusCode(); + if (err >= 0) + { + Serial.print("Got status code: "); + Serial.println(err); + + if (err == 200) + { + server_domain = domain; + server_port = port; + } + + err = http.skipResponseHeaders(); + if (err >= 0) + { + int bodyLen = http.contentLength(); + Serial.print("Content length is: "); + Serial.println(bodyLen); + Serial.println(); + Serial.println("Body returned follows:"); + + // Now we've got to the body, so we can print it out + unsigned long timeoutStart = millis(); + char c; + // Whilst we haven't timed out & haven't reached the end of the body + while ((http.connected() || http.available()) && + ((millis() - timeoutStart) < kNetworkTimeout)) + { + if (http.available()) + { + c = http.read(); + // Print out this character + Serial.print(c); + + bodyLen--; + // We read something, reset the timeout counter + timeoutStart = millis(); + } + else + { + // We haven't got any data, so let's pause to allow some to + // arrive + delay(kNetworkDelay); + } + } + } + else + { + Serial.print("Failed to skip response headers: "); + Serial.println(err); + } + } + else + { + Serial.print("Getting response failed: "); + Serial.println(err); + } + } + else + { + Serial.print("Connect failed: "); + Serial.println(err); + } +} + +void setUpWebserver(AsyncWebServer &server, const IPAddress &localIP) +{ + //======================== Webserver ======================== + // WARNING IOS (and maybe macos) WILL NOT POP UP IF IT CONTAINS THE WORD "Success" https://www.esp8266.com/viewtopic.php?f=34&t=4398 + // SAFARI (IOS) IS STUPID, G-ZIPPED FILES CAN'T END IN .GZ https://github.com/homieiot/homie-esp8266/issues/476 this is fixed by the webserver serve static function. + // SAFARI (IOS) there is a 128KB limit to the size of the HTML. The HTML can reference external resources/images that bring the total over 128KB + // SAFARI (IOS) popup browser has some severe limitations (javascript disabled, cookies disabled) + + // Required + server.on("/connecttest.txt", [](AsyncWebServerRequest *request) + { request->redirect("http://logout.net"); }); // windows 11 captive portal workaround + server.on("/wpad.dat", [](AsyncWebServerRequest *request) + { request->send(404); }); // Honestly don't understand what this is but a 404 stops win 10 keep calling this repeatedly and panicking the esp32 :) + + // Background responses: Probably not all are Required, but some are. Others might speed things up? + // A Tier (commonly used by modern systems) + server.on("/generate_204", [](AsyncWebServerRequest *request) + { request->redirect(localIPURL); }); // android captive portal redirect + server.on("/redirect", [](AsyncWebServerRequest *request) + { request->redirect(localIPURL); }); // microsoft redirect + server.on("/hotspot-detect.html", [](AsyncWebServerRequest *request) + { request->redirect(localIPURL); }); // apple call home + server.on("/canonical.html", [](AsyncWebServerRequest *request) + { request->redirect(localIPURL); }); // firefox captive portal call home + server.on("/success.txt", [](AsyncWebServerRequest *request) + { request->send(200); }); // firefox captive portal call home + server.on("/ncsi.txt", [](AsyncWebServerRequest *request) + { request->redirect(localIPURL); }); // windows call home + + // B Tier (uncommon) + // server.on("/chrome-variations/seed",[](AsyncWebServerRequest *request){request->send(200);}); //chrome captive portal call home + // server.on("/service/update2/json",[](AsyncWebServerRequest *request){request->send(200);}); //firefox? + // server.on("/chat",[](AsyncWebServerRequest *request){request->send(404);}); //No stop asking Whatsapp, there is no internet connection + // server.on("/startpage",[](AsyncWebServerRequest *request){request->redirect(localIPURL);}); + + // return 404 to webpage icon + server.on("/favicon.ico", [](AsyncWebServerRequest *request) + { request->send(404); }); // webpage icon + + // Serve Basic HTML Page + server.on("/", HTTP_ANY, [](AsyncWebServerRequest *request) + { + String htmlContent = index_html; + Serial.printf("wifi scan complete: %d . WIFI_SCAN_RUNNING: %d", WiFi.scanComplete(), WIFI_SCAN_RUNNING); + if(WiFi.scanComplete() > 0) { + // Scan complete, process results + Serial.println("done scanning wifi"); + htmlContent = generateHTMLWithSSIDs(); + // WiFi.scanNetworks(true); // Start a new scan in async mode + } + AsyncWebServerResponse *response = request->beginResponse(200, "text/html", htmlContent); + response->addHeader("Cache-Control", "public,max-age=31536000"); // save this file to cache for 1 year (unless you refresh) + request->send(response); + Serial.println("Served Basic HTML Page"); }); + + // the catch all + server.onNotFound([](AsyncWebServerRequest *request) + { + request->redirect(localIPURL); + Serial.print("onnotfound "); + Serial.print(request->host()); // This gives some insight into whatever was being requested on the serial monitor + Serial.print(" "); + Serial.print(request->url()); + Serial.print(" sent redirect to " + localIPURL + "\n"); }); + + server.on("/submit", HTTP_POST, [](AsyncWebServerRequest *request) + { + String ssid; + String password; + + // Check if SSID parameter exists and assign it + if(request->hasParam("ssid", true)) { + ssid = request->getParam("ssid", true)->value(); + } + + // Check if Password parameter exists and assign it + if(request->hasParam("password", true)) { + password = request->getParam("password", true)->value(); + } + + // Attempt to connect to the Wi-Fi network with these credentials + connectToWifi(ssid, password); + + // Redirect user or send a response back + if (WiFi.status() == WL_CONNECTED) { + String htmlContent = post_connected_html; + AsyncWebServerResponse *response = request->beginResponse(200, "text/html", htmlContent); + response->addHeader("Cache-Control", "public,max-age=31536000"); // save this file to cache for 1 year (unless you refresh) + request->send(response); + Serial.println("Served Post connection HTML Page"); + } else { + request->send(200, "text/plain", "Failed to connect to " + ssid); + } }); + + server.on("/submit_01os", HTTP_POST, [](AsyncWebServerRequest *request) + { + String server_address; + + // Check if SSID parameter exists and assign it + if(request->hasParam("server_address", true)) { + server_address = request->getParam("server_address", true)->value(); + } + + // Attempt to connect to the Wi-Fi network with these credentials + connectTo01OS(server_address); + + // Redirect user or send a response back + request->send(200, "text/plain", "Attempting to connect to 01OS " + server_address); }); +} + +// ----------------------- END OF WIFI CAPTIVE PORTAL ------------------- + +// ----------------------- START OF PLAYBACK ------------------- + +#define CONFIG_I2S_BCK_PIN 19 +#define CONFIG_I2S_LRCK_PIN 33 +#define CONFIG_I2S_DATA_PIN 22 +#define CONFIG_I2S_DATA_IN_PIN 23 +#define SPEAKER_I2S_NUMBER I2S_NUM_0 +#define MODE_MIC 0 +#define MODE_SPK 1 +#define DATA_SIZE 1024 + +uint8_t microphonedata0[1024 * 10]; +uint8_t speakerdata0[1024 * 1]; +int speaker_offset; +int data_offset; + +bool recording = false; +WebSocketsClient webSocket; + +class ButtonChecker +{ +public: + void loop() + { + lastTickState = thisTickState; + thisTickState = M5.Btn.isPressed() != 0; + } + + bool justPressed() + { + return thisTickState && !lastTickState; + } + + bool justReleased() + { + return !thisTickState && lastTickState; + } + +private: + bool lastTickState = false; + bool thisTickState = false; +}; + +ButtonChecker button = ButtonChecker(); + +void InitI2SSpeakerOrMic(int mode) +{ + Serial.printf("InitI2sSpeakerOrMic %d\n", mode); + esp_err_t err = ESP_OK; + + i2s_driver_uninstall(SPEAKER_I2S_NUMBER); + i2s_config_t i2s_config = { + .mode = (i2s_mode_t)(I2S_MODE_MASTER), + .sample_rate = 16000, + .bits_per_sample = + I2S_BITS_PER_SAMPLE_16BIT, // is fixed at 12bit, stereo, MSB + .channel_format = I2S_CHANNEL_FMT_ALL_RIGHT, +#if ESP_IDF_VERSION > ESP_IDF_VERSION_VAL(4, 1, 0) + .communication_format = + I2S_COMM_FORMAT_STAND_I2S, // Set the format of the communication. +#else // 设置通讯格式 + .communication_format = I2S_COMM_FORMAT_I2S, +#endif + .intr_alloc_flags = ESP_INTR_FLAG_LEVEL1, + .dma_buf_count = 6, + .dma_buf_len = 60, + }; + if (mode == MODE_MIC) + { + i2s_config.mode = + (i2s_mode_t)(I2S_MODE_MASTER | I2S_MODE_RX | I2S_MODE_PDM); + } + else + { + i2s_config.mode = (i2s_mode_t)(I2S_MODE_MASTER | I2S_MODE_TX); + i2s_config.use_apll = false; + i2s_config.tx_desc_auto_clear = true; + } + + err += i2s_driver_install(SPEAKER_I2S_NUMBER, &i2s_config, 0, NULL); + i2s_pin_config_t tx_pin_config; + +#if (ESP_IDF_VERSION > ESP_IDF_VERSION_VAL(4, 3, 0)) + tx_pin_config.mck_io_num = I2S_PIN_NO_CHANGE; +#endif + tx_pin_config.bck_io_num = CONFIG_I2S_BCK_PIN; + tx_pin_config.ws_io_num = CONFIG_I2S_LRCK_PIN; + tx_pin_config.data_out_num = CONFIG_I2S_DATA_PIN; + tx_pin_config.data_in_num = CONFIG_I2S_DATA_IN_PIN; + + // Serial.println("Init i2s_set_pin"); + err += i2s_set_pin(SPEAKER_I2S_NUMBER, &tx_pin_config); + // Serial.println("Init i2s_set_clk"); + err += i2s_set_clk(SPEAKER_I2S_NUMBER, 16000, I2S_BITS_PER_SAMPLE_16BIT, + I2S_CHANNEL_MONO); +} + +void speaker_play(uint8_t *payload, uint32_t len) +{ + Serial.printf("received %lu bytes", len); + size_t bytes_written; + InitI2SSpeakerOrMic(MODE_SPK); + i2s_write(SPEAKER_I2S_NUMBER, payload, len, + &bytes_written, portMAX_DELAY); +} + +void webSocketEvent(WStype_t type, uint8_t *payload, size_t length) +{ + switch (type) + { + case WStype_DISCONNECTED: + Serial.printf("[WSc] Disconnected!\n"); + break; + case WStype_CONNECTED: + Serial.printf("[WSc] Connected to url: %s\n", payload); + + // send message to server when Connected + break; + case WStype_TEXT: + Serial.printf("[WSc] get text: %s\n", payload); + { + std::string str(payload, payload + length); + bool isAudio = str.find("\"audio\"") != std::string::npos; + if (isAudio && str.find("\"start\"") != std::string::npos) + { + Serial.println("start playback"); + speaker_offset = 0; + InitI2SSpeakerOrMic(MODE_SPK); + } + else if (isAudio && str.find("\"end\"") != std::string::npos) + { + Serial.println("end playback"); + // speaker_play(speakerdata0, speaker_offset); + // speaker_offset = 0; + } + } + // send message to server + // webSocket.sendTXT("message here"); + break; + case WStype_BIN: + Serial.printf("[WSc] get binary length: %u\n", length); + memcpy(speakerdata0 + speaker_offset, payload, length); + speaker_offset += length; + size_t bytes_written; + i2s_write(SPEAKER_I2S_NUMBER, speakerdata0, speaker_offset, &bytes_written, portMAX_DELAY); + speaker_offset = 0; + + // send data to server + // webSocket.sendBIN(payload, length); + break; + case WStype_ERROR: + case WStype_FRAGMENT_TEXT_START: + case WStype_FRAGMENT_BIN_START: + case WStype_FRAGMENT: + case WStype_FRAGMENT_FIN: + break; + } +} + +void websocket_setup(String server_domain, int port) +{ + if (WiFi.status() != WL_CONNECTED) + { + Serial.println("Not connected to WiFi. Abandoning setup websocket"); + return; + } + Serial.println("connected to WiFi"); + webSocket.begin(server_domain, port, "/"); + webSocket.onEvent(webSocketEvent); + // webSocket.setAuthorization("user", "Password"); + webSocket.setReconnectInterval(5000); +} + +void flush_microphone() +{ + Serial.printf("[microphone] flushing %d bytes of data\n", data_offset); + if (data_offset == 0) + return; + webSocket.sendBIN(microphonedata0, data_offset); + data_offset = 0; +} + +// ----------------------- END OF PLAYBACK ------------------- + +bool hasSetupWebsocket = false; + +void setup() +{ + // Set the transmit buffer size for the Serial object and start it with a baud rate of 115200. + Serial.setTxBufferSize(1024); + Serial.begin(115200); + + // Wait for the Serial object to become available. + while (!Serial) + ; + + WiFi.mode(WIFI_AP_STA); + + // Print a welcome message to the Serial port. + Serial.println("\n\nCaptive Test, V0.5.0 compiled " __DATE__ " " __TIME__ " by CD_FER"); //__DATE__ is provided by the platformio ide + Serial.printf("%s-%d\n\r", ESP.getChipModel(), ESP.getChipRevision()); + + startSoftAccessPoint(ssid, password, localIP, gatewayIP); + + setUpDNSServer(dnsServer, localIP); + + WiFi.scanNetworks(true); + + setUpWebserver(server, localIP); + server.begin(); + + Serial.print("\n"); + Serial.print("Startup Time:"); // should be somewhere between 270-350 for Generic ESP32 (D0WDQ6 chip, can have a higher startup time on first boot) + Serial.println(millis()); + Serial.print("\n"); + + M5.begin(true, false, true); + M5.dis.drawpix(0, CRGB(128, 128, 0)); +} + +void loop() +{ + dnsServer.processNextRequest(); // I call this atleast every 10ms in my other projects (can be higher but I haven't tested it for stability) + delay(DNS_INTERVAL); // seems to help with stability, if you are doing other things in the loop this may not be needed + + // Check WiFi connection status + if (WiFi.status() == WL_CONNECTED && !hasSetupWebsocket) + { + if (server_domain != "") + { + Serial.println("Setting up websocket to 01OS " + server_domain + ":" + server_port); + + websocket_setup(server_domain, server_port); + InitI2SSpeakerOrMic(MODE_SPK); + + hasSetupWebsocket = true; + + Serial.println("Websocket connection flow completed"); + } + else + { + Serial.println("No valid 01OS server address yet..."); + } + // If connected, you might want to do something, like printing the IP address + // Serial.println("Connected to WiFi!"); + // Serial.println("IP Address: " + WiFi.localIP().toString()); + // Serial.println("SSID " + WiFi.SSID()); + } + if (WiFi.status() == WL_CONNECTED && hasSetupWebsocket) + { + button.loop(); + if (button.justPressed()) + { + Serial.println("Recording..."); + webSocket.sendTXT("{\"role\": \"user\", \"type\": \"audio\", \"format\": \"bytes.raw\", \"start\": true}"); + InitI2SSpeakerOrMic(MODE_MIC); + recording = true; + data_offset = 0; + Serial.println("Recording ready."); + } + else if (button.justReleased()) + { + Serial.println("Stopped recording."); + webSocket.sendTXT("{\"role\": \"user\", \"type\": \"audio\", \"format\": \"bytes.raw\", \"end\": true}"); + flush_microphone(); + recording = false; + data_offset = 0; + } + else if (recording) + { + Serial.printf("Reading chunk at %d...\n", data_offset); + size_t bytes_read; + i2s_read( + SPEAKER_I2S_NUMBER, + (char *)(microphonedata0 + data_offset), + DATA_SIZE, &bytes_read, (100 / portTICK_RATE_MS)); + data_offset += bytes_read; + Serial.printf("Read %d bytes in chunk.\n", bytes_read); + + if (data_offset > 1024 * 9) + { + flush_microphone(); + } + } + + M5.update(); + webSocket.loop(); + } +} \ No newline at end of file diff --git a/01OS/01OS/clients/esp32/wifi_captiveportal/wifi_captiveportal.ino b/01OS/01OS/clients/esp32/wifi_captiveportal/wifi_captiveportal.ino deleted file mode 100644 index 477bc10..0000000 --- a/01OS/01OS/clients/esp32/wifi_captiveportal/wifi_captiveportal.ino +++ /dev/null @@ -1,392 +0,0 @@ -#include //not needed in the arduino ide - -// Captive Portal -#include //https://github.com/me-no-dev/AsyncTCP using the latest dev version from @me-no-dev -#include -#include //https://github.com/me-no-dev/ESPAsyncWebServer using the latest dev version from @me-no-dev -#include //Used for mpdu_rx_disable android workaround -#include -#include -#include - -// Pre reading on the fundamentals of captive portals https://textslashplain.com/2022/06/24/captive-portals/ - -const char *ssid = "captive"; // FYI The SSID can't have a space in it. -// const char * password = "12345678"; //Atleast 8 chars -const char *password = NULL; // no password - -#define MAX_CLIENTS 4 // ESP32 supports up to 10 but I have not tested it yet -#define WIFI_CHANNEL 6 // 2.4ghz channel 6 https://en.wikipedia.org/wiki/List_of_WLAN_channels#2.4_GHz_(802.11b/g/n/ax) - -const IPAddress localIP(4, 3, 2, 1); // the IP address the web server, Samsung requires the IP to be in public space -const IPAddress gatewayIP(4, 3, 2, 1); // IP address of the network should be the same as the local IP for captive portals -const IPAddress subnetMask(255, 255, 255, 0); // no need to change: https://avinetworks.com/glossary/subnet-mask/ - -const String localIPURL = "http://4.3.2.1"; // a string version of the local IP with http, used for redirecting clients to your webpage - -String generateHTMLWithSSIDs() -{ - String html = "

Select Wi-Fi Network



"; - - return html; -} - -const char index_html[] PROGMEM = R"=====( - - - - WiFi Setup - - - - -

WiFi Setup

-
-
-
-
-

- -
- - -)====="; - -const char post_connected_html[] PROGMEM = R"=====( - - - - 01OS Setup - - - - -

01OS Setup

-
-
-
- -
- - -)====="; - -DNSServer dnsServer; -AsyncWebServer server(80); - -void setUpDNSServer(DNSServer &dnsServer, const IPAddress &localIP) -{ -// Define the DNS interval in milliseconds between processing DNS requests -#define DNS_INTERVAL 30 - - // Set the TTL for DNS response and start the DNS server - dnsServer.setTTL(3600); - dnsServer.start(53, "*", localIP); -} - -void startSoftAccessPoint(const char *ssid, const char *password, const IPAddress &localIP, const IPAddress &gatewayIP) -{ -// Define the maximum number of clients that can connect to the server -#define MAX_CLIENTS 4 -// Define the WiFi channel to be used (channel 6 in this case) -#define WIFI_CHANNEL 6 - - // Set the WiFi mode to access point and station - // WiFi.mode(WIFI_MODE_AP); - - // Define the subnet mask for the WiFi network - const IPAddress subnetMask(255, 255, 255, 0); - - // Configure the soft access point with a specific IP and subnet mask - WiFi.softAPConfig(localIP, gatewayIP, subnetMask); - - // Start the soft access point with the given ssid, password, channel, max number of clients - WiFi.softAP(ssid, password, WIFI_CHANNEL, 0, MAX_CLIENTS); - - // Disable AMPDU RX on the ESP32 WiFi to fix a bug on Android - esp_wifi_stop(); - esp_wifi_deinit(); - wifi_init_config_t my_config = WIFI_INIT_CONFIG_DEFAULT(); - my_config.ampdu_rx_enable = false; - esp_wifi_init(&my_config); - esp_wifi_start(); - vTaskDelay(100 / portTICK_PERIOD_MS); // Add a small delay -} - -void connectToWifi(String ssid, String password) -{ - WiFi.begin(ssid.c_str(), password.c_str()); - - // Wait for connection to establish - int attempts = 0; - while (WiFi.status() != WL_CONNECTED && attempts < 20) - { - delay(1000); - Serial.print("."); - attempts++; - } - - if (WiFi.status() == WL_CONNECTED) - { - Serial.println("Connected to Wi-Fi"); - } - else - { - Serial.println("Failed to connect to Wi-Fi. Check credentials."); - } -} - -// Number of milliseconds to wait without receiving any data before we give up -const int kNetworkTimeout = 30*1000; -// Number of milliseconds to wait if no data is available before trying again -const int kNetworkDelay = 1000; - -void connectTo01OS(String server_address) -{ - int err = 0; - WiFiClient c; - HttpClient http(c); - - String domain = server_address.substring(0, server_address.indexOf(':')); - String portStr = server_address.substring(server_address.indexOf(':') + 1); - int port = portStr.toInt(); - - Serial.println("Connecting to 01OS at "+domain+":"+port+"/ping"); - err = http.get(domain.c_str(), port, "/ping"); - //err = http.get("arduino.cc", "/"); - - if (err == 0) - { - Serial.println("Started the ping request"); - - err = http.responseStatusCode(); - if (err >= 0) - { - Serial.print("Got status code: "); - Serial.println(err); - - err = http.skipResponseHeaders(); - if (err >= 0) - { - int bodyLen = http.contentLength(); - Serial.print("Content length is: "); - Serial.println(bodyLen); - Serial.println(); - Serial.println("Body returned follows:"); - - // Now we've got to the body, so we can print it out - unsigned long timeoutStart = millis(); - char c; - // Whilst we haven't timed out & haven't reached the end of the body - while ((http.connected() || http.available()) && - ((millis() - timeoutStart) < kNetworkTimeout)) - { - if (http.available()) - { - c = http.read(); - // Print out this character - Serial.print(c); - - bodyLen--; - // We read something, reset the timeout counter - timeoutStart = millis(); - } - else - { - // We haven't got any data, so let's pause to allow some to - // arrive - delay(kNetworkDelay); - } - } - } - else - { - Serial.print("Failed to skip response headers: "); - Serial.println(err); - } - } - else - { - Serial.print("Getting response failed: "); - Serial.println(err); - } - } - else - { - Serial.print("Connect failed: "); - Serial.println(err); - } -} - -void setUpWebserver(AsyncWebServer &server, const IPAddress &localIP) -{ - //======================== Webserver ======================== - // WARNING IOS (and maybe macos) WILL NOT POP UP IF IT CONTAINS THE WORD "Success" https://www.esp8266.com/viewtopic.php?f=34&t=4398 - // SAFARI (IOS) IS STUPID, G-ZIPPED FILES CAN'T END IN .GZ https://github.com/homieiot/homie-esp8266/issues/476 this is fixed by the webserver serve static function. - // SAFARI (IOS) there is a 128KB limit to the size of the HTML. The HTML can reference external resources/images that bring the total over 128KB - // SAFARI (IOS) popup browser has some severe limitations (javascript disabled, cookies disabled) - - // Required - server.on("/connecttest.txt", [](AsyncWebServerRequest *request) - { request->redirect("http://logout.net"); }); // windows 11 captive portal workaround - server.on("/wpad.dat", [](AsyncWebServerRequest *request) - { request->send(404); }); // Honestly don't understand what this is but a 404 stops win 10 keep calling this repeatedly and panicking the esp32 :) - - // Background responses: Probably not all are Required, but some are. Others might speed things up? - // A Tier (commonly used by modern systems) - server.on("/generate_204", [](AsyncWebServerRequest *request) - { request->redirect(localIPURL); }); // android captive portal redirect - server.on("/redirect", [](AsyncWebServerRequest *request) - { request->redirect(localIPURL); }); // microsoft redirect - server.on("/hotspot-detect.html", [](AsyncWebServerRequest *request) - { request->redirect(localIPURL); }); // apple call home - server.on("/canonical.html", [](AsyncWebServerRequest *request) - { request->redirect(localIPURL); }); // firefox captive portal call home - server.on("/success.txt", [](AsyncWebServerRequest *request) - { request->send(200); }); // firefox captive portal call home - server.on("/ncsi.txt", [](AsyncWebServerRequest *request) - { request->redirect(localIPURL); }); // windows call home - - // B Tier (uncommon) - // server.on("/chrome-variations/seed",[](AsyncWebServerRequest *request){request->send(200);}); //chrome captive portal call home - // server.on("/service/update2/json",[](AsyncWebServerRequest *request){request->send(200);}); //firefox? - // server.on("/chat",[](AsyncWebServerRequest *request){request->send(404);}); //No stop asking Whatsapp, there is no internet connection - // server.on("/startpage",[](AsyncWebServerRequest *request){request->redirect(localIPURL);}); - - // return 404 to webpage icon - server.on("/favicon.ico", [](AsyncWebServerRequest *request) - { request->send(404); }); // webpage icon - - // Serve Basic HTML Page - server.on("/", HTTP_ANY, [](AsyncWebServerRequest *request) - { - String htmlContent = index_html; - Serial.printf("wifi scan complete: %d . WIFI_SCAN_RUNNING: %d", WiFi.scanComplete(), WIFI_SCAN_RUNNING); - if(WiFi.scanComplete() > 0) { - // Scan complete, process results - Serial.println("done scanning wifi"); - htmlContent = generateHTMLWithSSIDs(); - // WiFi.scanNetworks(true); // Start a new scan in async mode - } - AsyncWebServerResponse *response = request->beginResponse(200, "text/html", htmlContent); - response->addHeader("Cache-Control", "public,max-age=31536000"); // save this file to cache for 1 year (unless you refresh) - request->send(response); - Serial.println("Served Basic HTML Page"); }); - - // the catch all - server.onNotFound([](AsyncWebServerRequest *request) - { - request->redirect(localIPURL); - Serial.print("onnotfound "); - Serial.print(request->host()); // This gives some insight into whatever was being requested on the serial monitor - Serial.print(" "); - Serial.print(request->url()); - Serial.print(" sent redirect to " + localIPURL + "\n"); }); - - server.on("/submit", HTTP_POST, [](AsyncWebServerRequest *request) - { - String ssid; - String password; - - // Check if SSID parameter exists and assign it - if(request->hasParam("ssid", true)) { - ssid = request->getParam("ssid", true)->value(); - } - - // Check if Password parameter exists and assign it - if(request->hasParam("password", true)) { - password = request->getParam("password", true)->value(); - } - - // Attempt to connect to the Wi-Fi network with these credentials - connectToWifi(ssid, password); - - // Redirect user or send a response back - if (WiFi.status() == WL_CONNECTED) { - String htmlContent = post_connected_html; - AsyncWebServerResponse *response = request->beginResponse(200, "text/html", htmlContent); - response->addHeader("Cache-Control", "public,max-age=31536000"); // save this file to cache for 1 year (unless you refresh) - request->send(response); - Serial.println("Served Post connection HTML Page"); - } else { - request->send(200, "text/plain", "Failed to connect to " + ssid); - } - }); - - server.on("/submit_01os", HTTP_POST, [](AsyncWebServerRequest *request) - { - String server_address; - - // Check if SSID parameter exists and assign it - if(request->hasParam("server_address", true)) { - server_address = request->getParam("server_address", true)->value(); - } - - // Attempt to connect to the Wi-Fi network with these credentials - connectTo01OS(server_address); - - // Redirect user or send a response back - request->send(200, "text/plain", "Attempting to connect to 01OS " + server_address); }); -} - -void setup() -{ - // Set the transmit buffer size for the Serial object and start it with a baud rate of 115200. - Serial.setTxBufferSize(1024); - Serial.begin(115200); - - // Wait for the Serial object to become available. - while (!Serial) - ; - - WiFi.mode(WIFI_AP_STA); - - // Print a welcome message to the Serial port. - Serial.println("\n\nCaptive Test, V0.5.0 compiled " __DATE__ " " __TIME__ " by CD_FER"); //__DATE__ is provided by the platformio ide - Serial.printf("%s-%d\n\r", ESP.getChipModel(), ESP.getChipRevision()); - - startSoftAccessPoint(ssid, password, localIP, gatewayIP); - - setUpDNSServer(dnsServer, localIP); - - WiFi.scanNetworks(true); - - setUpWebserver(server, localIP); - server.begin(); - - Serial.print("\n"); - Serial.print("Startup Time:"); // should be somewhere between 270-350 for Generic ESP32 (D0WDQ6 chip, can have a higher startup time on first boot) - Serial.println(millis()); - Serial.print("\n"); -} - -void loop() -{ - dnsServer.processNextRequest(); // I call this atleast every 10ms in my other projects (can be higher but I haven't tested it for stability) - delay(DNS_INTERVAL); // seems to help with stability, if you are doing other things in the loop this may not be needed - - // Check WiFi connection status - //if (WiFi.status() == WL_CONNECTED) - //{ - // If connected, you might want to do something, like printing the IP address - //Serial.println("Connected to WiFi!"); - //Serial.println("IP Address: " + WiFi.localIP().toString()); - //Serial.println("SSID " + WiFi.SSID()); - //} -} \ No newline at end of file