diff --git a/CHANGELOG b/CHANGELOG index b50d1f6ab..d7f348162 100644 --- a/CHANGELOG +++ b/CHANGELOG @@ -1,3 +1,26 @@ +2023-11-09 + - force gpio_pad_select_gpio in dac_controlset in case somebody uses UART gpio's (or other pre-programmed) + +2023-11-08 + - execute dac_controlset even whne there is no i2s (for gpio) + +2023-11-07 + - led-vu gain + misc fixes + - bump plugin version to 0.600 + +2023-11-03 + - don't reboot when external decoder is connected even with a LMS server + +2023-10-28 + - fix recovery size (remove bootstrap) + - improve NVS initialization structure + +2023-10-27 + - fix vorbis (and opus) memory leak + +2023-10-25 + - fix vorbis codec close + 2023-10-23 - fix Spotify track insertion - [WEB] Allow running without LMS with option "Audio/Disable Squeezelite" diff --git a/Dockerfile b/Dockerfile index fb8815c65..716f93c3d 100644 --- a/Dockerfile +++ b/Dockerfile @@ -25,7 +25,11 @@ ENV GCC_TOOLS_BASE=/opt/esp/tools/xtensa-esp32-elf/esp-2021r2-patch3-8.4.0/xtens # pushd components/wifi-manager/webapp/ && npm install && npm run-script build && popd # # to run the docker with netwotrk port published on the host: +# (windows) # docker run --rm -p 5000:5000/tcp -v %cd%:/project -w /project -it sle118/squeezelite-esp32-idfv435 +# (linux) +# docker run --rm -p 5000:5000/tcp -v `pwd`:/project -w /project -it sle118/squeezelite-esp32-idfv435 + ARG IDF_CLONE_URL=https://github.com/espressif/esp-idf.git ARG IDF_CLONE_BRANCH_OR_TAG=master diff --git a/README.md b/README.md index ab9cc9021..ba740fa0a 100644 --- a/README.md +++ b/README.md @@ -194,15 +194,19 @@ if "model" is not set or is not recognized, then default "I2S" is used. The opti So far, TAS57xx, TAS5713, AC101, WM8978 and ES8388 are recognized models where the proper init sequence/volume/power controls are sent. For other codecs that might require an I2C commands, please use the parameter "dac_controlset" that allows definition of simple commands to be sent over i2c for init, power, speaker and headset on and off using a JSON syntax: ```json -{ : [ {"reg":,"val":,"mode":|"or"|"and"}, ... {{"reg":,"val":,"mode":|"or"|"and"} ], - : [ {"reg":,"val":,"mode":|"or"|"and"}, ... {{"reg":,"val":,"mode":|"or"|"and"} ], +{ : [ , , ... ], + : [ , , ... ], ... } ``` -Where `` is one of init, poweron, poweroff, speakeron, speakeroff, headseton, headsetoff - -This is standard JSON notation, so if you are not familiar with it, Google is your best friend. Be aware that the '...' means you can have as many entries as you want, it's not part of the syntax. Every section is optional, but it does not make sense to set i2c in the 'dac_config' parameter and not setting anything here. The parameter 'mode' allows to *or* the register with the value or to *and* it. Don't set 'mode' if you simply want to write. The 'val parameter can be an array [v1, v2,...] to write a series of bytes in a single i2c burst (in that case 'mode' is ignored). **Note that all values must be decimal**. You can use a validator like [this](https://jsonlint.com) to verify your syntax +Where `` is one of init, poweron, poweroff, speakeron, speakeroff, headseton, headsetoff (it **must** be an array even for a single item). Item is any of the following elements +``` +{"reg":,"val":,"mode":|"or"|"and"} +{"gpio":,"level":0|1} +{"delay":} +``` +This is standard JSON notation, so if you are not familiar with it, Google is your best friend. Be aware that the '...' means you can have as many entries as you want, it's not part of the syntax. Every section is optional, but it does not make sense to set i2c in the 'dac_config' parameter and not setting anything here. -The 'power' command is used when powering on/off the DAC after the idle period (see -C option of squeezelite) and the 'speaker/headset' commands are sent when switching between speakers and headsets (see headset jack detection). +The `reg` key allow to write registers on i2c bus. The parameter `mode` allows to *or* the register with the value or to *and* it. Don't set `mode` if you simply want to write. The `val` parameter can be an array [v1, v2,...] to write a serie of bytes in a single i2c burst (in that case 'mode' is ignored). **Note that all values must be decimal**. You can use a validator like [this](https://jsonlint.com) to verify your syntax. The `gpio` key is simply to set a gpio as part of DAC action and `delay` allows a pause between elements. The 'power' command is used when powering on/off the DAC after the idle period (see -C option of squeezelite) and the 'speaker/headset' commands are sent when switching between speakers and headsets (see headset jack detection). diff --git a/components/led_strip/led_strip.c b/components/led_strip/led_strip.c index 5441dc237..80d9a0528 100644 --- a/components/led_strip/led_strip.c +++ b/components/led_strip/led_strip.c @@ -391,11 +391,8 @@ bool led_strip_init(struct led_strip_t *led_strip) static EXT_RAM_ATTR StackType_t xStack[LED_STRIP_TASK_SIZE] __attribute__ ((aligned (4))); if ((led_strip == NULL) || - //(led_strip->rmt_channel >= RMT_CHANNEL_MAX) || - //(led_strip->gpio > GPIO_NUM_33) || (led_strip->led_strip_working == NULL) || (led_strip->led_strip_showing == NULL) || - //(led_strip->led_strip_length == 0) || (led_strip->access_semaphore == NULL)) { return false; } diff --git a/components/platform_config/platform_config.c b/components/platform_config/platform_config.c index 3f6c538a1..2753a87d5 100644 --- a/components/platform_config/platform_config.c +++ b/components/platform_config/platform_config.c @@ -538,7 +538,7 @@ bool config_set_group_bit(int bit_num,bool flag){ return result; } -void config_set_default(nvs_type_t type, const char *key, void * default_value, size_t blob_size) { +void config_set_default(nvs_type_t type, const char *key, const void * default_value, size_t blob_size) { if(!config_lock(LOCK_MAX_WAIT/portTICK_PERIOD_MS)){ ESP_LOGE(TAG, "Unable to lock config"); return; diff --git a/components/platform_config/platform_config.h b/components/platform_config/platform_config.h index 9e3aa8037..6af2ecd0e 100644 --- a/components/platform_config/platform_config.h +++ b/components/platform_config/platform_config.h @@ -65,7 +65,7 @@ esp_err_t config_set_cjson_str_and_free(const char *key, cJSON *value); esp_err_t config_set_cjson(const char *key, cJSON *value, bool free_cjson); void config_get_uint16t_from_str(const char *key, uint16_t *value, uint16_t default_value); void config_delete_key(const char *key); -void config_set_default(nvs_type_t type, const char *key, void * default_value, size_t blob_size); +void config_set_default(nvs_type_t type, const char *key, const void * default_value, size_t blob_size); void * config_alloc_get(nvs_type_t nvs_type, const char *key) ; bool wait_for_commit(); char * config_alloc_get_json(bool bFormatted); diff --git a/components/platform_console/cmd_system.c b/components/platform_console/cmd_system.c index 5f1cac3e8..d528864bb 100644 --- a/components/platform_console/cmd_system.c +++ b/components/platform_console/cmd_system.c @@ -73,8 +73,10 @@ static void register_heap(); static void register_dump_heap(); static void register_version(); static void register_restart(); +#if CONFIG_WITH_CONFIG_UI static void register_deep_sleep(); static void register_light_sleep(); +#endif static void register_factory_boot(); static void register_restart_ota(); static void register_set_services(); @@ -92,8 +94,8 @@ FILE * system_open_memstream(const char * cmdname,char **buf,size_t *buf_size){ void register_system() { - register_set_services(); register_setdevicename(); + register_set_services(); register_free(); register_heap(); register_dump_heap(); @@ -585,7 +587,7 @@ static void register_tasks() /** 'deep_sleep' command puts the chip into deep sleep mode */ - +#if CONFIG_WITH_CONFIG_UI static struct { struct arg_int *wakeup_time; struct arg_int *wakeup_gpio_num; @@ -649,6 +651,8 @@ static void register_deep_sleep() }; ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) ); } +#endif + static int enable_disable(FILE * f,char * nvs_name, struct arg_lit *arg){ esp_err_t err = config_set_value(NVS_TYPE_STR, nvs_name, arg->count>0?"Y":"N"); const char * name = arg->hdr.longopts?arg->hdr.longopts:arg->hdr.glossary; @@ -744,9 +748,6 @@ cJSON * set_services_cb(){ else { cJSON_AddStringToObject(values,set_services_args.telnet->hdr.longopts,"Disabled"); } -#if defined(CONFIG_WITH_METRICS) - metrics_add_feature_variant("telnet",p); -#endif FREE_AND_NULL(p); } @@ -778,6 +779,8 @@ static void register_set_services(){ cmd_to_json_with_cb(&cmd,&set_services_cb); ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) ); } + +#if CONFIG_WITH_CONFIG_UI static struct { struct arg_int *wakeup_time; struct arg_int *wakeup_gpio_num; @@ -871,4 +874,4 @@ static void register_light_sleep() }; ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) ); } - +#endif diff --git a/components/platform_console/cmd_wifi.c b/components/platform_console/cmd_wifi.c index ef0033ecb..e6e32d151 100644 --- a/components/platform_console/cmd_wifi.c +++ b/components/platform_console/cmd_wifi.c @@ -180,10 +180,10 @@ void register_wifi_join() void register_wifi() { - #ifdef WIFI_CMDLINE + #ifdef WIFI_CMDLINE register_wifi_join(); if(bypass_network_manager){ initialise_wifi(); } - #endif +#endif } diff --git a/components/services/accessors.h b/components/services/accessors.h index b0e11238c..7fb312a03 100644 --- a/components/services/accessors.h +++ b/components/services/accessors.h @@ -13,7 +13,7 @@ #include "driver/i2s.h" #include "driver/spi_master.h" #include "gpio_exp.h" - +#include "cJSON.h" extern const char *i2c_name_type; extern const char *spi_name_type; diff --git a/components/spotify/cspot/bell/external/nanopb/generator/nanopb_generator.py b/components/spotify/cspot/bell/external/nanopb/generator/nanopb_generator.py old mode 100644 new mode 100755 diff --git a/components/spotify/cspot/bell/external/nanopb/generator/nanopb_generator.py2 b/components/spotify/cspot/bell/external/nanopb/generator/nanopb_generator.py2 old mode 100644 new mode 100755 diff --git a/components/spotify/cspot/bell/external/nanopb/generator/protoc b/components/spotify/cspot/bell/external/nanopb/generator/protoc old mode 100644 new mode 100755 diff --git a/components/spotify/cspot/bell/external/nanopb/generator/protoc-gen-nanopb b/components/spotify/cspot/bell/external/nanopb/generator/protoc-gen-nanopb old mode 100644 new mode 100755 diff --git a/components/spotify/cspot/include/TrackPlayer.h b/components/spotify/cspot/include/TrackPlayer.h index 31f989397..1577c4767 100644 --- a/components/spotify/cspot/include/TrackPlayer.h +++ b/components/spotify/cspot/include/TrackPlayer.h @@ -37,7 +37,6 @@ class TrackPlayer : bell::Task { typedef std::function DataCallback; typedef std::function EOFCallback; - typedef std::function isAiringCallback; TrackPlayer(std::shared_ptr ctx, std::shared_ptr trackQueue, diff --git a/components/spotify/cspot/include/TrackQueue.h b/components/spotify/cspot/include/TrackQueue.h index 174d51af1..60c1d3185 100644 --- a/components/spotify/cspot/include/TrackQueue.h +++ b/components/spotify/cspot/include/TrackQueue.h @@ -54,6 +54,7 @@ class QueuedTrack { uint32_t requestedPosition; std::string identifier; + bool loading = false; // Will return nullptr if the track is not ready std::shared_ptr getAudioFile(); @@ -100,7 +101,7 @@ class TrackQueue : public bell::Task { bool hasTracks(); bool isFinished(); bool skipTrack(SkipDirection dir, bool expectNotify = true); - void updateTracks(uint32_t requestedPosition = 0, bool initial = false); + bool updateTracks(uint32_t requestedPosition = 0, bool initial = false); TrackInfo getTrackInfo(std::string_view identifier); std::shared_ptr consumeTrack( std::shared_ptr prevSong, int& offset); diff --git a/components/spotify/cspot/src/SpircHandler.cpp b/components/spotify/cspot/src/SpircHandler.cpp index 8f225b76e..92531a641 100644 --- a/components/spotify/cspot/src/SpircHandler.cpp +++ b/components/spotify/cspot/src/SpircHandler.cpp @@ -205,16 +205,18 @@ void SpircHandler::handleFrame(std::vector& data) { playbackState->syncWithRemote(); // 1st track is the current one, but update the position - trackQueue->updateTracks( + bool cleared = trackQueue->updateTracks( playbackState->remoteFrame.state.position_ms + - ctx->timeProvider->getSyncedTimestamp() - - playbackState->innerFrame.state.position_measured_at, - false); + ctx->timeProvider->getSyncedTimestamp() - + playbackState->innerFrame.state.position_measured_at); this->notify(); - sendEvent(EventType::FLUSH); - trackPlayer->resetState(); + // need to re-load all if streaming track is completed + if (cleared) { + sendEvent(EventType::FLUSH); + trackPlayer->resetState(); + } break; } case MessageType_kMessageTypeShuffle: { diff --git a/components/spotify/cspot/src/TrackPlayer.cpp b/components/spotify/cspot/src/TrackPlayer.cpp index a252b3cbf..c96f2ec7d 100644 --- a/components/spotify/cspot/src/TrackPlayer.cpp +++ b/components/spotify/cspot/src/TrackPlayer.cpp @@ -201,6 +201,7 @@ void TrackPlayer::runTask() { } eof = false; + track->loading = true; CSPOT_LOG(info, "Playing"); @@ -255,6 +256,7 @@ void TrackPlayer::runTask() { // always move back to LOADING (ensure proper seeking after last track has been loaded) currentTrackStream = nullptr; + track->loading = false; } if (eof) { diff --git a/components/spotify/cspot/src/TrackQueue.cpp b/components/spotify/cspot/src/TrackQueue.cpp index b859539f9..5c416bbe3 100644 --- a/components/spotify/cspot/src/TrackQueue.cpp +++ b/components/spotify/cspot/src/TrackQueue.cpp @@ -587,8 +587,9 @@ bool TrackQueue::isFinished() { return currentTracksIndex >= currentTracks.size() - 1; } -void TrackQueue::updateTracks(uint32_t requestedPosition, bool initial) { +bool TrackQueue::updateTracks(uint32_t requestedPosition, bool initial) { std::scoped_lock lock(tracksMutex); + bool cleared = true; if (initial) { // Clear preloaded tracks @@ -609,13 +610,28 @@ void TrackQueue::updateTracks(uint32_t requestedPosition, bool initial) { playableSemaphore->give(); } else { - // Clear preloaded tracks - preloadedTracks.clear(); - // Copy requested track list currentTracks = playbackState->remoteTracks; - // Push a song on the preloaded queue - queueNextTrack(0, requestedPosition); + // try to not re-load track if we are still loading it + if (preloadedTracks[0]->loading) { + // remove everything except first track + preloadedTracks.erase(preloadedTracks.begin() + 1, preloadedTracks.end()); + + // Push a song on the preloaded queue + CSPOT_LOG(info, "Keeping current track"); + queueNextTrack(1); + + cleared = false; + } else { + // Clear preloaded tracks + preloadedTracks.clear(); + + // Push a song on the preloaded queue + CSPOT_LOG(info, "Re-loading current track"); + queueNextTrack(0, requestedPosition); + } } + + return cleared; } diff --git a/components/squeezelite/displayer.c b/components/squeezelite/displayer.c index 3d07af517..99940cc8e 100644 --- a/components/squeezelite/displayer.c +++ b/components/squeezelite/displayer.c @@ -1295,7 +1295,7 @@ static void ledv_handler( u8_t *data, int len) { LOG_INFO("LED Visualizer mode %u with bars:%u max:%u style:%d gain:%u", led_visu.mode, led_visu.n, led_visu.max, led_visu.style, led_visu.gain); } else { - led_vu_clear(); + led_vu_clear(); LOG_INFO("Stopping led visualizer"); } diff --git a/components/squeezelite/external/dac_external.c b/components/squeezelite/external/dac_external.c index d9eb81896..08f182340 100644 --- a/components/squeezelite/external/dac_external.c +++ b/components/squeezelite/external/dac_external.c @@ -61,7 +61,6 @@ static bool init(char *config, int i2c_port_num, i2s_config_t *i2s_config, bool char *p; i2c_addr = adac_init(config, i2c_port_num); - if (!i2c_addr) return true; ESP_LOGI(TAG, "DAC on I2C @%d", i2c_addr); @@ -137,6 +136,7 @@ bool i2c_json_execute(char *set) { if ((action = cJSON_GetObjectItemCaseSensitive(item, "gpio")) != NULL) { cJSON *level = cJSON_GetObjectItemCaseSensitive(item, "level"); ESP_LOGI(TAG, "set GPIO %d at %d", action->valueint, level->valueint); + if (action->valueint < GPIO_NUM_MAX) gpio_pad_select_gpio(action->valueint); gpio_set_direction_x(action->valueint, GPIO_MODE_OUTPUT); gpio_set_level_x(action->valueint, level->valueint); continue; diff --git a/components/squeezelite/opus.c b/components/squeezelite/opus.c index ec13d62d8..0086c3f0e 100644 --- a/components/squeezelite/opus.c +++ b/components/squeezelite/opus.c @@ -186,7 +186,7 @@ static int read_opus_header(void) { switch (u->status) { case OGG_SYNC: u->status = OGG_ID_HEADER; - OG(&gu, stream_init, &u->state, OG(&gu, page_serialno, &u->page)); + OG(&gu, stream_reset_serialno, &u->state, OG(&gu, page_serialno, &u->page)); fetch = false; break; case OGG_ID_HEADER: @@ -359,10 +359,10 @@ static void opus_open(u8_t size, u8_t rate, u8_t chan, u8_t endianness) { u->status = OGG_SYNC; u->overframes = 0; - - OG(&gu, sync_clear, &u->sync); - OG(&gu, stream_clear, &u->state); - OG(&gu, stream_init, &u->state, -1); + + OG(&go, stream_clear, &u->state); + OG(&go, sync_clear, &u->sync); + OG(&go, stream_init, &u->state, -1); } static void opus_close(void) { @@ -372,8 +372,8 @@ static void opus_close(void) { free(u->overbuf); u->overbuf = NULL; - OG(&gu, stream_clear, &u->state); - OG(&gu, sync_clear, &u->sync); + OG(&go, stream_clear, &u->state); + OG(&go, sync_clear, &u->sync); } static bool load_opus(void) { @@ -394,7 +394,7 @@ static bool load_opus(void) { } g_handle->ogg_stream_clear = dlsym(g_handle->handle, "ogg_stream_clear"); - g_handle->.ogg_stream_reset = dlsym(g_handle->handle, "ogg_stream_reset"); + g_handle->ogg_stream_reset = dlsym(g_handle->handle, "ogg_stream_reset"); g_handle->ogg_stream_eos = dlsym(g_handle->handle, "ogg_stream_eos"); g_handle->ogg_stream_reset_serialno = dlsym(g_handle->handle, "ogg_stream_reset_serialno"); g_handle->ogg_sync_clear = dlsym(g_handle->handle, "ogg_sync_clear"); diff --git a/components/squeezelite/slimproto.c b/components/squeezelite/slimproto.c index 5cbc78b02..f053684ed 100644 --- a/components/squeezelite/slimproto.c +++ b/components/squeezelite/slimproto.c @@ -984,8 +984,8 @@ void slimproto(log_level level, char *server, u8_t mac[6], const char *name, con // in embedded we give up after a while no matter what if (++failed_connect > MAX_SERVER_RETRIES && !server) { slimproto_ip = serv_addr.sin_addr.s_addr = discover_server(NULL, MAX_SERVER_RETRIES); - if (!slimproto_ip) return; - } else if (reconnect && MAX_SERVER_RETRIES && failed_connect > 5 * MAX_SERVER_RETRIES) return; + if (!slimproto_ip && !output.external) return; + } else if (reconnect && MAX_SERVER_RETRIES && failed_connect > 5 * MAX_SERVER_RETRIES && !output.external) return; #else // rediscover server if it was not set at startup or exit if (!server && ++failed_connect > 5) { diff --git a/components/squeezelite/vorbis.c b/components/squeezelite/vorbis.c index 5c1ebeb81..1a3f6e80f 100644 --- a/components/squeezelite/vorbis.c +++ b/components/squeezelite/vorbis.c @@ -72,6 +72,13 @@ static struct vorbis { // vorbis symbols to be dynamically loaded - from either vorbisfile or vorbisidec (tremor) version of library vorbis_info *(* ov_info)(OggVorbis_File *vf, int link); int (* ov_clear)(OggVorbis_File *vf); + void (* ov_info_init)(vorbis_info* vi); + void (* ov_info_clear)(vorbis_info* vi); + void (* ov_comment_init)(vorbis_comment* vc); + void (* ov_comment_clear(vorbis_comment *vc); + int (* ov_block_init)(vorbis_dsp_state* v, vorbis_block* vb); + int (* ov_block_clear)(vorbis_block* vb); + void (* ov_vorbis_dsp_clear)(vorbis_dsp_state* v); long (* ov_read)(OggVorbis_File *vf, char *buffer, int length, int bigendianp, int word, int sgned, int *bitstream); long (* ov_read_tremor)(OggVorbis_File *vf, char *buffer, int length, int *bitstream); int (* ov_open_callbacks)(void *datasource, OggVorbis_File *vf, const char *initial, long ibytes, ov_callbacks callbacks); @@ -396,33 +403,32 @@ static decode_state vorbis_decode(void) { } static void vorbis_open(u8_t size, u8_t rate, u8_t chan, u8_t endianness) { - LOG_INFO("OPENING CODEC"); if (v->opened) { - OV(&go, block_clear, &v->block); - OV(&go, info_clear, &v->info); - OV(&go, dsp_clear, &v->decoder); + OV(&gv, block_clear, &v->block); + OV(&gv, dsp_clear, &v->decoder); + OV(&gv, info_clear, &v->info); } v->opened = false; v->status = OGG_SYNC; v->overflow = 0; - - OG(&gu, sync_clear, &v->sync); - OG(&gu, stream_clear, &v->state); - OG(&gu, stream_init, &v->state, -1); + + OG(&go, stream_clear, &v->state); + OG(&go, sync_clear, &v->sync); + OG(&go, stream_init, &v->state, -1); } static void vorbis_close() { - return; - LOG_INFO("CLOSING CODEC"); if (v->opened) { - OV(&go, block_clear, &v->block); - OV(&go, info_clear, &v->info); - OV(&go, dsp_clear, &v->decoder); + OV(&gv, block_clear, &v->block); + OV(&gv, dsp_clear, &v->decoder); + // info must be last otherwise there is memory leak (where is it said... nowhere) + OV(&gv, info_clear, &v->info); + // we don' t have comments to free } v->opened = false; - + OG(&go, stream_clear, &v->state); OG(&go, sync_clear, &v->sync); } @@ -469,6 +475,11 @@ static bool load_vorbis() { v_handle.ov_read = dlsym(handle, "ov_read"); v_handle.ov_info = dlsym(handle, "ov_info"); v_handle.ov_clear = dlsym(handle, "ov_clear"); + v.handle.ov_info_clear = dlsym(gv.handle, "vorbis_info_clear"); + v.handle.ov_comment_init = dlsym(gv.handle, "vorbis_comment_init"); + v.handle.ov_comment_clear = dlsym(gv.handle, "vorbis_comment_clear"); + v.handle.ov_block_init = dlsym(gv.handle, "vorbis_block_init"); + v.handle.ov_block_clear = dlsym(gv.handle, "vorbis_block_clear"); v_handle.ov_open_callbacks = dlsym(handle, "ov_open_callbacks"); if ((err = dlerror()) != NULL) { diff --git a/components/wifi-manager/webapp/src/bootswatch b/components/wifi-manager/webapp/src/bootswatch deleted file mode 160000 index 3038872b8..000000000 --- a/components/wifi-manager/webapp/src/bootswatch +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 3038872b8389763b64857bac00e102a71251a331 diff --git a/docker/ToggleGitTracking.ps1 b/docker/ToggleGitTracking.ps1 new file mode 100644 index 000000000..73c5c9b73 --- /dev/null +++ b/docker/ToggleGitTracking.ps1 @@ -0,0 +1,91 @@ +param ( + [Parameter(Position=0, Mandatory=$false)] + [ValidateSet("t", "u", "d")] + [string]$option +) + +# Get the current directory +$currentDir = Get-Location + +# Define target directories +$targetDir = "components\wifi-manager\webapp" +$distDir = "$targetDir\dist" + +# Get list of files from the 'dist' directory +$fsFiles = Get-ChildItem -Recurse $distDir -File | ForEach-Object { + $_.FullName.Substring($currentDir.Path.Length + 1).Replace("\", "/") +} + +# Define additional files to include +$additionalFiles = @("webpack.c", "webpack.h", "webapp.cmake") + +# Check if additional files exist in $targetDir and format them +$additionalFilesFormatted = @() +Get-ChildItem $targetDir -File | ForEach-Object { + if ($additionalFiles -contains $_.Name) { + $formatted = $_.FullName.Substring($currentDir.Path.Length + 1).Replace("\", "/") + $additionalFilesFormatted += $formatted + Write-Host "Found $formatted" + } +} + +# Get list of files from the Git index +$indexFiles = git ls-files -s $distDir | ForEach-Object { + ($_ -split "\s+")[3] +} + +# Combine and remove duplicates +$allFiles = $fsFiles + $additionalFilesFormatted + $indexFiles | Sort-Object -Unique +# ... (previous code remains unchanged) + +# Apply the git command based on the option +$allFiles | ForEach-Object { + $relativePath = $_ + $isInIndex = $indexFiles -contains $relativePath + + if ($null -eq $option) { + $gitStatus = & git status --porcelain -- $relativePath + if ($gitStatus) { + $status = ($gitStatus -split "\s")[0] + Write-Host "$relativePath has Git status: $status" + } else { + Write-Host "$relativePath is not tracked" + } + } + elseif ($isInIndex) { + if ($option -eq "d") { + $resetResult = & git reset -- $relativePath 2>&1 + if ($resetResult -match 'error:') { + Write-Host "Error resetting ${relativePath}: $resetResult" + + continue + } + $checkoutResult = & git checkout -- $relativePath 2>&1 + if ($checkoutResult -match 'error:') { + Write-Host "Error checking out ${relativePath}: $checkoutResult" + + continue + } + Write-Host "Discarded changes in $relativePath" + } + # ... (rest of the code remains unchanged) + } + # else { + # # if ($option -eq "d") { + # # Remove-Item -Path $relativePath -Force + # # Write-Host "Removed untracked file $relativePath" + # # } else { + # # Write-Host "File $relativePath is not tracked." + # # } + + # } + else { + if ($option -eq "t") { + git add $relativePath + git update-index --no-skip-worktree $relativePath + Write-Host "Started tracking changes in $relativePath" + } else { + Write-Host "File $relativePath is not tracked." + } + } +} diff --git a/main/esp_app_main.c b/main/esp_app_main.c index 0b805169b..7f3d6a408 100644 --- a/main/esp_app_main.c +++ b/main/esp_app_main.c @@ -297,7 +297,6 @@ void register_default_nvs(){ else { register_default_string_val("cspot_config", ""); } - } #endif @@ -309,7 +308,7 @@ void register_default_nvs(){ #endif register_default_with_mac("host_name", DEFAULT_HOST_NAME); register_default_with_mac("ap_ssid", CONFIG_DEFAULT_AP_SSID); -for (int i = 0; i < sizeof(defaultStringVals) / sizeof(DefaultStringVal); ++i) { + for (int i = 0; i < sizeof(defaultStringVals) / sizeof(DefaultStringVal); ++i) { register_default_string_val(defaultStringVals[i].key, defaultStringVals[i].value); } for (int i = 0; i < sizeof(defaultNumVals) / sizeof(DefaultNumVal); ++i) { diff --git a/plugin/SqueezeESP32.zip b/plugin/SqueezeESP32.zip index 45851b1b3..155aea376 100644 Binary files a/plugin/SqueezeESP32.zip and b/plugin/SqueezeESP32.zip differ diff --git a/plugin/SqueezeESP32/install.xml b/plugin/SqueezeESP32/install.xml index 278a50b34..ce8c682c1 100644 --- a/plugin/SqueezeESP32/install.xml +++ b/plugin/SqueezeESP32/install.xml @@ -10,6 +10,6 @@ PLUGIN_SQUEEZEESP32 PLUGIN_SQUEEZEESP32_DESC Plugins::SqueezeESP32::Plugin - 0.501 + 0.600 Philippe diff --git a/plugin/repo.xml b/plugin/repo.xml index 132d4a8ca..df5527bef 100644 --- a/plugin/repo.xml +++ b/plugin/repo.xml @@ -1,10 +1,10 @@ - + https://github.com/sle118/squeezelite-esp32 Philippe - 842ee0f7b8ccaf2e6df4a741565068f9e4c7a27e + 335b585e22eddbddacce00f34001d8bf2ee0f54f philippe_44@outlook.com SqueezeESP32 additional player id (100/101) http://raw.githubusercontent.com/sle118/squeezelite-esp32/master-v4.3/plugin/SqueezeESP32.zip