本實驗製作一個WiFi音箱,使用著介面使用Node-RED Dashboard,透過MQTT protocol控制分布在不同地點的音箱,來播放音樂、網路收音機、Text to Speech廣播。可分別針對個別或全部音箱即時或排程播送。
本實驗可應用於像校園廣播系統,可排程上下課鐘聲、或活動音樂分區播放、透過Node-RED dashboard介面輸入文字即可廣播語音。
一、實驗使用元件:
- ESP32 NodeMCU-32S開發版
- MAX98357A模組
- RGB LED
- NPN 電晶體
- 3W4Ω喇叭
- 200Ω電阻
二、開發軟體環境:
- ESP-ADF(ESP-IDF)
- VSCode
- Node_RED
- Mosquitto MQTT broker
- SQLite
三、功能說明:
- 即時顯示每個喇叭狀態(斷線、已連線或播放中)
- Dashboard 能自動動態增加連線喇叭數量。
- 設備端LED顯示狀態(LED紅燈為斷線、綠燈正常連線中)
- 可針對各別或全部喇叭操作
- 可即時播放音樂、網路收音機或TTS語音。
- 可排程(月、日、時、分、星期)播放音樂、網路收音機或TTS語音。
- MQTT使用TLS連線與帳密安全保護
- 能使用HTTPS(secure HTTP) URI
四、實驗過程:
1.定義MQTT TOPIC and message flows:
TOPIC分成兩大類,一為Node-RED送出控制Wifi Speaker動作,與Wifi Speaker回報狀態(STATE)給Node-RED,分別為/WIFI_SPEAKER/DeviceID/CMD 與 /WIFI_SPEAKER/DeviceID/STATE,DeviceID為每個WiFi Speaker MAC Address 的後3bytes,用於區別對個別的音箱做控制與狀態顯示。詳細定義如下圖:
- CMD topic :
TOPIC:/WIFI_SPEAKER/deviceID/CMD,
PAYLOAD: {"action":"play", "uri":"https://...../..mp3"}
PAYLOAD: {"action":"stop"}
PAYLOAD: {"action":"pause"}
PAYLOAD: {"action":"resume"}
PAYLOAD: {"action":"volume", "value":number} - STATE topic:
PAYLOAD: {"ID":"deviceID", "NAME":"deviceName", "TYPE":"NETWORK","STATE":"OFFLINE"}
PAYLOAD: {"ID":"deviceID", "NAME":"deviceName", "TYPE":"NETWORK","STATE":"CONNECT"}
PAYLOAD: {"ID":"deviceID", "TYPE":"MUSIC_PLAY","STATE":"play"}
PAYLOAD: {"ID":"deviceID", "TYPE":"MUSIC_PLAY","STATE":"pause"}
PAYLOAD: {"ID":"deviceID", "TYPE":"MUSIC_PLAY","STATE":"resume"}
PAYLOAD: {"ID":"deviceID", "TYPE":"MUSIC_PLAY","STATE":"stop"}
PAYLOAD: {"ID":"deviceID", "TYPE":"MUSIC_PLAY","STATE":"FINISH"}
兩個端點Node-RED與device(Wifi Speaker)對於接收到MQTT Topic要實行的端點動作如下圖所示:
2.ESP-ADF:
ESP-ADF (Espressif Audio Development Framework),是ESP-IDF的擴充組件,用於開發Audio Application。在使用VSCode下,按下F1即可安裝ESP-ADF擴充組件,如下圖所示
有關ESP-ADF詳細文件說明位址:https://docs.espressif.com/projects/esp-adf/en/latest/。
本實驗以播放MP3格式的聲音為主,加入其他audio element codec即可播放其他格式audio,便於開發。
主要架構以audio pipeline透過ringbuffer串聯input audio element, process(codec) audio element與 output audio element,如下圖所示(取自官方文件)
本實驗I2S Output to DAC,不使用ESP32內建8bit DAC,使用MAX98357A 包含DAC與Ampiler。
3.TTS(Text To Speech)播放:
本實驗使用Google Translate TTS把Text轉成MP3語音於無線音箱播放。可使用兩種方式實行。
- 安裝Node-RED google-translate-tts套件,將text轉成MP3 audio file存到server local file,再將local audio file的uri透過MQTT protocol送到WiFi Speaker播放,如下flow所示:
- 安裝Node-RED node-red-contrib-google-tts套件,將欲播放的TEXT當成uri參數,透過node-red-contrib-google-tts轉成google translate URI,再透過MQTT protocol傳URI給WiFi Speaker播放。flow如下圖所示:
- URI參數: a. tl=:為指定語言(en:英文, zh-TW中文)。 b. q=:為欲播放的語音文字。搜尋網路上有關google translate TTS的探討,URI亦可簡化類似:https://translate.google.com/translate_tts?ie=UTF-8&client=tw-ob&tl=en&ttsspeed=1&q=TEXT。
4.硬體線路:
- PIN_NUM_23連接npn電晶體與RGB LED用來指示Wifi連線MQTT是否正常,紅燈為斷線,綠燈為正常連線。
- PIN_NUM_5 I2S SCLK, PIN_NUM_25 I2S LRCK, PIN_NUM_26 I2S DSDIN
4.Node-RED flows:
五、實驗成果展示
六、原始碼
parm_define.h
#define PRO_CPU 0 #define APP_CPU 1 #define SSID "your-ssid" #define PWD "your-pwd" #define MQTT_HOST "your_mqtt_broker" #define MQTT_PORT 8883 #define MQTT_CLIENT_USER "your_mqtt_user" #define MQTT_CLIENT_PASSWORD "your_mqtt_pwd" #define DEVICE_NAME "define_your_device_name" static const char cert[] = "-----BEGIN CERTIFICATE-----\n" "................................................................\n" ".....................\n" "-----END CERTIFICATE-----"; static const uint8_t client_cert[]="-----BEGIN CERTIFICATE-----\n" "................................................................\n" "..................\n" "-----END CERTIFICATE-----"; static const uint8_t client_key[] = "-----BEGIN RSA PRIVATE KEY-----\n" "................................................................\n" ".....................\n" "-----END RSA PRIVATE KEY-----";
wifi_speaker.c
#include <string.h> #include "freertos/FreeRTOS.h" #include "freertos/task.h" #include "freertos/event_groups.h" #include "nvs_flash.h" #include "audio_element.h" #include "audio_pipeline.h" #include "audio_event_iface.h" #include "audio_common.h" #include "http_stream.h" #include "i2s_stream.h" #include "mp3_decoder.h" #include "driver/i2c.h" #include "esp_log.h" #include "esp_audio.h" #include "esp_wifi.h" #include "esp_peripherals.h" #include "esp_netif.h" #include "periph_wifi.h" #include "esp_tls.h" #include "driver/gpio.h" #include "cJSON.h" #include "mqtt_client.h" #include "param_define.h" //wifi and mqtt connection parameters static audio_event_iface_handle_t evt={0}; static audio_pipeline_handle_t pipeline; static audio_element_handle_t http_stream_reader, i2s_stream_writer, mp3_decoder; static esp_mqtt_client_handle_t hMQTTClient; static char sID[7]; static char cmd_topic[50]={0}; static char state_topic[50]={0}; static esp_err_t mqtt_event_handler_cb(esp_mqtt_event_handle_t event) { esp_mqtt_client_handle_t client = event->client; char msg[100]={0}; char topic[100] = {0}; switch (event->event_id) { case MQTT_EVENT_CONNECTED: sprintf(msg, "{\"ID\":\"%s\",\"NAME\":\"%s\",\"TYPE\":\"NETWORK\",\"STATE\":\"CONNECTED\"}", sID, DEVICE_NAME); esp_mqtt_client_publish(client, state_topic, msg, 0, 1, 0); esp_mqtt_client_subscribe(client, cmd_topic, 1); gpio_set_level(GPIO_NUM_23, 0); break; case MQTT_EVENT_DISCONNECTED: gpio_set_level(GPIO_NUM_23, 1); esp_mqtt_client_reconnect(client); break; case MQTT_EVENT_SUBSCRIBED: break; case MQTT_EVENT_UNSUBSCRIBED: break; case MQTT_EVENT_PUBLISHED: break; case MQTT_EVENT_DATA: strncpy(topic, event->topic, event->topic_len); if (strcmp(topic, cmd_topic) == 0) { cJSON *root = cJSON_Parse(event->data); char *action=cJSON_GetObjectItem(root, "action")->valuestring; if (strcmp(action,"play")==0) { char *uri = cJSON_GetObjectItem(root, "uri")->valuestring; int http_state = audio_element_get_state(http_stream_reader); if (http_state == AEL_STATE_PAUSED) { audio_pipeline_resume(pipeline); while((http_state = audio_element_get_state(http_stream_reader)) != AEL_STATE_RUNNING) { vTaskDelay(10/portTICK_PERIOD_MS); } } audio_pipeline_stop(pipeline); audio_pipeline_wait_for_stop(pipeline); audio_pipeline_reset_elements(pipeline); audio_pipeline_reset_ringbuffer(pipeline); audio_element_set_uri(http_stream_reader,uri); audio_pipeline_run(pipeline); sprintf(msg, "{\"ID\":\"%s\",\"TYPE\":\"MUSIC_PLAY\",\"STATE\":\"play\"}", sID); esp_mqtt_client_publish(client,state_topic,msg,0,1,0); } if (strcmp(action,"stop")==0) { audio_pipeline_stop(pipeline); audio_pipeline_wait_for_stop(pipeline); sprintf(msg, "{\"ID\":\"%s\",\"TYPE\":\"MUSIC_PLAY\",\"STATE\":\"stop\"}", sID); esp_mqtt_client_publish(client,state_topic,msg,0,1,0); } if (strcmp(action,"pause")==0) { audio_pipeline_pause(pipeline); sprintf(msg, "{\"ID\":\"%s\",\"TYPE\":\"MUSIC_PLAY\",\"STATE\":\"pause\"}", sID); esp_mqtt_client_publish(client,state_topic,msg,0,1,0); } if (strcmp(action,"resume")==0) { audio_pipeline_resume(pipeline); sprintf(msg, "{\"ID\":\"%s\",\"TYPE\":\"MUSIC_PLAY\",\"STATE\":\"resume\"}", sID); esp_mqtt_client_publish(client,state_topic,msg,0,1,0); } if (strcmp(action,"volume")==0) { int volume = cJSON_GetObjectItem(root, "value")->valueint; i2s_alc_volume_set(i2s_stream_writer, volume); sprintf(msg, "{\"ID\":\"%s\",\"TYPE\":\"MUSIC_PLAY\",\"STATE\":\"VOLUME\",\"volume\":%d}", sID, volume); esp_mqtt_client_publish(client,state_topic,msg,0,1,0); } } break; case MQTT_EVENT_ERROR: if (event->error_handle->error_type == MQTT_ERROR_TYPE_TCP_TRANSPORT) { } break; default: break; } return ESP_OK; } static void mqtt_event_handler(void *handler_args, esp_event_base_t base, int32_t event_id, void *event_data) { mqtt_event_handler_cb(event_data); } static void mqtt_client_start(void) { char msg[100]; sprintf(msg, "{\"ID\":\"%s\",\"NAME\":\"%s\",\"TYPE\":\"NETWORK\",\"STATE\":\"OFFLINE\"}", sID, DEVICE_NAME); esp_mqtt_client_config_t mqtt_cfg = { .host = MQTT_HOST, .port = MQTT_PORT, //.uri = MQTT_URI, .username = MQTT_CLIENT_USER, .password = MQTT_CLIENT_PASSWORD, .lwt_topic = state_topic, .lwt_msg = msg, .keepalive=30, .cert_pem=(const char *)cert, .cert_len=0, .client_cert_pem=(const char *)client_cert, .client_cert_len=0, .client_key_pem=(const char *)client_key, .client_key_len=0, .transport = MQTT_TRANSPORT_OVER_SSL }; hMQTTClient = esp_mqtt_client_init(&mqtt_cfg); esp_mqtt_client_register_event(hMQTTClient, ESP_EVENT_ANY_ID, mqtt_event_handler, hMQTTClient); esp_mqtt_client_start(hMQTTClient); } void wifi_speaker_audio_pipeline_init(void) { audio_pipeline_cfg_t pipeline_cfg = DEFAULT_AUDIO_PIPELINE_CONFIG(); //pipeline_cfg.rb_size=40960; pipeline = audio_pipeline_init(&pipeline_cfg); mem_assert(pipeline); // audio element http_stream http_stream_cfg_t http_cfg = HTTP_STREAM_CFG_DEFAULT(); http_cfg.type = AUDIO_STREAM_READER; http_cfg.task_core = APP_CPU; http_stream_reader = http_stream_init(&http_cfg); //audio element i2s_stream i2s_stream_cfg_t i2s_cfg = I2S_STREAM_CFG_DEFAULT(); i2s_cfg.task_core = APP_CPU; i2s_cfg.use_alc=true; i2s_cfg.type = AUDIO_STREAM_WRITER; i2s_stream_writer = i2s_stream_init(&i2s_cfg); // audio element mp3_decoder mp3_decoder_cfg_t mp3_cfg = DEFAULT_MP3_DECODER_CONFIG(); mp3_cfg.task_core = APP_CPU; mp3_decoder = mp3_decoder_init(&mp3_cfg); // register audio element in the order http->mp3->i2s(input -> process -> output) audio_pipeline_register(pipeline, http_stream_reader, "http"); audio_pipeline_register(pipeline, mp3_decoder, "mp3"); audio_pipeline_register(pipeline, i2s_stream_writer, "i2s"); // Link it together http_stream-->mp3_decoder-->i2s_stream-->[MAX98357A]"); const char *link_tag[3] = {"http", "mp3", "i2s"}; audio_pipeline_link(pipeline, &link_tag[0], 3); } void app_main(void) { //led turn off gpio_pad_select_gpio(GPIO_NUM_23); gpio_set_direction(GPIO_NUM_23,GPIO_MODE_OUTPUT); gpio_set_level(GPIO_NUM_23, 1); esp_err_t err = nvs_flash_init(); if (err == ESP_ERR_NVS_NO_FREE_PAGES) { // NVS partition was truncated and needs to be erased // Retry nvs_flash_init ESP_ERROR_CHECK(nvs_flash_erase()); err = nvs_flash_init(); } ESP_ERROR_CHECK(esp_netif_init()); //esp_log_level_set("*", ESP_LOG_WARN); //esp_log_level_set(TAG, ESP_LOG_DEBUG); wifi_speaker_audio_pipeline_init(); esp_periph_config_t periph_cfg = DEFAULT_ESP_PERIPH_SET_CONFIG(); esp_periph_set_handle_t set = esp_periph_set_init(&periph_cfg); periph_wifi_cfg_t wifi_cfg = { .ssid = SSID, .password = PWD, }; esp_periph_handle_t wifi_handle = periph_wifi_init(&wifi_cfg); esp_periph_start(set, wifi_handle); esp_err_t cr = periph_wifi_wait_for_connected(wifi_handle, 120000/portTICK_PERIOD_MS); if (cr == ESP_FAIL) { return; } // register audio event audio_event_iface_cfg_t evt_cfg = AUDIO_EVENT_IFACE_DEFAULT_CFG(); evt = audio_event_iface_init(&evt_cfg); // Listening event from all elements of pipeline audio_pipeline_set_listener(pipeline, evt); // Listening event from peripherals audio_event_iface_set_listener(esp_periph_set_get_event_iface(set), evt); uint8_t tmac[6]; esp_efuse_mac_get_default(tmac); sprintf(sID, "%02x%02x%02x", tmac[3],tmac[4],tmac[5]); sprintf(cmd_topic, "/WIFI_SPEAKER/%s/CMD", sID); sprintf(state_topic, "/WIFI_SPEAKER/%s/STATE", sID); // default volume level (range -64 ~ 64) i2s_alc_volume_set(i2s_stream_writer, -20); mqtt_client_start(); while(1) { audio_event_iface_msg_t msg; esp_err_t ret = audio_event_iface_listen(evt, &msg, portMAX_DELAY); if (ret != ESP_OK) { continue; } if (msg.source_type == AUDIO_ELEMENT_TYPE_ELEMENT && msg.source == (void *) mp3_decoder && msg.cmd == AEL_MSG_CMD_REPORT_MUSIC_INFO) { audio_element_info_t music_info = {0}; audio_element_getinfo(mp3_decoder, &music_info); audio_element_setinfo(i2s_stream_writer, &music_info); i2s_stream_set_clk(i2s_stream_writer, music_info.sample_rates, music_info.bits, music_info.channels); continue; } if (msg.source_type == AUDIO_ELEMENT_TYPE_ELEMENT && msg.source == (void *) i2s_stream_writer && msg.cmd == AEL_MSG_CMD_REPORT_STATUS && (int)msg.data == AEL_STATUS_STATE_FINISHED) { char mqttdata[100]; sprintf(mqttdata, "{\"ID\":\"%s\",\"TYPE\":\"MUSIC_PLAY\",\"STATE\":\"FINISH\"}", sID); esp_mqtt_client_publish(hMQTTClient,state_topic,mqttdata,0,1,0); continue; } if (msg.source_type == AUDIO_ELEMENT_TYPE_ELEMENT && msg.source == (void *) http_stream_reader && msg.cmd == AEL_MSG_CMD_REPORT_STATUS && (int)msg.data == AEL_STATUS_ERROR_OPEN) { char mqttdata[100]; sprintf(mqttdata, "{\"ID\":\"%s\",\"TYPE\":\"MUSIC_PLAY\",\"STATE\":\"openerror\"}", sID); esp_mqtt_client_publish(hMQTTClient,state_topic,mqttdata,0,1,0); continue; } } }