From ed6d0a253a08c12301c146a375a3776feab2838c Mon Sep 17 00:00:00 2001 From: Mit4el Date: Tue, 20 Jun 2023 00:51:56 +0300 Subject: [PATCH] =?UTF-8?q?=D0=9C=D0=BE=D0=B4=D1=83=D0=BB=D1=8C=20FTP?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- lib/SimpleFTPServer/.project | 11 + lib/SimpleFTPServer/FtpServer.cpp | 2354 +++++++++++++++++ lib/SimpleFTPServer/FtpServer.h | 758 ++++++ lib/SimpleFTPServer/FtpServerKey.h | 128 + lib/SimpleFTPServer/LICENSE.md | 24 + lib/SimpleFTPServer/README.md | 108 + lib/SimpleFTPServer/SimpleFTPServer.h | 18 + .../Arduino_Ethernet/Arduino_Ethernet.ino | 62 + .../Arduino_Ethernet_SdFat2.ino | 133 + .../Arduino_esp32_SD/Arduino_esp32_SD.ino | 96 + .../ESP32_AP_FFAT_WiFi/ESP32_AP_FFAT_WiFi.ino | 124 + .../ESP32_FFAT_WiFi/ESP32_FFAT_WiFi.ino | 105 + .../ESP32_FFAT_enc28j60.ino | 130 + .../ESP32_SPIFM_WiFi/ESP32_SPIFM_WiFi.ino | 148 ++ .../RaspberryPiPicoW_LittleFS_WiFi.ino | 103 + .../STM32_SPIFlash_enc28j60.ino | 136 + .../STM32_SdFat_enc28j60.ino | 135 + .../STM32_SdFat_w5500/STM32_SdFat_w5500.ino | 128 + .../examples/Wio_terminal/Wio_terminal.ino | 106 + .../Wio_terminal_SdFat/Wio_terminal_SdFat.ino | 104 + .../Wio_terminal_SdFat_TFT_monitor.ino | 367 +++ .../esp8266_esp32_LittleFS.ino | 110 + .../esp8266_esp32_SPIFFS.ino | 110 + lib/SimpleFTPServer/keywords.txt | 17 + lib/SimpleFTPServer/library.json | 21 + lib/SimpleFTPServer/library.properties | 11 + myProfile.json | 10 +- src/modules/exec/Ftp/Ftp.cpp | 97 + src/modules/exec/Ftp/modinfo.json | 68 + src/modules/sensors/Scd40/Scd40.cpp | 40 +- 30 files changed, 5739 insertions(+), 23 deletions(-) create mode 100644 lib/SimpleFTPServer/.project create mode 100644 lib/SimpleFTPServer/FtpServer.cpp create mode 100644 lib/SimpleFTPServer/FtpServer.h create mode 100644 lib/SimpleFTPServer/FtpServerKey.h create mode 100644 lib/SimpleFTPServer/LICENSE.md create mode 100644 lib/SimpleFTPServer/README.md create mode 100644 lib/SimpleFTPServer/SimpleFTPServer.h create mode 100644 lib/SimpleFTPServer/examples/Arduino_Ethernet/Arduino_Ethernet.ino create mode 100644 lib/SimpleFTPServer/examples/Arduino_Ethernet_SdFat2/Arduino_Ethernet_SdFat2.ino create mode 100644 lib/SimpleFTPServer/examples/Arduino_esp32_SD/Arduino_esp32_SD.ino create mode 100644 lib/SimpleFTPServer/examples/ESP32_AP_FFAT_WiFi/ESP32_AP_FFAT_WiFi.ino create mode 100644 lib/SimpleFTPServer/examples/ESP32_FFAT_WiFi/ESP32_FFAT_WiFi.ino create mode 100644 lib/SimpleFTPServer/examples/ESP32_FFAT_enc28j60/ESP32_FFAT_enc28j60.ino create mode 100644 lib/SimpleFTPServer/examples/ESP32_SPIFM_WiFi/ESP32_SPIFM_WiFi.ino create mode 100644 lib/SimpleFTPServer/examples/RaspberryPiPicoW_LittleFS_WiFi/RaspberryPiPicoW_LittleFS_WiFi.ino create mode 100644 lib/SimpleFTPServer/examples/STM32_SPIFlash_enc28j60/STM32_SPIFlash_enc28j60.ino create mode 100644 lib/SimpleFTPServer/examples/STM32_SdFat_enc28j60/STM32_SdFat_enc28j60.ino create mode 100644 lib/SimpleFTPServer/examples/STM32_SdFat_w5500/STM32_SdFat_w5500.ino create mode 100644 lib/SimpleFTPServer/examples/Wio_terminal/Wio_terminal.ino create mode 100644 lib/SimpleFTPServer/examples/Wio_terminal_SdFat/Wio_terminal_SdFat.ino create mode 100644 lib/SimpleFTPServer/examples/Wio_terminal_SdFat_TFT_monitor/Wio_terminal_SdFat_TFT_monitor.ino create mode 100644 lib/SimpleFTPServer/examples/esp8266_esp32_LittleFS/esp8266_esp32_LittleFS.ino create mode 100644 lib/SimpleFTPServer/examples/esp8266_esp32_SPIFFS/esp8266_esp32_SPIFFS.ino create mode 100644 lib/SimpleFTPServer/keywords.txt create mode 100644 lib/SimpleFTPServer/library.json create mode 100644 lib/SimpleFTPServer/library.properties create mode 100644 src/modules/exec/Ftp/Ftp.cpp create mode 100644 src/modules/exec/Ftp/modinfo.json diff --git a/lib/SimpleFTPServer/.project b/lib/SimpleFTPServer/.project new file mode 100644 index 00000000..aa361b16 --- /dev/null +++ b/lib/SimpleFTPServer/.project @@ -0,0 +1,11 @@ + + + SimpleFTPServer + + + + + + + + diff --git a/lib/SimpleFTPServer/FtpServer.cpp b/lib/SimpleFTPServer/FtpServer.cpp new file mode 100644 index 00000000..8ec4a184 --- /dev/null +++ b/lib/SimpleFTPServer/FtpServer.cpp @@ -0,0 +1,2354 @@ +/* + * FtpServer Arduino, esp8266 and esp32 library for Ftp Server + * Derived form Jean-Michel Gallego version + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + * + * Use Ethernet library + * + * Commands implemented: + * USER, PASS, AUTH (AUTH only return 'not implemented' code) + * CDUP, CWD, PWD, QUIT, NOOP + * MODE, PASV, PORT, STRU, TYPE + * ABOR, DELE, LIST, NLST, MLST, MLSD + * APPE, RETR, STOR + * MKD, RMD + * RNTO, RNFR + * MDTM, MFMT + * FEAT, SIZE + * SITE FREE + * HELP + * + * Tested with those clients: + * under Windows: + * FTP Rush + * Filezilla + * WinSCP + * NcFTP, ncftpget, ncftpput + * Firefox + * command line ftp.exe + * under Ubuntu: + * gFTP + * Filezilla + * NcFTP, ncftpget, ncftpput + * lftp + * ftp + * Firefox + * under Android: + * AndFTP + * FTP Express + * Firefox + * with a second Arduino and sketch of SurferTim at + * http://playground.arduino.cc/Code/FTP + * + */ + +#include + +FtpServer::FtpServer( uint16_t _cmdPort, uint16_t _pasvPort ) + : ftpServer( _cmdPort ), dataServer( _pasvPort ) +{ + cmdPort = _cmdPort; + pasvPort = _pasvPort; + + millisDelay = 0; + nbMatch = 0; + iCL = 0; + + iniVariables(); +} + +void FtpServer::begin( const char * _user, const char * _pass, const char * _welcomeMessage ) +{ + if ( strcmp( _user, "anonymous" ) != 0) { + DEBUG_PRINTLN(F("NOT ANONYMOUS")); + DEBUG_PRINTLN(_user); + this->anonymousConnection = false; // needed to reset after end of anonymnous and begin of not anonymous + } + // Tells the ftp server to begin listening for incoming connection + ftpServer.begin(); + #if (defined(ESP8266) && (FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_ASYNC || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266 || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_242)) || defined(ARDUINO_ARCH_RP2040) || FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_SEEED_RTL8720DN + ftpServer.setNoDelay( true ); + #endif +// localIp = _localIP == FTP_NULLIP() || (uint32_t) _localIP == 0 ? NET_CLASS.localIP() : _localIP ; + localIp = NET_CLASS.localIP(); //_localIP == FTP_NULLIP() || (uint32_t) _localIP == 0 ? NET_CLASS.localIP() : _localIP ; +// strcpy( user, FTP_USER ); +// strcpy( pass, FTP_PASS ); + if( strlen( _user ) > 0 && strlen( _user ) < FTP_CRED_SIZE ) { + //strcpy( user, _user ); + this->user = _user; + } + if( strlen( _pass ) > 0 && strlen( _pass ) < FTP_CRED_SIZE ) { +// strcpy( pass, _pass ); + this->pass = _pass; + } +// strcpy(_welcomeMessage, welcomeMessage); + + this->welcomeMessage = _welcomeMessage; + + dataServer.begin(); +#if (defined(ESP8266) && (FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_ASYNC || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266 || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_242)) || defined(ARDUINO_ARCH_RP2040) || FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_SEEED_RTL8720DN + dataServer.setNoDelay( true ); +#endif + + millisDelay = 0; + cmdStage = FTP_Stop; + iniVariables(); +} + +void FtpServer::begin( const char * _welcomeMessage ) { + this->anonymousConnection = true; + this->begin( "anonymous", "anonymous", _welcomeMessage); +} + +void FtpServer::end() +{ + if(client.connected()) { + disconnectClient(); + } + +#if FTP_SERVER_NETWORK_TYPE == NETWORK_ESP32 && !defined(ARDUINO_ARCH_RP2040) + ftpServer.end(); + dataServer.end(); +#endif + + DEBUG_PRINTLN(F("Stop server!")); + + if (FtpServer::_callback) { + FtpServer::_callback(FTP_DISCONNECT, free(), capacity()); + } + + cmdStage = FTP_Init; + transferStage = FTP_Close; + dataConn = FTP_NoConn; +} +void FtpServer::setLocalIp(IPAddress localIp) +{ + this->localIp = localIp; +} +void FtpServer::credentials( const char * _user, const char * _pass ) +{ + if( strlen( _user ) > 0 && strlen( _user ) < FTP_CRED_SIZE ) +// strcpy( user, _user ); + this->user = user; + if( strlen( _pass ) > 0 && strlen( _pass ) < FTP_CRED_SIZE ) +// strcpy( pass, _pass ); + this->pass = _pass; +} + +void FtpServer::iniVariables() +{ + // Default for data port + dataPort = FTP_DATA_PORT_DFLT; + + // Default Data connection is Active + dataConn = FTP_NoConn; + + // Set the root directory + strcpy( cwdName, "/" ); + + rnfrCmd = false; + transferStage = FTP_Close; +} + +uint8_t FtpServer::handleFTP() { +#ifdef FTP_ADDITIONAL_DEBUG +// int8_t data0 = data.status(); + ftpTransfer transferStage0 = transferStage; + ftpCmd cmdStage0 = cmdStage; + ftpDataConn dataConn0 = dataConn; +#endif + + if ((int32_t) (millisDelay - millis()) <= 0) { + if (cmdStage == FTP_Stop) { + if (client.connected()) { + DEBUG_PRINTLN(F("Disconnect client!")); + disconnectClient(); + } + cmdStage = FTP_Init; + } else if (cmdStage == FTP_Init) { // Ftp server waiting for connection + abortTransfer(); + iniVariables(); + DEBUG_PRINT(F(" Ftp server waiting for connection on port ")); + DEBUG_PRINTLN(cmdPort); + + cmdStage = FTP_Client; + } else if (cmdStage == FTP_Client) { // Ftp server idle +#if (FTP_SERVER_NETWORK_TYPE == NETWORK_WiFiNINA) +// if (client && !client.connected()) { +// client.stop(); +// DEBUG_PRINTLN(F("CLIENT STOP!!")); +// } + byte status; + client = ftpServer.available(&status); + /* + * CLOSED = 0, + LISTEN = 1, + SYN_SENT = 2, + SYN_RCVD = 3, + ESTABLISHED = 4, + FIN_WAIT_1 = 5, + FIN_WAIT_2 = 6, + CLOSE_WAIT = 7, + CLOSING = 8, + LAST_ACK = 9, + TIME_WAIT = 10 + * + */ +// DEBUG_PRINTLN(status); +#elif (defined(ESP8266) && (FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_ASYNC || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266 || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_242)) + + if( ftpServer.hasClient()) + { + client.stop(); + client = ftpServer.available(); + } +#else + if (client && !client.connected()) { + client.stop(); + DEBUG_PRINTLN(F("CLIENT STOP!!")); + } + client = ftpServer.accept(); +#endif + if (client.connected()) // A client connected + { + clientConnected(); + millisEndConnection = millis() + 1000L * FTP_AUTH_TIME_OUT; // wait client id for 10 s. + cmdStage = FTP_User; + } + } else if (readChar() > 0) // got response + { + processCommand(); + if (cmdStage == FTP_Stop) + millisEndConnection = millis() + 1000L * FTP_AUTH_TIME_OUT; // wait authentication for 10 s. + else if (cmdStage < FTP_Cmd) + millisDelay = millis() + 200; // delay of 100 ms + else + millisEndConnection = millis() + 1000L * FTP_TIME_OUT; + } else if (!client.connected()) { + if (FtpServer::_callback) { + FtpServer::_callback(FTP_DISCONNECT, free(), capacity()); + } + + cmdStage = FTP_Init; + } + if (transferStage == FTP_Retrieve) // Retrieve data + { + if (!doRetrieve()) { + transferStage = FTP_Close; + } + } else if (transferStage == FTP_Store) // Store data + { + if (!doStore()) { + if (FtpServer::_callback) { + FtpServer::_callback(FTP_FREE_SPACE_CHANGE, free(), capacity()); + } + + transferStage = FTP_Close; + } + } else if (transferStage == FTP_List || transferStage == FTP_Nlst) // LIST or NLST + { + if (!doList()) { + transferStage = FTP_Close; + } + } else if (transferStage == FTP_Mlsd) // MLSD listing + { + if (!doMlsd()) { + + transferStage = FTP_Close; + } + } else if (cmdStage > FTP_Client + && !((int32_t) (millisEndConnection - millis()) > 0)) { + client.println(F("530 Timeout")); + millisDelay = millis() + 200; // delay of 200 ms + cmdStage = FTP_Stop; + } + +#ifdef FTP_ADDITIONAL_DEBUG + if (cmdStage != cmdStage0 || transferStage != transferStage0 + || dataConn != dataConn0) { + DEBUG_PRINT(F(" Command Old: ")); + DEBUG_PRINT(cmdStage0); + DEBUG_PRINT(F(" Transfer Old: ")); + DEBUG_PRINT(transferStage0); + DEBUG_PRINT(F(" Data Old: ")); + DEBUG_PRINTLN(dataConn0); + + DEBUG_PRINT(F(" Command : ")); + DEBUG_PRINT(cmdStage); + DEBUG_PRINT(F(" Transfer : ")); + DEBUG_PRINT(transferStage); + DEBUG_PRINT(F(" Data : ")); + DEBUG_PRINTLN(dataConn); + } +#endif + } + return cmdStage | (transferStage << 3) | (dataConn << 6); +} + +void FtpServer::clientConnected() +{ + DEBUG_PRINTLN( F(" Client connected!") ); + client.print(F("220---")); client.print(welcomeMessage); client.println(F(" ---")); + client.println(F("220--- By Renzo Mischianti ---")); + client.print(F("220 -- Version ")); client.print(FTP_SERVER_VERSION); client.println(F(" --")); + iCL = 0; + if (FtpServer::_callback) { + FtpServer::_callback(FTP_CONNECT, free(), capacity()); + } + +} + +void FtpServer::disconnectClient() +{ + DEBUG_PRINTLN( F(" Disconnecting client") ); + + abortTransfer(); + client.println(F("221 Goodbye") ); + + if (FtpServer::_callback) { + FtpServer::_callback(FTP_DISCONNECT, free(), capacity()); + } + + if( client ) { + } + if( data ) { + data.stop(); + } +} + +bool FtpServer::processCommand() +{ + /////////////////////////////////////// + // // + // AUTHENTICATION COMMANDS // + // // + /////////////////////////////////////// + + // RoSchmi added the next two lines + DEBUG_PRINT("Command is: "); + DEBUG_PRINTLN(command); + + // + // USER - User Identity + // + if( CommandIs( "USER" )) + { + DEBUG_PRINT(F("USER: ")); + DEBUG_PRINT(parameter); + DEBUG_PRINT(F(" ")); + DEBUG_PRINTLN(user) + + if (this->anonymousConnection && ! strcmp( parameter, user )) { + DEBUG_PRINTLN( F(" Anonymous authentication Ok. Waiting for commands.") ); + + client.println(F("230 Ok") ); + cmdStage = FTP_Cmd; + } else if( ! strcmp( parameter, user )) + { + client.println(F("331 Ok. Password required") ); + strcpy( cwdName, "/" ); + cmdStage = FTP_Pass; + } + else + { + client.println(F("530 ") ); + cmdStage = FTP_Stop; + } + } + // + // PASS - Password + // + else if( CommandIs( "PASS" )) + { + if( cmdStage != FTP_Pass ) + { + client.println(F("503 ") ); + cmdStage = FTP_Stop; + } + if( ! strcmp( parameter, pass )) + { + DEBUG_PRINTLN( F(" Authentication Ok. Waiting for commands.") ); + + client.println(F("230 Ok") ); + cmdStage = FTP_Cmd; + } + else + { + client.println( F("530 ") ); + cmdStage = FTP_Stop; + } + } + // + // FEAT - New Features + // + else if( CommandIs( "FEAT" )) + { + client.println(F("211-Extensions suported:")); + client.println(F(" MLST type*;modify*;size*;") ); + client.println(F(" MLSD") ); + client.println(F(" MDTM") ); + client.println(F(" MFMT") ); +#ifdef UTF8_SUPPORT + client.println(F(" UTF8") ); +#endif + client.println(F(" SIZE") ); + client.println(F(" SITE FREE") ); + client.println(F("211 End.") ); + } + // + // AUTH - Not implemented + // + else if( CommandIs( "AUTH" )) + client.println(F("502 ") ); + // + // Unrecognized commands at stage of authentication + // + else if( cmdStage < FTP_Cmd ) + { + client.println(F("530 ") ); + cmdStage = FTP_Stop; + } + + /////////////////////////////////////// + // // + // ACCESS CONTROL COMMANDS // + // // + /////////////////////////////////////// + + // + // PWD - Print Directory + // + else if( CommandIs( "PWD" ) || + ( CommandIs( "CWD" ) && ParameterIs( "." ))) { + client.print( F("257 \"")); client.print( cwdName ); client.print( F("\"") ); client.println( F(" is your current directory") ); + // + // CDUP - Change to Parent Directory + // + } else if( CommandIs( "CDUP" ) || + ( CommandIs( "CWD" ) && ParameterIs( ".." ))) + { + bool ok = false; + + if( strlen( cwdName ) > 1 ) // do nothing if cwdName is root + { + // if cwdName ends with '/', remove it (must not append) + if( cwdName[ strlen( cwdName ) - 1 ] == '/' ) { + cwdName[ strlen( cwdName ) - 1 ] = 0; + } + // search last '/' + char * pSep = strrchr( cwdName, '/' ); + ok = pSep > cwdName; + // if found, ends the string on its position + if( ok ) + { + * pSep = 0; + ok = exists( cwdName ); + } + } + // if an error appends, move to root + if( ! ok ) { + strcpy( cwdName, "/" ); + } + client.print( F("250 Ok. Current directory is ") ); client.println( cwdName ); + } + // + // CWD - Change Working Directory + // + else if( CommandIs( "CWD" )) + { + char path[ FTP_CWD_SIZE ]; + if( haveParameter() && makeExistsPath( path )) + { + strcpy( cwdName, path ); + client.print( F("250 Directory changed to ") ); client.print(cwdName); client.println(); + } + } + // + // QUIT + // + else if( CommandIs( "QUIT" )) + { + client.println(F("221 Goodbye") ); + disconnectClient(); + cmdStage = FTP_Stop; + } + + /////////////////////////////////////// + // // + // TRANSFER PARAMETER COMMANDS // + // // + /////////////////////////////////////// + + // + // MODE - Transfer Mode + // + else if( CommandIs( "MODE" )) + { + if( ParameterIs( "S" )) { + client.println(F("200 S Ok") ); + } else { + client.println(F("504 Only S(tream) is suported") ); + } + } + // + // PASV - Passive Connection management + // + else if( CommandIs( "PASV" )) + { + data.stop(); + dataServer.begin(); + if (((((uint32_t) NET_CLASS.localIP()) & ((uint32_t) NET_CLASS.subnetMask())) == + (((uint32_t) client.remoteIP()) & ((uint32_t) NET_CLASS.subnetMask()))) && (uint32_t)localIp <= 0) { + dataIp = NET_CLASS.localIP(); + } else { + dataIp = localIp; + } + DEBUG_PRINT( F(" IP: ") ); + DEBUG_PRINT( int( dataIp[0]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINT( int( dataIp[1]) ); DEBUG_PRINT( F(".") ); + DEBUG_PRINT( int( dataIp[2]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINTLN( int( dataIp[3]) ); + +#if !defined(ARDUINO_ARCH_RP2040) && ((FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_ESP8266_ASYNC) || (FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_ESP8266) || (FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_ESP8266) || (FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_ESP32)) // || (FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_WiFiNINA) || (FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_SEEED_RTL8720DN)) + if (dataIp.toString() == F("0.0.0.0")) { + dataIp = NET_CLASS.softAPIP(); + } +#endif + DEBUG_PRINT( F(" Soft IP: ") ); + DEBUG_PRINT( int( dataIp[0]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINT( int( dataIp[1]) ); DEBUG_PRINT( F(".") ); + DEBUG_PRINT( int( dataIp[2]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINTLN( int( dataIp[3]) ); + + dataPort = pasvPort; + DEBUG_PRINTLN( F(" Connection management set to passive") ); + DEBUG_PRINT( F(" Listening at ") ); + DEBUG_PRINT( int( dataIp[0]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINT( int( dataIp[1]) ); DEBUG_PRINT( F(".") ); + DEBUG_PRINT( int( dataIp[2]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINT( int( dataIp[3]) ); + DEBUG_PRINT( F(":") ); DEBUG_PRINTLN( dataPort ); + + client.print( F("227 Entering Passive Mode") ); client.print( F(" (") ); + client.print( int( dataIp[0]) ); client.print( F(",") ); client.print( int( dataIp[1]) ); client.print( F(",") ); + client.print( int( dataIp[2]) ); client.print( F(",") ); client.print( int( dataIp[3]) ); client.print( F(",") ); + client.print( ( dataPort >> 8 ) ); client.print( F(",") ); client.print( ( dataPort & 255 ) ); client.println( F(")") ); + dataConn = FTP_Pasive; + } + // + // PORT - Data Port + // + else if( CommandIs( "PORT" )) + { + data.stop(); + // get IP of data client + dataIp[ 0 ] = atoi( parameter ); + char * p = strchr( parameter, ',' ); + for( uint8_t i = 1; i < 4; i ++ ) + { + dataIp[ i ] = atoi( ++ p ); + p = strchr( p, ',' ); + } + // get port of data client + dataPort = 256 * atoi( ++ p ); + p = strchr( p, ',' ); + dataPort += atoi( ++ p ); + if( p == NULL ) { + client.println(F("501 Can't interpret parameters") ); + } else + { + DEBUG_PRINT( F(" Data IP set to ") ); DEBUG_PRINT( int( dataIp[0]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINT( int( dataIp[1]) ); + DEBUG_PRINT( F(".") ); DEBUG_PRINT( int( dataIp[2]) ); DEBUG_PRINT( F(".") ); DEBUG_PRINTLN( int( dataIp[3]) ); + DEBUG_PRINT( F(" Data port set to ") ); DEBUG_PRINTLN( dataPort ); + + client.println(F("200 PORT command successful") ); + dataConn = FTP_Active; + } + } + // + // STRU - File Structure + // + else if( CommandIs( "STRU" )) + { + if( ParameterIs( "F" )) { + client.println(F("200 F Ok") ); + // else if( ParameterIs( "R" )) + // client.println(F("200 B Ok") ); + }else{ + client.println(F("504 Only F(ile) is suported") ); + } + } + // + // TYPE - Data Type + // + else if( CommandIs( "TYPE" )) + { + if( ParameterIs( "A" )) { + client.println(F("200 TYPE is now ASCII")); + } else if( ParameterIs( "I" )) { + client.println(F("200 TYPE is now 8-bit binary") ); + } else { + client.println(F("504 Unknow TYPE") ); + } + } + + /////////////////////////////////////// + // // + // FTP SERVICE COMMANDS // + // // + /////////////////////////////////////// + + // + // ABOR - Abort + // + else if( CommandIs( "ABOR" )) + { + abortTransfer(); + client.println(F("226 Data connection closed")); + } + // + // DELE - Delete a File + // + else if( CommandIs( "DELE" )) + { + char path[ FTP_CWD_SIZE ]; + if( haveParameter() && makeExistsPath( path )) { + if( remove( path )) { + if (FtpServer::_callback) { + FtpServer::_callback(FTP_FREE_SPACE_CHANGE, free(), capacity()); + } + + client.print( F("250 Deleted ") ); client.println( parameter ); + } else { + client.print( F("450 Can't delete ") ); client.println( parameter ); + } + } + } + // + // LIST - List + // NLST - Name List + // MLSD - Listing for Machine Processing (see RFC 3659) + // + else if( CommandIs( "LIST" ) || CommandIs( "NLST" ) || CommandIs( "MLSD" )) + { + DEBUG_PRINT("List of file!!"); + + if( dataConnect()){ + if( openDir( & dir )) + { + DEBUG_PRINT("Dir opened!!"); + + nbMatch = 0; + if( CommandIs( "LIST" )) + transferStage = FTP_List; + else if( CommandIs( "NLST" )) + transferStage = FTP_Nlst; + else + transferStage = FTP_Mlsd; + } + else { + DEBUG_PRINT("List Data stop!!"); + data.stop(); + } + } + } + // + // MLST - Listing for Machine Processing (see RFC 3659) + // + else if( CommandIs( "MLST" )) + { + char path[ FTP_CWD_SIZE ]; + uint16_t dat=0, tim=0; + char dtStr[ 15 ]; + bool isdir; + if( haveParameter() && makeExistsPath( path )){ + if( ! getFileModTime( path, &dat, &tim )) { + client.print( F("550 Unable to retrieve time for ") ); client.println( parameter ); + } else + { + isdir = isDir( path ); + client.println( F("250-Begin") ); + client.print( F(" Type=") ); client.print( ( isdir ? F("dir") : F("file")) ); + client.print( F(";Modify=") ); client.print( makeDateTimeStr( dtStr, dat, tim ) ); + if( ! isdir ) + { + if( openFile( path, FTP_FILE_READ )) + { + client.print( F(";Size=") ); client.print( long( fileSize( file )) ); + file.close(); + } + } + client.print( F("; ") ); client.println( path ); + client.println( F("250 End.") ); + } + } + } + // + // NOOP + // + else if( CommandIs( "NOOP" )) { + client.println(F("200 Zzz...") ); + } + // +#ifdef UTF8_SUPPORT + // OPTS + // + else if( CommandIs( "OPTS" )) { + if( ParameterIs( "UTF8 ON" ) || ParameterIs( "utf8 on" )) { + client.println(F("200 OK, UTF8 ON") ); + DEBUG_PRINTLN(F("200 OK, UTF8 ON") ); + } else { + client.println(F("504 Unknow OPTS") ); + DEBUG_PRINTLN(F("504 Unknow OPTS") ); + } + } + // +#endif + // HELP + // + else if( CommandIs( "HELP" )) { + client.println(F("200 Commands implemented:") ); + client.println(F(" USER, PASS, AUTH (AUTH only return 'not implemented' code)") ); + client.println(F(" CDUP, CWD, PWD, QUIT, NOOP") ); + client.println(F(" MODE, PASV, PORT, STRU, TYPE") ); + client.println(F(" ABOR, DELE, LIST, NLST, MLST, MLSD") ); + client.println(F(" APPE, RETR, STOR") ); + client.println(F(" MKD, RMD") ); + client.println(F(" RNTO, RNFR") ); + client.println(F(" MDTM, MFMT") ); + client.println(F(" FEAT, SIZE") ); + client.println(F(" SITE FREE") ); + client.println(F(" HELP") ); + } + // + // RETR - Retrieve + // + else if( CommandIs( "RETR" )) + { + char path[ FTP_CWD_SIZE ]; + if( haveParameter() && makeExistsPath( path )) { + if( ! openFile( path, FTP_FILE_READ )) { + client.print( F("450 Can't open ") ); client.print( parameter ); + } else if( dataConnect( false )) + { + DEBUG_PRINT( F(" Sending ") ); DEBUG_PRINT( parameter ); DEBUG_PRINT( F(" size ") ); DEBUG_PRINTLN( long( fileSize( file )) ); + + if (FtpServer::_transferCallback) { + FtpServer::_transferCallback(FTP_DOWNLOAD_START, parameter, long( fileSize( file ))); + } + + + client.print( F("150-Connected to port ") ); client.println( dataPort ); + client.print( F("150 ") ); client.print( long( fileSize( file )) ); client.println( F(" bytes to download") ); + millisBeginTrans = millis(); + bytesTransfered = 0; + transferStage = FTP_Retrieve; + } + } + } + // + // STOR - Store + // APPE - Append + // + else if( CommandIs( "STOR" ) || CommandIs( "APPE" )) + { + char path[ FTP_CWD_SIZE ]; + if( haveParameter() && makePath( path )) + { + bool open; + if( exists( path )) { + DEBUG_PRINTLN(F("APPEND FILE!!")); + open = openFile( path, ( CommandIs( "APPE" ) ? FTP_FILE_WRITE_APPEND : FTP_FILE_WRITE_CREATE )); + } else { + DEBUG_PRINTLN(F("CREATE FILE!!")); + open = openFile( path, FTP_FILE_WRITE_CREATE ); + } + + data.stop(); + data.flush(); + + DEBUG_PRINT(F("open/create ")); + DEBUG_PRINTLN(open); + if( ! open ){ + client.print( F("451 Can't open/create ") ); client.println( parameter ); + }else if( ! dataConnect()) // && !data.available()) + file.close(); + else + { + DEBUG_PRINT( F(" Receiving ") ); DEBUG_PRINTLN( parameter ); + + millisBeginTrans = millis(); + bytesTransfered = 0; + transferStage = FTP_Store; + + if (FtpServer::_transferCallback) { + + FtpServer::_transferCallback(FTP_UPLOAD_START, parameter, bytesTransfered); + } + + } + } + } + // + // MKD - Make Directory + // + else if( CommandIs( "MKD" )) + { + char path[ FTP_CWD_SIZE ]; + if( haveParameter() && makePath( path )) + { + if( exists( path )) { + client.print( F("521 \"") ); client.print( parameter ); client.println( F("\" directory already exists") ); + } else + { + DEBUG_PRINT( F(" Creating directory ")); DEBUG_PRINTLN( parameter ); + +#if STORAGE_TYPE != STORAGE_SPIFFS + if( makeDir( path )) { + client.print( F("257 \"") ); client.print( parameter ); client.print( F("\"") ); client.println( F(" created") ); + } else { +#endif + client.print( F("550 Can't create \"") ); client.print( parameter ); client.println( F("\"") ); +#if STORAGE_TYPE != STORAGE_SPIFFS + } +#endif + } + } + } + // + // RMD - Remove a Directory + // + else if( CommandIs( "RMD" )) + { + char path[ FTP_CWD_SIZE ]; + if( haveParameter() && makeExistsPath( path )) { + if( removeDir( path )) + { + DEBUG_PRINT( F(" Deleting ") ); DEBUG_PRINTLN( path ); + + client.print( F("250 \"") ); client.print( parameter ); client.println( F("\" deleted") ); + } + else { + client.print( F("550 Can't remove \"") ); client.print( parameter ); client.println( F("\". Directory not empty?") ); + } + } + } + // + // RNFR - Rename From + // + else if( CommandIs( "RNFR" )) + { + rnfrName[ 0 ] = 0; + if( haveParameter() && makeExistsPath( rnfrName )) + { + DEBUG_PRINT( F(" Ready for renaming ") ); DEBUG_PRINTLN( rnfrName ); + + client.println(F("350 RNFR accepted - file exists, ready for destination") ); + rnfrCmd = true; + } + } + // + // RNTO - Rename To + // + else if( CommandIs( "RNTO" )) + { + char path[ FTP_CWD_SIZE ]; + char dirp[ FTP_FIL_SIZE ]; + if( strlen( rnfrName ) == 0 || ! rnfrCmd ) { + client.println(F("503 Need RNFR before RNTO") ); + } else if( haveParameter() && makePath( path )) + { + if( exists( path )) { + client.print( F("553 ") ); client.print( parameter ); client.println( F(" already exists") ); + } else + { + strcpy( dirp, path ); + char * psep = strrchr( dirp, '/' ); + bool fail = psep == NULL; + if( ! fail ) + { + if( psep == dirp ) + psep ++; + * psep = 0; +// fail = ! isDir( dirp ); +// if( fail ) { +// client.print( F("550 \"") ); client.print( dirp ); client.println( F("\" is not directory") ); +// } else +// { + DEBUG_PRINT( F(" Renaming ") ); DEBUG_PRINT( rnfrName ); DEBUG_PRINT( F(" to ") ); DEBUG_PRINTLN( path ); + + if( rename( rnfrName, path )) + client.println(F("250 File successfully renamed or moved") ); + else + fail = true; +// } + } + if( fail ) + client.println(F("451 Rename/move failure") ); + } + } + rnfrCmd = false; + } + /* + // + // SYST - System + // + else if( CommandIs( "SYST" )) + FtpOutCli << F("215 MSDOS") << endl; + */ + + /////////////////////////////////////// + // // + // EXTENSIONS COMMANDS (RFC 3659) // + // // + /////////////////////////////////////// + + // + // MDTM && MFMT - File Modification Time (see RFC 3659) + // + else if( CommandIs( "MDTM" ) || CommandIs( "MFMT" )) + { + if( haveParameter()) + { + char path[ FTP_CWD_SIZE ]; + char * fname = parameter; + uint16_t year; + uint8_t month, day, hour, minute, second, setTime; + char dt[ 15 ]; + bool mdtm = CommandIs( "MDTM" ); + + setTime = getDateTime( dt, & year, & month, & day, & hour, & minute, & second ); + // fname point to file name + fname += setTime; + if( strlen( fname ) <= 0 ) { + client.println(F("501 No file name") ); + } else if( makeExistsPath( path, fname )) { + if( setTime ) // set file modification time + { + if( timeStamp( path, year, month, day, hour, minute, second )) { + client.print( F("213 ") ); client.println( dt ); + } else { + client.println(F("550 Unable to modify time" )); + } + } + else if( mdtm ) // get file modification time + { + uint16_t dat=0, tim=0; + char dtStr[ 15 ]; + if( getFileModTime( path, &dat, &tim )) { + client.print( F("213 ") ); client.println( makeDateTimeStr( dtStr, dat, tim ) ); + } else { + client.println("550 Unable to retrieve time" ); + } + } + } + } + } + // + // SIZE - Size of the file + // + else if( CommandIs( "SIZE" )) + { + char path[ FTP_CWD_SIZE ]; + if( haveParameter() && makeExistsPath( path )) { + if( ! openFile( path, FTP_FILE_READ )) { + client.print( F("450 Can't open ") ); client.println( parameter ); + } else + { + client.print( F("213 ") ); client.println( long( fileSize( file )) ); + file.close(); + } + } + } + // + // SITE - System command + // + else if( CommandIs( "SITE" )) + { + if( ParameterIs( "FREE" )) + { + uint32_t capa = capacity(); + if(( capa >> 10 ) < 1000 ) { // less than 1 Giga + client.print( F("200 ") ); client.print( free() ); client.print( F(" kB free of ") ); + client.print( capa ); client.println( F(" kB capacity") ); + }else { + client.print( F("200 ") ); client.print( ( free() >> 10 ) ); client.print( F(" MB free of ") ); + client.print( ( capa >> 10 ) ); client.println( F(" MB capacity") ); + } + } + else { + client.print( F("500 Unknow SITE command ") ); client.println( parameter ); + } + } + // + // Unrecognized commands ... + // + else + client.println(F("500 Unknow command") ); + return true; +} + +int FtpServer::dataConnect( bool out150 ) +{ + if( ! data.connected()) { + if( dataConn == FTP_Pasive ) + { + uint16_t count = 1000; // wait up to a second + while( ! data.connected() && count -- > 0 ) + { + #if (FTP_SERVER_NETWORK_TYPE == NETWORK_WiFiNINA) + data = dataServer.available(); + #elif (defined(ESP8266) && (FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_ASYNC || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266 || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_242)) // || defined(ARDUINO_ARCH_RP2040) + if( dataServer.hasClient()) + { + data.stop(); + data = dataServer.available(); + } + #else + data = dataServer.accept(); + #endif + delay( 1 ); + } + } + else if( dataConn == FTP_Active ) + data.connect( dataIp, dataPort ); + } + +//#ifdef ESP8266 + if( ! ( data.connected() || data.available())) { +//#else +// if( ! ( data.connected() )) { +//#endif + client.println(F("425 No data connection")); + } else if( out150 ) { + client.print( F("150 Accepted data connection to port ") ); client.println( dataPort ); + } +//#ifdef ESP8266 + return data.connected() || data.available(); +//#else +// return data.connected(); +//#endif + +} + +bool FtpServer::dataConnected() +{ + if( data.connected()) + return true; + data.stop(); + client.println(F("426 Data connection closed. Transfer aborted") ); + transferStage = FTP_Close; + return false; +} + +bool FtpServer::openDir( FTP_DIR * pdir ) +{ + bool openD; +#if (STORAGE_TYPE == STORAGE_LITTLEFS && (defined(ESP8266) || defined(ARDUINO_ARCH_RP2040))) + if( cwdName == 0 ) { + dir = STORAGE_MANAGER.openDir( "/" ); + } else { + dir = STORAGE_MANAGER.openDir( cwdName ); + } + openD = dir.rewind(); + + if( ! openD ) { + client.print( F("550 Can't open directory ") ); client.println( cwdName ); + } +#elif STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC + if( cwdName == 0 ) { + dir = STORAGE_MANAGER.open( "/" ); + } else { + dir = STORAGE_MANAGER.open( cwdName ); + } + openD = true; + if( ! openD ) { + client.print( F("550 Can't open directory ") ); client.println( cwdName ); + } +#elif STORAGE_TYPE == STORAGE_FFAT || (STORAGE_TYPE == STORAGE_LITTLEFS && defined(ESP32)) + if( cwdName == 0 ) { + dir = STORAGE_MANAGER.open( "/" ); + } else { + dir = STORAGE_MANAGER.open( cwdName ); + } + openD = true; + if( ! openD ) { + client.print( F("550 Can't open directory ") ); client.println( cwdName ); + } +#elif STORAGE_TYPE == STORAGE_SEEED_SD + if( cwdName == 0 ) { + DEBUG_PRINT("cwdName forced -> "); + DEBUG_PRINTLN(cwdName ); + + FTP_DIR d = STORAGE_MANAGER.open( "/" ); + dir=d; + } else { + DEBUG_PRINT("cwdName -> "); + DEBUG_PRINTLN(cwdName ); + + FTP_DIR d = STORAGE_MANAGER.open( cwdName ); + dir=d; + } + + openD = dir.isDirectory(); + + if( ! openD ) { + client.print( F("550 Can't open directory ") ); client.println( cwdName ); + } +#elif STORAGE_TYPE == STORAGE_SPIFFS + if( cwdName == 0 || strcmp(cwdName, "/") == 0 ) { + DEBUG_PRINT("DIRECTORY / EXIST "); +#if ESP8266 + dir = STORAGE_MANAGER.openDir( "/" ); +#else + dir = STORAGE_MANAGER.open( "/" ); +#endif + openD = true; + + } else { + openD = false; + } + if( ! openD ) { + client.print( F("550 Can't open directory ") ); client.println( cwdName ); + } +#else + if( cwdName == 0 ) { + openD = pdir->open( "/" ); + } else { + openD = pdir->open( cwdName ); + } + if( ! openD ) { + client.print( F("550 Can't open directory ") ); client.println( cwdName ); + } +#endif + return openD; +} + +bool FtpServer::doRetrieve() +{ + if( ! dataConnected()) + { + file.close(); + return false; + } + int16_t nb = file.read( buf, FTP_BUF_SIZE ); + if( nb > 0 ) + { + data.write( buf, nb ); + DEBUG_PRINT(F("NB --> ")); + DEBUG_PRINTLN(nb); + bytesTransfered += nb; + + if (FtpServer::_transferCallback) { + FtpServer::_transferCallback(FTP_DOWNLOAD, getFileName(&file), bytesTransfered); + } + +// RoSchmi +#if STORAGE_TYPE != STORAGE_SEEED_SD + return true; +#endif + } + closeTransfer(); + return false; +} + +bool FtpServer::doStore() +{ + int16_t na = data.available(); + if( na == 0 ) { + DEBUG_PRINTLN("NO DATA AVAILABLE!"); +#if FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_SEEED_RTL8720DN + data.stop(); +#endif + if( data.connected()) { + return true; + } else + { + closeTransfer(); + return false; + } + } + + if( na > FTP_BUF_SIZE ) { + na = FTP_BUF_SIZE; + } + int16_t nb = data.read((uint8_t *) buf, na ); + int16_t rc = 0; + if( nb > 0 ) + { + DEBUG_PRINT("NB -> "); + DEBUG_PRINTLN(nb); + + rc = file.write( buf, nb ); + DEBUG_PRINT("RC -> "); + DEBUG_PRINTLN(rc); + bytesTransfered += nb; + + if (FtpServer::_transferCallback) { + + FtpServer::_transferCallback(FTP_UPLOAD, getFileName(&file), bytesTransfered); + } + } + if( nb < 0 || rc == nb ) { + return true; + } + client.println(F("552 Probably insufficient storage space") ); + file.close(); + data.stop(); + return false; +} + +void generateFileLine(FTP_CLIENT_NETWORK_CLASS* data, bool isDirectory, const char* fn, long fz, const char* time, const char* user, bool writeFilename = true) { + if( isDirectory ) { + // data.print( F("+/,\t") ); + // DEBUG_PRINT(F("+/,\t")); + + data->print( F("drwxrwsr-x\t2\t")); + data->print( user ); + data->print( F("\t") ); + data->print( long( 4096 ) ); + data->print( F("\t") ); + + DEBUG_PRINT( F("drwxrwsr-x\t2\t") ); + DEBUG_PRINT( user ); + DEBUG_PRINT( F("\t") ); + + DEBUG_PRINT( long( 4096 ) ); + DEBUG_PRINT( F("\t") ); + + data->print(time); + DEBUG_PRINT(time); + + data->print( F("\t") ); + if (writeFilename) data->println( fn ); + + DEBUG_PRINT( F("\t") ); + if (writeFilename) DEBUG_PRINTLN( fn ); + + } else { +// data.print( F("+r,s") ); +// DEBUG_PRINT(F("+r,s")); + + data->print( F("-rw-rw-r--\t1\t") ); + data->print( user ); + data->print( F("\t") ); + data->print( fz ); + data->print( F("\t") ); + + DEBUG_PRINT( F("-rw-rw-r--\t1\t") ); + DEBUG_PRINT( user ); + DEBUG_PRINT( F("\t") ); + DEBUG_PRINT( fz ); + DEBUG_PRINT( F("\t") ); + + data->print(time); + DEBUG_PRINT(time); + + data->print( F("\t") ); + if (writeFilename) data->println( fn ); + + DEBUG_PRINT( F("\t") ); + if (writeFilename) DEBUG_PRINTLN( fn ); + } + +} + +#if defined(ESP32) || defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) +// +// Formats printable String from a time_t timestamp +// +String makeDateTimeStrList(time_t ft, bool dateContracted = false) +{ + String tmp; + // a buffer with enough space for the formats + char buf[25]; + char *b = buf; + + // break down the provided file time + struct tm _tm; + gmtime_r(&ft, &_tm); + + if (dateContracted) + { + // "%Y%m%d%H%M%S", e.g. "20200517123400" + strftime(b, sizeof(buf), "%Y%m%d%H%M%S", &_tm); + } + else + { + // "%h %d %H:%M", e.g. "May 17 12:34" for file dates of the current year + // "%h %d %Y" , e.g. "May 17 2019" for file dates of any other years + + // just convert both ways, select later what's to be shown + // buf becomes "May 17 2019May 17 12:34" + strftime(b, sizeof(buf), "%h %d %H:%M%h %d %Y", &_tm); + + // check for a year != year from now + int fileYear = _tm.tm_year; + time_t nowTime = time(NULL); + gmtime_r(&nowTime, &_tm); + if (fileYear == _tm.tm_year) + { + // cut off 2nd half - year variant + b[12] = '\0'; + } + else + { + // skip 1st half - time variant + b += 12; + } + } + tmp = b; + return tmp; +} + +// https://files.stairways.com/other/ftp-list-specs-info.txt +void generateFileLine(FTP_CLIENT_NETWORK_CLASS* data, bool isDirectory, const char* fn, long fz, time_t time, const char* user, bool writeFilename = true) { + generateFileLine(data, isDirectory, fn, fz, makeDateTimeStrList(time).c_str(), user, writeFilename); +} +#endif + +bool FtpServer::doList() +{ + if( ! dataConnected()) + { +#if STORAGE_TYPE != STORAGE_SPIFFS && STORAGE_TYPE != STORAGE_LITTLEFS && STORAGE_TYPE != STORAGE_SEEED_SD + dir.close(); +#endif + return false; + } +#if STORAGE_TYPE == STORAGE_SPIFFS + #if ESP8266 + if( dir.next()) + #else + FTP_FILE fileDir = dir.openNextFile(); + if( fileDir ) + #endif + { + +// data.print( F("+r,s") ); +// #if ESP8266 +// data.print( long( dir.fileSize()) ); +// data.print( F(",\t") ); +// data.println( dir.fileName() ); +// #else +// data.print( long( fileDir.size()) ); +// data.print( F(",\t") ); +// data.println( fileDir.name() ); +// #endif + + + +#ifdef ESP8266 + String fn = dir.fileName(); + long fz = long( dir.fileSize()); + if (fn[0]=='/') { fn.remove(0, fn.lastIndexOf("/")+1); } + time_t time = dir.fileTime(); + generateFileLine(&data, false, fn.c_str(), fz, time, this->user); +#else + long fz = long( fileDir.size()); + const char* fnC = fileDir.name(); + const char* fn; + if ( fnC[0] == '/' ) { + fn = &fnC[1]; + }else{ + fn = fnC; + } + + time_t time = fileDir.getLastWrite(); + generateFileLine(&data, false, fn, fz, time, this->user); + +#endif + + nbMatch ++; + return true; + } +#elif STORAGE_TYPE == STORAGE_LITTLEFS || STORAGE_TYPE == STORAGE_SEEED_SD || STORAGE_TYPE == STORAGE_FFAT + #if defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) + if( dir.next()) + #else +#if STORAGE_TYPE == STORAGE_SEEED_SD + FTP_FILE fileDir = STORAGE_MANAGER.open(dir.name()); + fileDir = dir.openNextFile(); +#else + FTP_FILE fileDir = dir.openNextFile(); +#endif + if( fileDir ) +#endif + { + + #if defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) + long fz = long( dir.fileSize()); +// const char* fn = dir.fileName().c_str(); + String aza = dir.fileName(); + const char* fn = aza.c_str(); //Serial.printf("test %s ", fn); + +// data.print( long( dir.fileSize()) ); +// data.print( F(",\t") ); +// data.println( dir.fileName() ); + #elif STORAGE_TYPE == STORAGE_SEEED_SD + const char* fnC = fileDir.name(); + const char* fn; + if ( fnC[0] == '/' ) { + fn = &fnC[1]; + }else{ + fn = fnC; + } + long fz = fileDir.size(); + #else + long fz = long( fileDir.size()); + const char* fn = fileDir.name(); + +// data.print( long( fileDir.size()) ); +// data.print( F("\t") ); +// data.println( fileDir.name() ); + +// DEBUG_PRINT( long( fileDir.size())); +// DEBUG_PRINT( F("\t") ); +// DEBUG_PRINTLN( fileDir.name() ); + #endif + #if defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) + time_t time = dir.fileTime(); + generateFileLine(&data, dir.isDirectory(), fn, fz, time, this->user); + #elif ESP32 + time_t time = fileDir.getLastWrite(); + generateFileLine(&data, fileDir.isDirectory(), fn, fz, time, this->user); + #else + generateFileLine(&data, fileDir.isDirectory(), fn, fz, "Jan 01 00:00", this->user); + #endif + nbMatch ++; + return true; + } +#elif STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC + FTP_FILE fileDir = dir.openNextFile(); + if( fileDir ) + { + +// data.print( F("+r,s") ); +// data.print( long( fileDir.size()) ); +// data.print( F(",\t") ); +// data.println( fileDir.name() ); + + String fn = fileDir.name(); + if (fn[0]=='/') { fn.remove(0, fn.lastIndexOf("/")+1); } + + generateFileLine(&data, fileDir.isDirectory(), fn.c_str(), long( fileDir.size()), "Jan 01 00:00", this->user); + + nbMatch ++; + return true; + } + +#elif STORAGE_TYPE == STORAGE_FATFS + if( dir.nextFile()) + { +// if( dir.isDir()) { +// data.print( F("+/,\t") ); +// } else { +// data.print( F("+r,s") ); data.print( long( dir.fileSize()) ); data.print( F(",\t") ); +// } +// data.println( dir.fileName() ); + + String fn = dir.fileName(); + if (fn[0]=='/') { fn.remove(0, fn.lastIndexOf("/")+1); } + + generateFileLine(&data, dir.isDir(), fn.c_str(), long( dir.fileSize()), "Jan 01 00:00", this->user); + + nbMatch ++; + return true; + } +#else + if( file.openNext( &dir, FTP_FILE_READ_ONLY )) + { +// if( file.isDir()) { +// data.print( F("+/,\t") ); +// } else { +// data.print( F("+r,s") ); data.print( long( fileSize( file )) ); data.print( F(",\t") ); +// } + + generateFileLine(&data, file.isDir(), "", long( fileSize( file )), "Jan 01 00:00", this->user, false); + + file.printName( & data ); + data.println(); + file.close(); + nbMatch ++; + return true; + } +#endif + client.print( F("226 ") ); client.print( nbMatch ); client.println( F(" matches total") ); +#if STORAGE_TYPE != STORAGE_SPIFFS && STORAGE_TYPE != STORAGE_LITTLEFS && STORAGE_TYPE != STORAGE_SEEED_SD + dir.close(); +#endif + data.stop(); + return false; +} + +bool FtpServer::doMlsd() +{ + if( ! dataConnected()) + { +#if STORAGE_TYPE != STORAGE_SPIFFS && STORAGE_TYPE != STORAGE_LITTLEFS && STORAGE_TYPE != STORAGE_SEEED_SD + dir.close(); +#endif + DEBUG_PRINTLN(F("Not connected!!")); + return false; + } + DEBUG_PRINTLN(F("Connected!!")); + +#if STORAGE_TYPE == STORAGE_SPIFFS + DEBUG_PRINTLN("DIR MLSD "); + #if ESP8266 + if( dir.next()) + #else + File fileDir = dir.openNextFile(); + if( fileDir ) + #endif + { + DEBUG_PRINTLN("DIR NEXT "); + char dtStr[ 15 ]; + + struct tm * timeinfo; + + #if ESP8266 + time_t time = dir.fileTime(); + #else + time_t time = fileDir.getLastWrite(); + #endif + + timeinfo = localtime ( &time ); + + // 2000 01 01 16 06 56 + + strftime (dtStr,15,"%Y%m%d%H%M%S",timeinfo); + + + #if ESP8266 + String fn = dir.fileName(); + fn.remove(0, fn.lastIndexOf("/")+1); + long fz = dir.fileSize(); + #else + String fn = fileDir.name(); + fn.remove(0, fn.lastIndexOf("/")+1); + long fz = fileDir.size(); + #endif + + data.print( F("Type=") ); + + data.print( F("file") ); + data.print( F(";Modify=") ); data.print(dtStr);// data.print( makeDateTimeStr( dtStr, time, time) ); + data.print( F(";Size=") ); data.print( fz ); + data.print( F("; ") ); data.println( fn ); + + DEBUG_PRINT( F("Type=") ); + DEBUG_PRINT( F("file") ); + + DEBUG_PRINT( F(";Modify=") ); DEBUG_PRINT(dtStr); //DEBUG_PRINT( makeDateTimeStr( dtStr, time, time) ); + DEBUG_PRINT( F(";Size=") ); DEBUG_PRINT( fz ); + DEBUG_PRINT( F("; ") ); DEBUG_PRINTLN( fn ); + + nbMatch ++; + return true; + } +#elif STORAGE_TYPE == STORAGE_LITTLEFS || STORAGE_TYPE == STORAGE_SEEED_SD || STORAGE_TYPE == STORAGE_FFAT + DEBUG_PRINTLN("DIR MLSD "); + #if defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) + if( dir.next()) + #else +#if STORAGE_TYPE == STORAGE_SEEED_SD + File fileDir = STORAGE_MANAGER.open(dir.name()); + fileDir = dir.openNextFile(); +#else + File fileDir = dir.openNextFile(); +#endif + DEBUG_PRINTLN(dir); + DEBUG_PRINTLN(fileDir); + if( fileDir ) + #endif + { + DEBUG_PRINTLN("DIR NEXT "); + char dtStr[ 15 ]; + + + #if STORAGE_TYPE == STORAGE_SEEED_SD + struct tm * timeinfo; + + strcpy(dtStr, "19700101000000"); + #else + struct tm * timeinfo; + + #if defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) + time_t time = dir.fileTime(); + #else + time_t time = fileDir.getLastWrite(); + #endif + + timeinfo = localtime ( &time ); + + // 2000 01 01 16 06 56 + + strftime (dtStr,15,"%Y%m%d%H%M%S",timeinfo); + #endif + + #if defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) + String fn = dir.fileName(); + long fz = dir.fileSize(); + FTP_DIR fileDir = dir; + #elif STORAGE_TYPE == STORAGE_SEEED_SD + String fn = fileDir.name(); + fn.remove(0, strlen(dir.name())); + if (fn[0]=='/') { fn.remove(0, fn.lastIndexOf("/")+1); } + long fz = fileDir.size(); + #else + String fn = fileDir.name(); + fn.remove(0, fn.lastIndexOf("/")+1); + long fz = fileDir.size(); + #endif + + data.print( F("Type=") ); + + data.print( ( fileDir.isDirectory() ? F("dir") : F("file")) ); + data.print( F(";Modify=") ); data.print(dtStr);// data.print( makeDateTimeStr( dtStr, time, time) ); + data.print( F(";Size=") ); data.print( fz ); + data.print( F("; ") ); data.println( fn ); + + DEBUG_PRINT( F("Type=") ); + DEBUG_PRINT( ( fileDir.isDirectory() ? F("dir") : F("file")) ); + + DEBUG_PRINT( F(";Modify=") ); DEBUG_PRINT(dtStr); //DEBUG_PRINT( makeDateTimeStr( dtStr, time, time) ); + DEBUG_PRINT( F(";Size=") ); DEBUG_PRINT( fz ); + DEBUG_PRINT( F("; ") ); DEBUG_PRINTLN( fn ); + + nbMatch ++; +// RoSchmi: next line was commented +#if STORAGE_TYPE == STORAGE_SEEED_SD + fileDir.close(); +#endif + return true; + } + +#elif STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC + DEBUG_PRINTLN("DIR MLSD "); + File fileDir = dir.openNextFile(); + if( fileDir ) + { + DEBUG_PRINTLN("DIR NEXT "); + char dtStr[ 15 ]; + + struct tm * timeinfo; + + strcpy(dtStr, "19700101000000"); + + +// long fz = dir.fileSize(); + String fn = fileDir.name(); + +//#ifdef ESP32 + fn.remove(0, fn.lastIndexOf("/")+1); +//#else if !defined(ESP8266) +// fn.remove(0, 1); +//#endif + + + long fz = fileDir.size(); + + data.print( F("Type=") ); + + data.print( ( fileDir.isDirectory() ? F("dir") : F("file")) ); + data.print( F(";Modify=") ); data.print(dtStr);// data.print( makeDateTimeStr( dtStr, time, time) ); + data.print( F(";Size=") ); data.print( fz ); + data.print( F("; ") ); data.println( fn ); + + DEBUG_PRINT( F("Type=") ); + DEBUG_PRINT( ( fileDir.isDirectory() ? F("dir") : F("file")) ); + + DEBUG_PRINT( F(";Modify=") ); DEBUG_PRINT(dtStr); //DEBUG_PRINT( makeDateTimeStr( dtStr, time, time) ); + DEBUG_PRINT( F(";Size=") ); DEBUG_PRINT( fz ); + DEBUG_PRINT( F("; ") ); DEBUG_PRINTLN( fn ); + + nbMatch ++; + return true; + } + +#elif STORAGE_TYPE == STORAGE_FATFS + if( dir.nextFile()) + { + char dtStr[ 15 ]; + data.print( F("Type=") ); data.print( ( dir.isDir() ? F("dir") : F("file")) ); + data.print( F(";Modify=") ); data.print( makeDateTimeStr( dtStr, dir.fileModDate(), dir.fileModTime()) ); + data.print( F(";Size=") ); data.print( long( dir.fileSize()) ); + data.print( F("; ") ); data.println( dir.fileName() ); + nbMatch ++; + return true; + } +#else + if( file.openNext( &dir, FTP_FILE_READ_ONLY )) + { + char dtStr[ 15 ]; + uint16_t filelwd, filelwt; + bool gfmt = getFileModTime( & filelwd, & filelwt ); + DEBUG_PRINT("gfmt --> "); + DEBUG_PRINTLN(gfmt); + if( gfmt ) + { + data.print( F("Type=") ); data.print( ( file.isDir() ? F("dir") : F("file")) ); + data.print( F(";Modify=") ); data.print( makeDateTimeStr( dtStr, filelwd, filelwt ) ); + data.print( F(";Size=") ); data.print( long( fileSize( file )) ); data.print( F("; ") ); + file.printName( & data ); + data.println(); + + DEBUG_PRINT( F("Type=") ); DEBUG_PRINT( ( file.isDir() ? F("dir") : F("file")) ); + DEBUG_PRINT( F(";Modify=") ); DEBUG_PRINT( makeDateTimeStr( dtStr, filelwd, filelwt ) ); + DEBUG_PRINT( F(";Size=") ); DEBUG_PRINT( long( fileSize( file )) ); DEBUG_PRINT( F("; ") ); +// DEBUG_PRINT(file.name()); + DEBUG_PRINTLN(); + nbMatch ++; + } + file.close(); + return gfmt; + } +#endif + client.println(F("226-options: -a -l") ); + client.print( F("226 ") ); client.print( nbMatch ); client.println( F(" matches total") ); +#if STORAGE_TYPE != STORAGE_SPIFFS && STORAGE_TYPE != STORAGE_LITTLEFS && STORAGE_TYPE != STORAGE_SEEED_SD && STORAGE_TYPE != STORAGE_SEEED_SD + dir.close(); +#endif + data.stop(); + DEBUG_PRINTLN(F("All file readed!!")); + return false; +} + +void FtpServer::closeTransfer() +{ + uint32_t deltaT = (int32_t) ( millis() - millisBeginTrans ); + if( deltaT > 0 && bytesTransfered > 0 ) + { + DEBUG_PRINT( F(" Transfer completed in ") ); DEBUG_PRINT( deltaT ); DEBUG_PRINTLN( F(" ms, ") ); + DEBUG_PRINT( bytesTransfered / deltaT ); DEBUG_PRINTLN( F(" kbytes/s") ); + + if (FtpServer::_transferCallback) { + FtpServer::_transferCallback(FTP_TRANSFER_STOP, getFileName(&file), bytesTransfered); + } + + + client.println(F("226-File successfully transferred") ); + client.print( F("226 ") ); client.print( deltaT ); client.print( F(" ms, ") ); + client.print( bytesTransfered / deltaT ); client.println( F(" kbytes/s") ); + } + else + client.println(F("226 File successfully transferred") ); + + file.close(); + data.stop(); +} + +void FtpServer::abortTransfer() +{ + if( transferStage != FTP_Close ) + { + if (FtpServer::_transferCallback) { + FtpServer::_transferCallback(FTP_TRANSFER_ERROR, getFileName(&file), bytesTransfered); + } + + file.close(); +#if STORAGE_TYPE != STORAGE_SPIFFS && STORAGE_TYPE != STORAGE_LITTLEFS && STORAGE_TYPE != STORAGE_SEEED_SD + dir.close(); +#endif + client.println(F("426 Transfer aborted") ); + DEBUG_PRINTLN( F(" Transfer aborted!") ); + + transferStage = FTP_Close; + } +// if( data.connected()) + data.stop(); +} + +// Read a char from client connected to ftp server +// +// update cmdLine and command buffers, iCL and parameter pointers +// +// return: +// -2 if buffer cmdLine is full +// -1 if line not completed +// 0 if empty line received +// length of cmdLine (positive) if no empty line received + +int8_t FtpServer::readChar() +{ + int8_t rc = -1; + + if( client.available()) + { + char c = client.read(); + DEBUG_PRINT("-"); + DEBUG_PRINT( c ); + + if( c == '\\' ) + c = '/'; + if( c != '\r' ){ + if( c != '\n' ) + { + if( iCL < FTP_CMD_SIZE ) + cmdLine[ iCL ++ ] = c; + else + rc = -2; // Line too long + } + else + { + cmdLine[ iCL ] = 0; + command[ 0 ] = 0; + parameter = NULL; + // empty line? + if( iCL == 0 ) + rc = 0; + else + { + rc = iCL; + // search for space between command and parameter + parameter = strchr( cmdLine, ' ' ); + if( parameter != NULL ) + { + if( parameter - cmdLine > 4 ) + rc = -2; // Syntax error + else + { + strncpy( command, cmdLine, parameter - cmdLine ); + command[ parameter - cmdLine ] = 0; + while( * ( ++ parameter ) == ' ' ) + ; + } + } + else if( strlen( cmdLine ) > 4 ) + rc = -2; // Syntax error. + else + strcpy( command, cmdLine ); + iCL = 0; + } + } + } + if( rc > 0 ) + for( uint8_t i = 0 ; i < strlen( command ); i ++ ) + command[ i ] = toupper( command[ i ] ); + if( rc == -2 ) + { + iCL = 0; + client.println(F("500 Syntax error")); + } + } + return rc; +} + +bool FtpServer::haveParameter() +{ + if( parameter != NULL && strlen( parameter ) > 0 ) + return true; + client.println("501 No file name" ); + return false; +} + +int utf8_strlen(const String& str) +{ + int c,i,ix,q; + for (q=0, i=0, ix=str.length(); i < ix; i++, q++) + { + c = (unsigned char) str[i]; + if (c>=0 && c<=127) i+=0; + else if ((c & 0xE0) == 0xC0) i+=1; + else if ((c & 0xF0) == 0xE0) i+=2; + else if ((c & 0xF8) == 0xF0) i+=3; + //else if (($c & 0xFC) == 0xF8) i+=4; // 111110bb //byte 5, unnecessary in 4 byte UTF-8 + //else if (($c & 0xFE) == 0xFC) i+=5; // 1111110b //byte 6, unnecessary in 4 byte UTF-8 + else return 0;//invalid utf8 + } + return q; +} + +//// ****** UTF8-Decoder: convert UTF8-string to extended ASCII ******* +//static byte c1; // Last character buffer +// +//// Convert a single Character from UTF8 to Extended ASCII +//// Return "0" if a byte has to be ignored +//byte utf8ascii(byte ascii) { +// if ( ascii<128 ) // Standard ASCII-set 0..0x7F handling +// { c1=0; +// return( ascii ); +// } +// +// // get previous input +// byte last = c1; // get last char +// c1=ascii; // remember actual character +// +// switch (last) // conversion depending on first UTF8-character +// { case 0xC2: return (ascii); break; +// case 0xC3: return (ascii | 0xC0); break; +// case 0x82: if(ascii==0xAC) return(0x80); // special case Euro-symbol +// } +// +// return (0); // otherwise: return zero, if character has to be ignored +//} +// +//// convert String object from UTF8 String to Extended ASCII +//String utf8ascii(String s) +//{ +// String r=""; +// char c; +// for (int i=0; i 1 ) + fullName[ strl ] = 0; + if( strlen( fullName ) >= FTP_CWD_SIZE ) + { + client.println(F("500 Command line too long")); + return false; + } +#ifdef UTF8_SUPPORT +// for( uint8_t i = 0; i < utf8_strlen( fullName ); i ++ ) { +// +// } + + DEBUG_PRINT(F("utf8_strlen")); + DEBUG_PRINTLN(utf8_strlen(fullName)); +// DEBUG_PRINT(F("utf8_strlen2")); +// DEBUG_PRINTLN(utf8_strlen2(fullName)); + + if (utf8_strlen(fullName)>=FILENAME_LENGTH) { + client.println(F("553 File name not allowed. Too long.") ); + return false; + } +#else + for( uint8_t i = 0; i < strlen( fullName ); i ++ ) { + if( ! legalChar( fullName[i])) + { + client.println(F("553 File name not allowed") ); + return false; + } + } + if (strlen(fullName)>=FILENAME_LENGTH) { + client.println(F("553 File name not allowed. Too long.") ); + return false; + } +#endif + return true; +} + +bool FtpServer::makeExistsPath( char * path, char * param ) +{ + if( ! makePath( path, param )) + return false; + // RoSchmi + //#if STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_SD +#if (STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC || STORAGE_TYPE == STORAGE_SEEED_SD) + if (strcmp(path, "/") == 0) return true; +#endif + DEBUG_PRINT("PATH --> ") + DEBUG_PRINT(path) + if( exists( path )) { + DEBUG_PRINTLN(" ...EXIST!") + return true; + } + DEBUG_PRINTLN(" ...NOT EXIST!") + client.print(F("550 ")); client.print( path ); client.println( F(" not found.") ); + return false; +} + +// Calculate year, month, day, hour, minute and second +// from first parameter sent by MDTM command (YYYYMMDDHHMMSS) +// Accept longer parameter YYYYMMDDHHMMSSmmm where mmm are milliseconds +// but don't take in account additional digits +// +// parameters: +// dt: 15 length string for 14 digits and terminator +// pyear, pmonth, pday, phour, pminute and psecond: pointer of +// variables where to store data +// +// return: +// 0 if parameter is not YYYYMMDDHHMMSS +// length of parameter + space +// +// Date/time are expressed as a 14 digits long string +// terminated by a space and followed by name of file + +uint8_t FtpServer::getDateTime( char * dt, uint16_t * pyear, uint8_t * pmonth, uint8_t * pday, + uint8_t * phour, uint8_t * pminute, uint8_t * psecond ) +{ + uint8_t i; + dt[ 0 ] = 0; + if( strlen( parameter ) < 15 ) //|| parameter[ 14 ] != ' ' ) + return 0; + for( i = 0; i < 14; i ++ ) + if( ! isdigit( parameter[ i ])) + return 0; + for( i = 14; i < 18; i ++ ) + if( parameter[ i ] == ' ' ) + break; + else if( ! isdigit( parameter[ i ])) + return 0; + if( i == 18 ) + return 0; + i ++ ; + + strncpy( dt, parameter, 14 ); + dt[ 14 ] = 0; + * psecond = atoi( dt + 12 ); + dt[ 12 ] = 0; + * pminute = atoi( dt + 10 ); + dt[ 10 ] = 0; + * phour = atoi( dt + 8 ); + dt[ 8 ] = 0; + * pday = atoi( dt + 6 ); + dt[ 6 ] = 0 ; + * pmonth = atoi( dt + 4 ); + dt[ 4 ] = 0 ; + * pyear = atoi( dt ); + strncpy( dt, parameter, 14 ); + DEBUG_PRINT( F(" Modification time: ") ); DEBUG_PRINT( * pyear ); DEBUG_PRINT( F("/") ); DEBUG_PRINT( int(* pmonth) ); DEBUG_PRINT( F("/") ); DEBUG_PRINT( int(* pday) ); + DEBUG_PRINT( F(" ") ); DEBUG_PRINT( int(* phour) ); DEBUG_PRINT( F(":") ); DEBUG_PRINT( int(* pminute) ); DEBUG_PRINT( F(":") ); DEBUG_PRINT( int(* psecond) ); + DEBUG_PRINT( F(" of file: ") ); DEBUG_PRINTLN( (char *) ( parameter + i ) ); + + return i; +} + +// Create string YYYYMMDDHHMMSS from date and time +// +// parameters: +// date, time +// tstr: where to store the string. Must be at least 15 characters long +// +// return: +// pointer to tstr + +char * FtpServer::makeDateTimeStr( char * tstr, uint16_t date, uint16_t time ) +{ + sprintf( tstr, "%04u%02u%02u%02u%02u%02u", + (( date & 0xFE00 ) >> 9 ) + 1980, ( date & 0x01E0 ) >> 5, date & 0x001F, + ( time & 0xF800 ) >> 11, ( time & 0x07E0 ) >> 5, ( time & 0x001F ) << 1 ); + return tstr; +} + + +uint32_t FtpServer::fileSize( FTP_FILE file ) { +#if (STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_LITTLEFS || STORAGE_TYPE == STORAGE_FFAT || STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC || STORAGE_TYPE == STORAGE_SEEED_SD) + return file.size(); +#else + return file.fileSize(); +#endif +} + +#if (STORAGE_TYPE == STORAGE_SEEED_SD) + bool FtpServer::openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ){ + DEBUG_PRINT(F("File to open ") ); + DEBUG_PRINT( path ); + DEBUG_PRINT(F(" readType ") ); + DEBUG_PRINTLN(readTypeInt); + + if (readTypeInt == 0X01) { + readTypeInt = FILE_READ; + }else { + readTypeInt = FILE_WRITE; + } + + file = STORAGE_MANAGER.open( path, readTypeInt ); + if (!file) { // && readTypeInt[0]==FILE_READ) { + return false; + }else{ + DEBUG_PRINTLN("TRUE"); + + return true; + } +} +#elif ((STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC) && defined(ESP8266))// FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_ESP8266_242) + bool FtpServer::openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ){ + DEBUG_PRINT(F("File to open ") ); + DEBUG_PRINT( path ); + DEBUG_PRINT(F(" readType ") ); + DEBUG_PRINTLN(readTypeInt); + + if (readTypeInt == 0X01) { + readTypeInt = FILE_READ; + }else { + readTypeInt = FILE_WRITE; + } + + file = STORAGE_MANAGER.open( path, readTypeInt ); + if (!file) { // && readTypeInt[0]==FILE_READ) { + return false; + }else{ + DEBUG_PRINTLN("TRUE"); + + return true; + } +} +#elif (STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_LITTLEFS || STORAGE_TYPE == STORAGE_FFAT ) + bool FtpServer::openFile( const char * path, const char * readType ) { + DEBUG_PRINT(F("File to open ") ); + DEBUG_PRINT( path ); + DEBUG_PRINT(F(" readType ") ); + DEBUG_PRINTLN(readType); + file = STORAGE_MANAGER.open( path, readType ); + if (!file && readType[0]=='r') { + return false; + }else{ + DEBUG_PRINTLN("TRUE"); + + return true; + } + } +#elif STORAGE_TYPE <= STORAGE_SDFAT2 || STORAGE_TYPE == STORAGE_SPIFM || ((STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC) && ARDUINO_ARCH_SAMD) + bool FtpServer::openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ){ + DEBUG_PRINT(F("File to open ") ); + DEBUG_PRINT( path ); + DEBUG_PRINT(F(" readType ") ); + DEBUG_PRINTLN(readTypeInt); + + file = STORAGE_MANAGER.open( path, readTypeInt ); + if (!file) { + return false; + }else{ + DEBUG_PRINTLN("TRUE"); + + return true; + } +} + +#else + bool FtpServer::openFile( char path[ FTP_CWD_SIZE ], const char * readType ) { + return openFile( (const char*) path, readType ); + } + bool FtpServer::openFile( const char * path, const char * readType ) { + DEBUG_PRINT(F("File to open ") ); + DEBUG_PRINT( path ); + DEBUG_PRINT(F(" readType ") ); + DEBUG_PRINTLN(readType); + #if ((STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC) && !defined(ESP32)) + if (readType == 0X01) { + readType = FILE_READ; + }else { + readType = FILE_WRITE; + } + #endif + file = STORAGE_MANAGER.open( path, readType ); + if (!file && readType[0]=='r') { + return false; + }else{ + DEBUG_PRINTLN("TRUE"); + + return true; + } + } +#endif + +// Return true if path points to a directory +bool FtpServer::isDir( char * path ) +{ +#if (STORAGE_TYPE == STORAGE_LITTLEFS && (defined(ESP8266) || defined(ARDUINO_ARCH_RP2040))) + FTP_DIR dir; + bool res; + dir = STORAGE_MANAGER.openDir( path ); + + res = true; + return res; + + #elif STORAGE_TYPE == STORAGE_SPIFFS + if (strcmp(path, "/") == 0) { return true; } + return false; // no directory support +#elif STORAGE_TYPE == STORAGE_SEEED_SD || STORAGE_TYPE == STORAGE_FFAT || (STORAGE_TYPE == STORAGE_LITTLEFS && defined(ESP32)) + FTP_DIR dir; + bool res; + dir = STORAGE_MANAGER.open( path ); + +// return true; + res = dir.isDirectory(); + return res; +#elif STORAGE_TYPE == STORAGE_FATFS + return STORAGE_MANAGER.isDir( path ); +#elif STORAGE_TYPE == STORAGE_SDFAT1 || STORAGE_TYPE == STORAGE_SDFAT2 +// bool res = (!dir.open(path, FTP_FILE_READ) || !dir.isDir()); +// dir.close(); +// return res; + if (strcmp(path, "/") == 0) { return true; } + if( ! openFile( path, FTP_FILE_READ )) { + return false; + } + return true; +#else + FTP_FILE file; + bool res; + + if( ! openFile( path, FTP_FILE_READ )) { + return false; + } +#if STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC +// if (strcmp(path, "/") == 0) return true; +// res = file.isDirectory(); +// DEBUG_PRINT(path); +// DEBUG_PRINT(" IS DIRECOTORY --> "); +// DEBUG_PRINTLN(res); + return true; +#else +// res = file.isDir(); +// DEBUG_PRINT("IS DIRECTORY --> " ); +// DEBUG_PRINTLN(res); +#endif + file.close(); + return res; +#endif +} + +bool FtpServer::timeStamp( char * path, uint16_t year, uint8_t month, uint8_t day, + uint8_t hour, uint8_t minute, uint8_t second ) +{ +#if STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_LITTLEFS || STORAGE_TYPE == STORAGE_FFAT || STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC || STORAGE_TYPE == STORAGE_SEEED_SD +// struct tm tmDate = { second, minute, hour, day, month, year }; +// time_t rawtime = mktime(&tmDate); + + return true; + // setTime(rawtime); + // SPIFFS USE time() call +// return STORAGE_MANAGER.timeStamp( path, year, month, day, hour, minute, second ); +#elif STORAGE_TYPE == STORAGE_FATFS + return STORAGE_MANAGER.timeStamp( path, year, month, day, hour, minute, second ); +#else + FTP_FILE file; + bool res; + + if( ! openFile( path, FTP_FILE_READ_WRITE )) + return false; + res = file.timestamp( T_WRITE, year, month, day, hour, minute, second ); + file.close(); + return res; +#endif +} + +bool FtpServer::getFileModTime( char * path, uint16_t * pdate, uint16_t * ptime ) +{ +#if STORAGE_TYPE == STORAGE_FATFS + return STORAGE_MANAGER.getFileModTime( path, pdate, ptime ); +#else +// FTP_FILE file; + bool res; + + if( ! openFile( path, FTP_FILE_READ )) { + return false; + } + res = getFileModTime( pdate, ptime ); + file.close(); + return res; +#endif +} + +// Assume SD library is SdFat (or family) and file is open + +#if STORAGE_TYPE != STORAGE_FATFS +bool FtpServer::getFileModTime( uint16_t * pdate, uint16_t * ptime ) +{ +#if STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_LITTLEFS || STORAGE_TYPE == STORAGE_FFAT + #if defined(ESP8266) || defined(ARDUINO_ARCH_RP2040) + return dir.fileTime(); + #else + return dir.getLastWrite(); + #endif +#elif STORAGE_TYPE == STORAGE_SDFAT1 + dir_t d; + + if( ! file.dirEntry( & d )) + return false; + * pdate = d.lastWriteDate; + * ptime = d.lastWriteTime; + return true; +#elif STORAGE_TYPE == STORAGE_SDFAT2 || STORAGE_TYPE == STORAGE_SPIFM + return file.getModifyDateTime( pdate, ptime ); +#endif + return false; +} +#endif + +#if STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC + bool FtpServer::rename( const char * path, const char * newpath ){ + + FTP_FILE myFileIn = STORAGE_MANAGER.open(path, FILE_READ); + FTP_FILE myFileOut = STORAGE_MANAGER.open(newpath, FILE_WRITE); + + if(myFileOut) { + while (myFileIn.available() > 0) + { + int i = myFileIn.readBytes((char*)buf, FTP_BUF_SIZE); + myFileOut.write(buf, i); + } + // done, close the destination file + myFileOut.close(); + myFileOut = STORAGE_MANAGER.open(newpath, FILE_READ); + + } + bool operation = false; + + DEBUG_PRINT(F("RENAME --> ")); + DEBUG_PRINT(myFileIn.size()); + DEBUG_PRINT(" size "); + DEBUG_PRINTLN(myFileOut.size()); + + if (myFileIn.size() == myFileOut.size()) { + operation = true; + } + + + if (!operation) return operation; + + myFileIn.close(); + myFileOut.close(); + + return remove( path ); + }; +#endif diff --git a/lib/SimpleFTPServer/FtpServer.h b/lib/SimpleFTPServer/FtpServer.h new file mode 100644 index 00000000..9bc5aa59 --- /dev/null +++ b/lib/SimpleFTPServer/FtpServer.h @@ -0,0 +1,758 @@ +/* + * FtpServer Arduino, esp8266 and esp32 library for Ftp Server + * Derived form Jean-Michel Gallego version + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + + +/******************************************************************************* + ** ** + ** DEFINITIONS FOR FTP SERVER ** + ** ** + *******************************************************************************/ + +#include + +#ifndef FTP_SERVER_H +#define FTP_SERVER_H + +#define FTP_SERVER_VERSION "2.1.6 (2023-02-02)" + +#if ARDUINO >= 100 +#include "Arduino.h" +#else +#include "WProgram.h" +#endif + +// +//#if(NETWORK_ESP8266_SD == DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP8266) +// #define ESP8266_GT_2_4_2_SD_STORAGE_SELECTED +// #define DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP8266 NETWORK_ESP8266 +//#endif + +#if !defined(FTP_SERVER_NETWORK_TYPE) +// select Network type based + #if defined(ESP8266) || defined(ESP31B) + #if(NETWORK_ESP8266_242 == DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP8266) + #define ARDUINO_ESP8266_RELEASE_2_4_2 + + #define FTP_SERVER_NETWORK_TYPE_SELECTED NETWORK_ESP8266_242 + + #define FTP_SERVER_NETWORK_TYPE NETWORK_ESP8266 + #else + #define FTP_SERVER_NETWORK_TYPE DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP8266 + #endif + + #define STORAGE_TYPE DEFAULT_STORAGE_TYPE_ESP8266 + #elif defined(ESP32) + #define FTP_SERVER_NETWORK_TYPE DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP32 + #define STORAGE_TYPE DEFAULT_STORAGE_TYPE_ESP32 + #elif defined(ARDUINO_ARCH_STM32) + #define FTP_SERVER_NETWORK_TYPE DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 + #define STORAGE_TYPE DEFAULT_STORAGE_TYPE_STM32 + #elif defined(ARDUINO_ARCH_RP2040) + #define FTP_SERVER_NETWORK_TYPE DEFAULT_FTP_SERVER_NETWORK_TYPE_RP2040 + #define STORAGE_TYPE DEFAULT_STORAGE_TYPE_RP2040 + #elif defined(ARDUINO_ARCH_SAMD) + #define FTP_SERVER_NETWORK_TYPE DEFAULT_FTP_SERVER_NETWORK_TYPE_SAMD + #define STORAGE_TYPE DEFAULT_STORAGE_TYPE_SAMD + #else + #define FTP_SERVER_NETWORK_TYPE DEFAULT_FTP_SERVER_NETWORK_TYPE_ARDUINO + #define STORAGE_TYPE DEFAULT_STORAGE_TYPE_ARDUINO + // #define STORAGE_SD_ENABLED + #endif +#endif + +#ifndef FTP_SERVER_NETWORK_TYPE_SELECTED + #define FTP_SERVER_NETWORK_TYPE_SELECTED FTP_SERVER_NETWORK_TYPE +#endif + + +#if defined(ESP8266) || defined(ESP31B) + #ifndef STORAGE_SD_FORCE_DISABLE + #define STORAGE_SD_ENABLED + #endif + #ifndef STORAGE_SPIFFS_FORCE_DISABLE + #define STORAGE_SPIFFS_ENABLED + #endif +#elif defined(ESP32) + #ifndef STORAGE_SD_FORCE_DISABLE + #define STORAGE_SD_ENABLED + #endif + #ifndef STORAGE_SPIFFS_FORCE_DISABLE + #define STORAGE_SPIFFS_ENABLED + #endif +#else + #ifndef STORAGE_SD_FORCE_DISABLE + #define STORAGE_SD_ENABLED + #endif +#endif + + +// Includes and defined based on Network Type +#if(FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_ASYNC) + + // Note: + // No SSL/WSS support for client in Async mode + // TLS lib need a sync interface! + + #if defined(ESP8266) + #include + //#include + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + //#define FTP_CLIENT_NETWORK_SSL_CLASS WiFiClientSecure + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + + #elif defined(ESP32) + #include + //#include + + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + //#define FTP_CLIENT_NETWORK_SSL_CLASS WiFiClientSecure + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + #elif defined(ESP31B) + #include + + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + //#define FTP_CLIENT_NETWORK_SSL_CLASS WiFiClientSecure + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + #else + #error "network type ESP8266 ASYNC only possible on the ESP mcu!" + #endif + +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266 || FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_242) + + #if !defined(ESP8266) && !defined(ESP31B) + #error "network type ESP8266 only possible on the ESP mcu!" + #endif + + #ifdef ESP8266 + #include + #else + #include + #endif + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + //#define FTP_CLIENT_NETWORK_SSL_CLASS WiFiClientSecure + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + #define NET_CLASS WiFi +// #define CommandIs( a ) (command != NULL && ! strcmp_P( command, PSTR( a ))) +// #define ParameterIs( a ) ( parameter != NULL && ! strcmp_P( parameter, PSTR( a ))) +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_ETHERNET_GENERIC) + + #include + #include + #define FTP_CLIENT_NETWORK_CLASS EthernetClient + #define FTP_SERVER_NETWORK_SERVER_CLASS EthernetServer + #define NET_CLASS Ethernet + +// #if defined(ESP8266) || defined(ESP32) +// #define CommandIs( a ) (command != NULL && ! strcmp_P( command, PSTR( a ))) +// #define ParameterIs( a ) ( parameter != NULL && ! strcmp_P( parameter, PSTR( a ))) +// #else +// #define CommandIs( a ) ( ! strcmp_PF( command, PSTR( a ))) +// #define ParameterIs( a ) ( ! strcmp_PF( parameter, PSTR( a ))) +// #endif +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_W5100 || FTP_SERVER_NETWORK_TYPE == NETWORK_ETHERNET_ENC) + + #include + #include + #define FTP_CLIENT_NETWORK_CLASS EthernetClient + #define FTP_SERVER_NETWORK_SERVER_CLASS EthernetServer + #define NET_CLASS Ethernet + +// #if defined(ESP8266) || defined(ESP32) +// #define CommandIs( a ) (command != NULL && ! strcmp_P( command, PSTR( a ))) +// #define ParameterIs( a ) ( parameter != NULL && ! strcmp_P( parameter, PSTR( a ))) +// #else +// #define CommandIs( a ) ( ! strcmp_PF( command, PSTR( a ))) +// #define ParameterIs( a ) ( ! strcmp_PF( parameter, PSTR( a ))) +// #endif +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_ENC28J60 || FTP_SERVER_NETWORK_TYPE == NETWORK_UIPETHERNET) + + #include + + #define FTP_CLIENT_NETWORK_CLASS UIPClient + #define FTP_SERVER_NETWORK_SERVER_CLASS UIPServer + #define NET_CLASS Ethernet +// #if define(ESP8266) || define(ESP32) +// #define CommandIs( a ) (command != NULL && ! strcmp_P( command, PSTR( a ))) +// #define ParameterIs( a ) ( parameter != NULL && ! strcmp_P( parameter, PSTR( a ))) +// #else +// #define CommandIs( a ) ( ! strcmp_PF( command, PSTR( a ))) +// #define ParameterIs( a ) ( ! strcmp_PF( parameter, PSTR( a ))) +// #endif +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_ETHERNET_LARGE) + + #include + #include + #define FTP_CLIENT_NETWORK_CLASS EthernetClient + #define FTP_SERVER_NETWORK_SERVER_CLASS EthernetServer + #define NET_CLASS Ethernet + +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_ETHERNET_STM) + + #include + #include + #define FTP_CLIENT_NETWORK_CLASS EthernetClient + #define FTP_SERVER_NETWORK_SERVER_CLASS EthernetServer + #define NET_CLASS Ethernet + +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_ESP32) + + #include + //#include + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + //#define FTP_CLIENT_NETWORK_SSL_CLASS WiFiClientSecure + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + #define NET_CLASS WiFi +// #define CommandIs( a ) (command != NULL && ! strcmp_P( command, PSTR( a ))) +// #define ParameterIs( a ) ( parameter != NULL && ! strcmp_P( parameter, PSTR( a ))) +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_ESP32_ETH) + + #include + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + #define NET_CLASS Ethernet +// #define CommandIs( a ) ( ! strcmp_PF( command, PSTR( a ))) +// #define ParameterIs( a ) ( ! strcmp_PF( parameter, PSTR( a ))) +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_WiFiNINA) + + #include + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + //#define FTP_CLIENT_NETWORK_SSL_CLASS WiFiSSLClient + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + #define NET_CLASS WiFi +// #define CommandIs( a ) ( ! strcmp_PF( command, PSTR( a ))) +// #define ParameterIs( a ) ( ! strcmp_PF( parameter, PSTR( a ))) +#elif(FTP_SERVER_NETWORK_TYPE == NETWORK_SEEED_RTL8720DN) + + #include + #define FTP_CLIENT_NETWORK_CLASS WiFiClient + //#define FTP_CLIENT_NETWORK_SSL_CLASS WiFiSSLClient + #define FTP_SERVER_NETWORK_SERVER_CLASS WiFiServer + #define NET_CLASS WiFi +// #define CommandIs( a ) ( ! strcmp_PF( command, PSTR( a ))) +// #define ParameterIs( a ) ( ! strcmp_PF( parameter, PSTR( a ))) +#else + #error "no network type selected!" +#endif + +#if defined(ESP8266) || defined(ESP32) || defined(ARDUINO_ARCH_RP2040) + #define CommandIs( a ) (command != NULL && ! strcmp_P( command, PSTR( a ))) + #define ParameterIs( a ) ( parameter != NULL && ! strcmp_P( parameter, PSTR( a ))) +#else + #define CommandIs( a ) ( ! strcmp_PF( command, PSTR( a ))) + #define ParameterIs( a ) ( ! strcmp_PF( parameter, PSTR( a ))) +#endif + +#if(STORAGE_TYPE == STORAGE_SPIFFS) + #if defined(ESP32) +// #define FS_NO_GLOBALS + #include + + #define FTP_FILE File + #define FTP_DIR File + #else + #ifdef ARDUINO_ESP8266_RELEASE_2_4_2 + #define FS_NO_GLOBALS + #include "FS.h" + #define FTP_FILE fs::File + #define FTP_DIR fs::Dir + #else + #include "FS.h" + #define FTP_FILE File + #define FTP_DIR Dir + #endif + + #endif + +#if ESP8266 + #define FTP_FILE_READ "r" + #define FTP_FILE_READ_ONLY "r" + #define FTP_FILE_READ_WRITE "w+" + #define FTP_FILE_WRITE_APPEND "a+" + #define FTP_FILE_WRITE_CREATE "w+" +#else + #define FTP_FILE_READ "r" + #define FTP_FILE_READ_ONLY "r" + #define FTP_FILE_READ_WRITE "w" + #define FTP_FILE_WRITE_APPEND "a" + #define FTP_FILE_WRITE_CREATE "w" +#endif + + #define STORAGE_MANAGER SPIFFS + + #define FILENAME_LENGTH 32 +#elif(STORAGE_TYPE == STORAGE_FFAT) + #include "FS.h" + #include "FFat.h" + + #define STORAGE_MANAGER FFat + + #define FTP_FILE File + #define FTP_DIR File + + #define FTP_FILE_READ "r" + #define FTP_FILE_READ_ONLY "r" + #define FTP_FILE_READ_WRITE "w" + #define FTP_FILE_WRITE_APPEND "a" + #define FTP_FILE_WRITE_CREATE "w" + + #define FILENAME_LENGTH 255 +#elif(STORAGE_TYPE == STORAGE_LITTLEFS) + #if ESP8266 || ARDUINO_ARCH_RP2040 + #include "LittleFS.h" + #define STORAGE_MANAGER LittleFS + #define FTP_FILE File + #define FTP_DIR Dir + + #define FTP_FILE_READ "r" + #define FTP_FILE_READ_ONLY "r" + #define FTP_FILE_READ_WRITE "w+" + #define FTP_FILE_WRITE_APPEND "a+" + #define FTP_FILE_WRITE_CREATE "w+" + #else +#ifdef ESP32 + #if ESP_ARDUINO_VERSION_MAJOR >= 2 + #include "FS.h" + #include "LittleFS.h" + #define STORAGE_MANAGER LittleFS + #else + #include "LITTLEFS.h" + #define STORAGE_MANAGER LITTLEFS + #endif +#else + #include "LittleFS.h" + #define STORAGE_MANAGER LittleFS +#endif + #define FTP_FILE File + #define FTP_DIR File + + #define FTP_FILE_READ "r" + #define FTP_FILE_READ_ONLY "r" + #define FTP_FILE_READ_WRITE "w" + #define FTP_FILE_WRITE_APPEND "a" + #define FTP_FILE_WRITE_CREATE "w" + #endif + #define FILENAME_LENGTH 32 +#elif(STORAGE_TYPE == STORAGE_SD) + #include + #include + + #define STORAGE_MANAGER SD + #define FTP_FILE File + #define FTP_DIR File + + #define FTP_FILE_READ FILE_READ + #define FTP_FILE_READ_ONLY FILE_READ +#ifdef ESP32 + #define FTP_FILE_READ_WRITE FILE_WRITE + #define FTP_FILE_WRITE_APPEND FILE_APPEND +#else + #define FTP_FILE_READ_WRITE FILE_WRITE + #define FTP_FILE_WRITE_APPEND FILE_WRITE +#endif + #define FTP_FILE_WRITE_CREATE FILE_WRITE + + #define FILENAME_LENGTH 255 +#elif(STORAGE_TYPE == STORAGE_SD_MMC) + #include + #include + + #define STORAGE_MANAGER SD_MMC + #define FTP_FILE File + #define FTP_DIR File + + #define FTP_FILE_READ FILE_READ + #define FTP_FILE_READ_ONLY FILE_READ + #define FTP_FILE_READ_WRITE FILE_WRITE +#ifdef ESP32 + #define FTP_FILE_READ_WRITE FILE_WRITE + #define FTP_FILE_WRITE_APPEND FILE_APPEND +#else + #define FTP_FILE_READ_WRITE FILE_WRITE + #define FTP_FILE_WRITE_APPEND FILE_WRITE +#endif + #define FTP_FILE_WRITE_CREATE FILE_WRITE + + #define FILENAME_LENGTH 255 +#elif(STORAGE_TYPE == STORAGE_SEEED_SD) + #include + #define STORAGE_MANAGER SD + + #include "SD/Seeed_SD.h" + + + +// #define STORAGE_MANAGER SPIFLASH +// #include "SFUD/Seeed_SFUD.h" + + #define FTP_FILE File + #define FTP_DIR File + + #define FTP_FILE_READ FILE_READ + #define FTP_FILE_READ_ONLY FILE_READ + #define FTP_FILE_READ_WRITE FILE_WRITE + #define FTP_FILE_WRITE_APPEND FILE_APPEND + #define FTP_FILE_WRITE_CREATE FILE_WRITE + + #define FILENAME_LENGTH 255 + +#elif (STORAGE_TYPE == STORAGE_SDFAT1) + #include + #include + + #define STORAGE_MANAGER sd + #define FTP_FILE SdFile + #define FTP_DIR SdFile + extern SdFat STORAGE_MANAGER; + + #define FTP_FILE_READ O_READ + #define FTP_FILE_READ_ONLY O_RDONLY + #define FTP_FILE_READ_WRITE O_RDWR + #define FTP_FILE_WRITE_APPEND O_WRITE | O_APPEND + #define FTP_FILE_WRITE_CREATE O_WRITE | O_CREAT + #define FILENAME_LENGTH 255 + +#elif (STORAGE_TYPE == STORAGE_SDFAT2) + #include + #include + + #define STORAGE_MANAGER sd + #define FTP_FILE FsFile + #define FTP_DIR FsFile + extern SdFat STORAGE_MANAGER; + + #define FTP_FILE_READ O_READ + #define FTP_FILE_READ_ONLY O_RDONLY + #define FTP_FILE_READ_WRITE O_RDWR + #define FTP_FILE_WRITE_APPEND O_WRITE | O_APPEND + #define FTP_FILE_WRITE_CREATE O_WRITE | O_CREAT + #define FILENAME_LENGTH 255 +#elif (STORAGE_TYPE == STORAGE_SPIFM) + #include + #include + #include + + #define STORAGE_MANAGER fatfs + #define FTP_FILE File + #define FTP_DIR File + extern FatFileSystem STORAGE_MANAGER; + extern Adafruit_SPIFlash flash; + #define FTP_FILE_READ FILE_READ + #define FTP_FILE_READ_ONLY FILE_READ + #define FTP_FILE_READ_WRITE FILE_WRITE + #define FTP_FILE_WRITE_APPEND FILE_WRITE + #define FTP_FILE_WRITE_CREATE FILE_WRITE + #define FILENAME_LENGTH 255 +#elif (STORAGE_TYPE == STORAGE_FATFS) + #include + #include + + #define STORAGE_MANAGER sdff + #define FTP_FILE FileFs + #define FTP_DIR DirFs + extern FatFsClass STORAGE_MANAGER; + #define O_READ FA_READ + #define O_WRITE FA_WRITE + #define O_RDWR FA_READ | FA_WRITE + #define O_CREAT FA_CREATE_ALWAYS + #define O_APPEND FA_OPEN_APPEND + + #define FTP_FILE_READ O_READ + #define FTP_FILE_READ_ONLY O_RDONLY + #define FTP_FILE_READ_WRITE O_RDWR + #define FTP_FILE_WRITE_APPEND O_WRITE | O_APPEND + #define FTP_FILE_WRITE_CREATE O_WRITE | O_CREAT + #define FILENAME_LENGTH 255 +#endif + +//#ifdef FTP_CLIENT_NETWORK_SSL_CLASS +//#define FTP_CLIENT_NETWORK_CLASS FTP_CLIENT_NETWORK_SSL_CLASS +//#endif + +#define OPEN_CLOSE_SPIFFS +#define OPEN_CLOSE_SD + +// Setup debug printing macros. +#ifdef FTP_SERVER_DEBUG + #define DEBUG_PRINT(...) { DEBUG_PRINTER.print(__VA_ARGS__); } + #define DEBUG_PRINTLN(...) { DEBUG_PRINTER.println(__VA_ARGS__); } +#else + #define DEBUG_PRINT(...) {} + #define DEBUG_PRINTLN(...) {} +#endif + +#define FTP_CMD_PORT 21 // Command port on wich server is listening +#define FTP_DATA_PORT_DFLT 20 // Default data port in active mode +#define FTP_DATA_PORT_PASV 50009 // Data port in passive mode + +#define FF_MAX_LFN 255 // max size of a long file name +#define FTP_CMD_SIZE FF_MAX_LFN+8 // max size of a command +#define FTP_CWD_SIZE FF_MAX_LFN+8 // max size of a directory name +#define FTP_FIL_SIZE FF_MAX_LFN // max size of a file name +#define FTP_CRED_SIZE 16 // max size of username and password +#define FTP_NULLIP() IPAddress(0,0,0,0) + +enum ftpCmd { FTP_Stop = 0, // In this stage, stop any connection + FTP_Init, // initialize some variables + FTP_Client, // wait for client connection + FTP_User, // wait for user name + FTP_Pass, // wait for user password + FTP_Cmd }; // answers to commands + +enum ftpTransfer { FTP_Close = 0, // In this stage, close data channel + FTP_Retrieve, // retrieve file + FTP_Store, // store file + FTP_List, // list of files + FTP_Nlst, // list of name of files + FTP_Mlsd }; // listing for machine processing + +enum ftpDataConn { FTP_NoConn = 0,// No data connexion + FTP_Pasive, // Pasive type + FTP_Active }; // Active type + +enum FtpOperation { + FTP_CONNECT, + FTP_DISCONNECT, + FTP_FREE_SPACE_CHANGE +}; + +enum FtpTransferOperation { + FTP_UPLOAD_START = 0, + FTP_UPLOAD = 1, + + FTP_DOWNLOAD_START = 2, + FTP_DOWNLOAD = 3, + + + FTP_TRANSFER_STOP = 4, + FTP_DOWNLOAD_STOP = 4, + FTP_UPLOAD_STOP = 4, + + FTP_TRANSFER_ERROR = 5, + FTP_DOWNLOAD_ERROR = 5, + FTP_UPLOAD_ERROR = 5 +}; + +class FtpServer +{ +public: + FtpServer( uint16_t _cmdPort = FTP_CMD_PORT, uint16_t _pasvPort = FTP_DATA_PORT_PASV ); + + void begin( const char * _user, const char * _pass, const char * welcomeMessage = "Welcome to Simply FTP server" ); + void begin( const char * welcomeMessage = "Welcome to Simply FTP server" ); + + void end(); + void setLocalIp(IPAddress localIp); + void credentials( const char * _user, const char * _pass ); + uint8_t handleFTP(); + + void setCallback(void (*_callbackParam)(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace) ) + { + _callback = _callbackParam; + } + + void setTransferCallback(void (*_transferCallbackParam)(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize) ) + { + _transferCallback = _transferCallbackParam; + } + +private: + void (*_callback)(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){}; + void (*_transferCallback)(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){}; + + void iniVariables(); + void clientConnected(); + void disconnectClient(); + bool processCommand(); + bool haveParameter(); + int dataConnect( bool out150 = true ); + bool dataConnected(); + bool doRetrieve(); + bool doStore(); + bool doList(); + bool doMlsd(); + void closeTransfer(); + void abortTransfer(); + bool makePath( char * fullName, char * param = NULL ); + bool makeExistsPath( char * path, char * param = NULL ); + bool openDir( FTP_DIR * pdir ); + bool isDir( char * path ); + uint8_t getDateTime( char * dt, uint16_t * pyear, uint8_t * pmonth, uint8_t * pday, + uint8_t * phour, uint8_t * pminute, uint8_t * second ); + char * makeDateTimeStr( char * tstr, uint16_t date, uint16_t time ); + bool timeStamp( char * path, uint16_t year, uint8_t month, uint8_t day, + uint8_t hour, uint8_t minute, uint8_t second ); + bool getFileModTime( char * path, uint16_t * pdate, uint16_t * ptime ); +#if STORAGE_TYPE != STORAGE_FATFS + bool getFileModTime( uint16_t * pdate, uint16_t * ptime ); +#endif + int8_t readChar(); + + const char* getFileName(FTP_FILE *file){ + #if STORAGE_TYPE <= STORAGE_SDFAT2 + int max_characters = 100; + char f_name[max_characters]; + file->getName(f_name, max_characters); + String filename = String(f_name); + return filename.c_str(); + #elif STORAGE_TYPE == STORAGE_FATFS + return file->fileName(); + #else + return file->name(); + #endif + } + bool exists( const char * path ) { +#if STORAGE_TYPE == STORAGE_SPIFFS || (STORAGE_TYPE == STORAGE_SD && FTP_SERVER_NETWORK_TYPE == NETWORK_ESP8266_242) + if (strcmp(path, "/") == 0) return true; +#endif +#if STORAGE_TYPE == STORAGE_FFAT || (STORAGE_TYPE == STORAGE_LITTLEFS && defined(ESP32)) + FTP_DIR f = STORAGE_MANAGER.open(path, "r"); + return (f == true); +#else + return STORAGE_MANAGER.exists( path ); +#endif + }; + bool remove( const char * path ) { return STORAGE_MANAGER.remove( path ); }; +#if STORAGE_TYPE == STORAGE_SPIFFS + bool makeDir( const char * path ) { return false; }; + bool removeDir( const char * path ) { return false; }; +#else + bool makeDir( const char * path ) { return STORAGE_MANAGER.mkdir( path ); }; + bool removeDir( const char * path ) { return STORAGE_MANAGER.rmdir( path ); }; +#endif + +#if STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC + bool rename( const char * path, const char * newpath ); +#else + bool rename( const char * path, const char * newpath ) { return STORAGE_MANAGER.rename( path, newpath ); }; +#endif +#if (STORAGE_TYPE == STORAGE_SEEED_SD) + bool openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ); +#elif (STORAGE_TYPE == STORAGE_SD && defined(ESP8266))// FTP_SERVER_NETWORK_TYPE_SELECTED == NETWORK_ESP8266_242) + bool openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ); +#elif (STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_LITTLEFS || STORAGE_TYPE == STORAGE_FFAT ) + bool openFile( const char * path, const char * readType ); +// bool openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ); +#elif STORAGE_TYPE <= STORAGE_SDFAT2 || STORAGE_TYPE == STORAGE_SPIFM || (STORAGE_TYPE == STORAGE_SD && ARDUINO_ARCH_SAMD) + bool openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ); +#else + bool openFile( char path[ FTP_CWD_SIZE ], const char * readType ); + bool openFile( const char * path, const char * readType ); +// bool openFile( char path[ FTP_CWD_SIZE ], int readTypeInt ); +#endif +// bool openFile( char path[ FTP_CWD_SIZE ], const char * readType ); +// bool openFile( const char * path, const char * readType ); + uint32_t fileSize( FTP_FILE file ); + +#if STORAGE_TYPE == STORAGE_SPIFFS || STORAGE_TYPE == STORAGE_LITTLEFS +#if ESP8266 || ARDUINO_ARCH_RP2040 + uint32_t capacity() { + FSInfo fi; + STORAGE_MANAGER.info(fi); + + return fi.totalBytes >> 1; + }; + uint32_t free() { + FSInfo fi; + STORAGE_MANAGER.info(fi); + + return (fi.totalBytes - fi.usedBytes) >> 1; + }; +#else + uint32_t capacity() { + return STORAGE_MANAGER.totalBytes() >> 1; + }; + uint32_t free() { + return (STORAGE_MANAGER.totalBytes() - + STORAGE_MANAGER.usedBytes()) >> 1; + }; +#endif +#elif STORAGE_TYPE == STORAGE_SD || STORAGE_TYPE == STORAGE_SD_MMC + uint32_t capacity() { return true; }; + uint32_t free() { return true; }; +#elif STORAGE_TYPE == STORAGE_SEEED_SD + uint32_t capacity() { + return STORAGE_MANAGER.totalBytes() >> 1; + }; + uint32_t free() { + return (STORAGE_MANAGER.totalBytes() - + STORAGE_MANAGER.usedBytes()) >> 1; + }; +#elif STORAGE_TYPE == STORAGE_SDFAT1 + uint32_t capacity() { return STORAGE_MANAGER.card()->cardSize() >> 1; }; + uint32_t free() { return STORAGE_MANAGER.vol()->freeClusterCount() * + STORAGE_MANAGER.vol()->sectorsPerCluster() >> 1; }; +#elif STORAGE_TYPE == STORAGE_SDFAT2 + uint32_t capacity() { return STORAGE_MANAGER.card()->sectorCount() >> 1; }; + uint32_t free() { return STORAGE_MANAGER.vol()->freeClusterCount() * + STORAGE_MANAGER.vol()->sectorsPerCluster() >> 1; }; +#elif STORAGE_TYPE == STORAGE_SPIFM + uint32_t capacity() { return flash.size() >> 10; }; + uint32_t free() { return 0; }; // TODO // +#elif STORAGE_TYPE == STORAGE_FATFS + uint32_t capacity() { return STORAGE_MANAGER.capacity(); }; + uint32_t free() { return STORAGE_MANAGER.free(); }; +#elif STORAGE_TYPE == STORAGE_FFAT + uint32_t capacity() { return STORAGE_MANAGER.totalBytes(); }; + uint32_t free() { return STORAGE_MANAGER.freeBytes(); }; +#endif + bool legalChar( char c ) // Return true if char c is allowed in a long file name + { + if( c == '"' || c == '*' || c == '?' || c == ':' || + c == '<' || c == '>' || c == '|' ) + return false; +#if STORAGE_TYPE == STORAGE_FATFS + return 0x1f < c && c < 0xff; +#else + return 0x1f < c && c < 0x7f; +#endif + } + + IPAddress localIp; // IP address of server as seen by clients + IPAddress dataIp; // IP address of client for data + FTP_SERVER_NETWORK_SERVER_CLASS ftpServer; + FTP_SERVER_NETWORK_SERVER_CLASS dataServer; + + + FTP_CLIENT_NETWORK_CLASS client; + FTP_CLIENT_NETWORK_CLASS data; + + FTP_FILE file; + FTP_DIR dir; + + ftpCmd cmdStage; // stage of ftp command connexion + ftpTransfer transferStage; // stage of data connexion + ftpDataConn dataConn; // type of data connexion + + bool anonymousConnection = false; + + uint8_t __attribute__((aligned(4))) // need to be aligned to 32bit for Esp8266 SPIClass::transferBytes() + buf[ FTP_BUF_SIZE ]; // data buffer for transfers + char cmdLine[ FTP_CMD_SIZE ]; // where to store incoming char from client + char cwdName[ FTP_CWD_SIZE ]; // name of current directory + char rnfrName[ FTP_CWD_SIZE ]; // name of file for RNFR command + const char * user; // user name + const char * pass; // password + char command[ 5 ]; // command sent by client + bool rnfrCmd; // previous command was RNFR + char * parameter; // point to begin of parameters sent by client + const char * welcomeMessage; + uint16_t cmdPort, + pasvPort, + dataPort; + uint16_t iCL; // pointer to cmdLine next incoming char + uint16_t nbMatch; + + uint32_t millisDelay, // + millisEndConnection, // + millisBeginTrans, // store time of beginning of a transaction + bytesTransfered; // +}; + +#endif // FTP_SERVER_H diff --git a/lib/SimpleFTPServer/FtpServerKey.h b/lib/SimpleFTPServer/FtpServerKey.h new file mode 100644 index 00000000..6a0cbd1c --- /dev/null +++ b/lib/SimpleFTPServer/FtpServerKey.h @@ -0,0 +1,128 @@ +/* + * FtpServer Arduino, esp8266 and esp32 library for Ftp Server + * Derived form Jean-Michel Gallego version + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +/******************************************************************************* + ** ** + ** SETTINGS FOR FTP SERVER ** + ** ** + *******************************************************************************/ + +#ifndef FTP_SERVER_CONFIG_H +#define FTP_SERVER_CONFIG_H + +// Uncomment to enable printing out nice debug messages. +// #define FTP_SERVER_DEBUG +// #define FTP_ADDITIONAL_DEBUG + +// Define where debug output will be printed. +#define DEBUG_PRINTER Serial + +#define STORAGE_SDFAT1 1 // Library SdFat version 1.4.x +#define STORAGE_SDFAT2 2 // Library SdFat version >= 2.0.2 +#define STORAGE_SPIFM 3 // Libraries Adafruit_SPIFlash and SdFat-Adafruit-Fork +#define STORAGE_FATFS 4 // Library FatFs +#define STORAGE_SD 5 // Standard SD library (suitable for Arduino esp8266 and esp32 +#define STORAGE_SPIFFS 6 // SPIFFS +#define STORAGE_LITTLEFS 7 // LITTLEFS +#define STORAGE_SEEED_SD 8 // Seeed_SD library +#define STORAGE_FFAT 9 // ESP32 FFAT +#define STORAGE_SD_MMC 10 // SD_MMC library + +#define NETWORK_ESP8266_ASYNC (1) +#define NETWORK_ESP8266 (2) // Standard ESP8266WiFi +#define NETWORK_ESP8266_242 (3) // ESP8266WiFi before 2.4.2 core +#define NETWORK_W5100 (4) // Standard Arduino Ethernet library +#define NETWORK_ETHERNET (4) // Standard Arduino Ethernet library +#define NETWORK_ENC28J60 (5) // UIPEthernet library +#define NETWORK_ESP32 (6) // Standard WiFi library +#define NETWORK_RP2040_WIFI (6) // Raspberry Pi Pico W standard WiFi library +#define NETWORK_ESP32_ETH (7) // Standard ETH library +#define NETWORK_WiFiNINA (8) // Standard WiFiNINA library +#define NETWORK_SEEED_RTL8720DN (9) // Standard SEED WiFi library +#define NETWORK_ETHERNET_LARGE (10) +#define NETWORK_ETHERNET_ENC (11) // EthernetENC library (evolution of UIPEthernet +#define NETWORK_ETHERNET_STM (12) +#define NETWORK_UIPETHERNET (13) // UIPEthernet library same of NETWORK_ENC28J60 +#define NETWORK_ETHERNET_GENERIC (14) // Ethernet generic + +// esp8266 configuration +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP8266 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP8266 NETWORK_ESP8266 + #define DEFAULT_STORAGE_TYPE_ESP8266 STORAGE_LITTLEFS +#endif +// esp32 configuration +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP32 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP32 NETWORK_ESP32 + #define DEFAULT_STORAGE_TYPE_ESP32 STORAGE_LITTLEFS + /** +To use Ethernet.h with esp32 fix would be to change in Ethernet.h the line +class EthernetServer : public Server { +to +class EthernetServer : public Stream { + +or + +in \esp32\2.0.6\cores\esp32\Server.h +A workaround is to change line 28 of the ESP32 core's Server.h from: + virtual void begin(uint16_t port=0) =0; +to + virtual void begin() =0; +However, the last one, that will break anything that uses the ESP32 WiFi library's WebServer class. + +https://github.com/arduino-libraries/Ethernet/issues/193 +https://github.com/arduino-libraries/Ethernet/issues/88 + * + */ +#endif +// Standard AVR Arduino configuration +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_ARDUINO + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_ARDUINO NETWORK_W5100 + #define DEFAULT_STORAGE_TYPE_ARDUINO STORAGE_SD +#endif +// STM32 configuration +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 NETWORK_W5100 + #define DEFAULT_STORAGE_TYPE_STM32 STORAGE_SDFAT2 +#endif +// Raspberry Pi Pico (rp2040) configuration +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_RP2040 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_RP2040 NETWORK_RP2040_WIFI + #define DEFAULT_STORAGE_TYPE_RP2040 STORAGE_LITTLEFS +#endif + +// Arduino SAMD21 like Arduino MKR Nano 33 IoT or Wio Terminal +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_ARDUINO_SAMD +// Wio Terminal +// #define DEFAULT_FTP_SERVER_NETWORK_TYPE_SAMD NETWORK_SEEED_RTL8720DN +// #define DEFAULT_STORAGE_TYPE_SAMD STORAGE_SEEED_SD + +// Arduino SAMD + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_SAMD NETWORK_WiFiNINA + #define DEFAULT_STORAGE_TYPE_SAMD STORAGE_SD +#endif + +#define UTF8_SUPPORT + +//#define SD_CS_PIN 4 +// Disconnect client after 5 minutes of inactivity (expressed in seconds) +#define FTP_TIME_OUT 5 * 60 + + +// Wait for authentication for 10 seconds (expressed in seconds) +#define FTP_AUTH_TIME_OUT 10 + + +// Size of file buffer for read/write +// Transfer speed depends of this value +// Best value depends on many factors: SD card, client side OS, ... +// But it can be reduced to 512 if memory usage is critical. +#define FTP_BUF_SIZE 1024 //2048 //1024 // 512 + +#endif // FTP_SERVER_CONFIG_H diff --git a/lib/SimpleFTPServer/LICENSE.md b/lib/SimpleFTPServer/LICENSE.md new file mode 100644 index 00000000..54153141 --- /dev/null +++ b/lib/SimpleFTPServer/LICENSE.md @@ -0,0 +1,24 @@ +The MIT License (MIT) + +Copyright (c) 2017 Renzo Mischianti www.mischianti.org All right reserved. + +You may copy, alter and reuse this code in any way you like, but please leave +reference to www.mischianti.org in your comments if you redistribute this code. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/lib/SimpleFTPServer/README.md b/lib/SimpleFTPServer/README.md new file mode 100644 index 00000000..17d083ec --- /dev/null +++ b/lib/SimpleFTPServer/README.md @@ -0,0 +1,108 @@ +# SimpleFTPServer + +[Instruction on FTP server on esp8266 and esp32](https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32) +[Simple FTP Server library now with support for Wio Terminal and SD](https://www.mischianti.org/2021/07/01/simple-ftp-server-library-now-with-support-for-wio-terminal-and-sd/) + +#### Simple FTP Server for + - Raspberry Pi Pico W (Flash: LittleFS) (To test SD and SdFat) + - esp8266 (Flash: SPIFFs, LittleFS. SD: SD, SdFat 2) + - esp32 (SPIFFS, LITTLEFS, FFAT, SD: SD, SdFat) + - stm32 (SdFat, SPI flash) + - Arduino (SD with 8.3 file format, SD: SD, SdFat 2) + - Wio Terminal (SdFat 2, Seed SD, and native FAT) + +#### Changelog +- 2022-02-02 2.1.6 Fix esp8266 Ethernet (w5x00) issue and explain solution for ESP32 Ethernet (w5x00), add new Networks management +- 2022-01-13 2.1.5 Fix SPIFM external SPI Flash date management (add SPIFM esp32 example) +- 2022-09-21 2.1.4 Add support for Raspberry Pi Pico W and rp2040 boards, Fix SD card config +- 2022-09-20 2.1.3 Soft AP IP management, more disconnect event and SD_MCC +- 2022-05-21 2.1.2 Fix SD path (#19) +- 2022-05-21 2.1.1 Minor fix +- 2022-03-30 2.1.0 Add UTF8 support and enabled It by default (Thanks to @plaber) +- 2022-03-30 2.0.0 Complete support for STM32 with SD and SPI Flash minor bux fix and HELP command support +- 2022-03-17 1.3.0 Fix enc28j60 and w5500 support and restructuring for local settings +- 2022-02-25 1.2.1 Fix anonymous user begin and fix SPIFFS wrong display +- 2022-02-22 1.2.0 Add anonymous user and implement correct RFC (#9 now work correctly with File Explorer) +- 2022-02-01 1.1.1 Add workaround to start FTP server before connection, add end and setLocalIP method. + + +

When I develop a new solution I'd like to divide the application in layer, and so I'd like focus my attention in only one aspect at time.

+ + + +

In detail I separate the REST layer (written inside the microcontroller) and the Front-End (written in Angular, React/Redux or vanilla JS), so I'd like to upload new web interface directly to the microcontroller via FTP.

+ + + +
+ + + +

For static information (Web pages for examples), that not change frequently, esp8266 or esp32 have internal SPIFFS (SPI Flash File System) and you can upload data via Arduino IDE as explained in the article "WeMos D1 mini (esp8266), integrated SPIFFS Filesystem" for esp8266 or "ESP32: integrated SPIFFS FileSystem" for esp32 or with LittleFS "WeMos D1 mini (esp8266), integrated LittleFS Filesystem" but for fast operation and future support It's usefully use FTP.

+ + + + +```cpp +/* + * FtpServer esp8266 and esp32 with SPIFFS + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#ifdef ESP8266 +#include +#elif defined ESP32 +#include +#include "SPIFFS.h" +#endif + +#include + +const char* ssid = "YOUR_SSID"; +const char* password = "YOUR_PASS"; + + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + + +void setup(void){ + Serial.begin(115200); + WiFi.begin(ssid, password); + Serial.println(""); + + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); + + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// +#ifdef ESP32 //esp32 we send true to format spiffs if cannot mount + if (SPIFFS.begin(true)) { +#elif defined ESP8266 + if (SPIFFS.begin()) { +#endif + Serial.println("SPIFFS opened!"); + ftpSrv.begin("esp8266","esp8266"); //username, password for ftp. set ports in ESP8266FtpServer.h (default 21, 50009 for PASV) + } +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! + // server.handleClient(); //example if running a webserver you still need to call .handleClient(); + +} +``` + +https://downloads.arduino.cc/libraries/logs/github.com/xreef/SimpleFTPServer/ \ No newline at end of file diff --git a/lib/SimpleFTPServer/SimpleFTPServer.h b/lib/SimpleFTPServer/SimpleFTPServer.h new file mode 100644 index 00000000..dc92d46b --- /dev/null +++ b/lib/SimpleFTPServer/SimpleFTPServer.h @@ -0,0 +1,18 @@ +/* + * FtpServer Arduino, esp8266 and esp32 library for Ftp Server + * Derived form https://github.com/nailbuster/esp8266FTPServer + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#ifndef SIMPLE_FTP_SERVER_H +#define SIMPLE_FTP_SERVER_H + +#include + +#endif + +#pragma once diff --git a/lib/SimpleFTPServer/examples/Arduino_Ethernet/Arduino_Ethernet.ino b/lib/SimpleFTPServer/examples/Arduino_Ethernet/Arduino_Ethernet.ino new file mode 100644 index 00000000..af178521 --- /dev/null +++ b/lib/SimpleFTPServer/examples/Arduino_Ethernet/Arduino_Ethernet.ino @@ -0,0 +1,62 @@ +/* + * FtpServer Arduino with Ethernet library and w5100 shield + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#include +#include +#include "SD.h" + +#include + +// Enter a MAC address for your controller below. +// Newer Ethernet shields have a MAC address printed on a sticker on the shield +byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xE1 }; + +// Set the static IP address to use if the DHCP fails to assign +byte macAddr[] = {0x5e, 0xa4, 0x18, 0xf0, 0x8a, 0xf2}; +IPAddress arduinoIP(192, 168, 1, 177); +IPAddress dnsIP(192, 168, 1, 1); +IPAddress gatewayIP(192, 168, 1, 1); +IPAddress subnetIP(255, 255, 255, 0); + +FtpServer ftpSrv; + +void setup(void){ + Serial.begin(115200); + delay(2000); + // If other chips are connected to SPI bus, set to high the pin connected + // to their CS before initializing Flash memory + pinMode( 4, OUTPUT ); + digitalWrite( 4, HIGH ); + pinMode( 10, OUTPUT ); + digitalWrite( 10, HIGH ); + + Serial.print("Starting SD."); + while (!SD.begin(4)) { + Serial.print("."); + } + Serial.println("finish!"); + + // start the Ethernet connection: + Serial.print("Starting ethernet."); + if (Ethernet.begin(mac) == 0) { + Serial.println("Failed to configure Ethernet using DHCP"); + Ethernet.begin(macAddr, arduinoIP, dnsIP, gatewayIP, subnetIP); + }else{ + Serial.println("ok to configure Ethernet using DHCP"); + } + + Serial.print("IP address "); + Serial.println(Ethernet.localIP()); + + Serial.println("SPIFFS opened!"); + ftpSrv.begin("esp8266","esp8266"); //username, password for ftp. +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! +} diff --git a/lib/SimpleFTPServer/examples/Arduino_Ethernet_SdFat2/Arduino_Ethernet_SdFat2.ino b/lib/SimpleFTPServer/examples/Arduino_Ethernet_SdFat2/Arduino_Ethernet_SdFat2.ino new file mode 100644 index 00000000..3892b1f6 --- /dev/null +++ b/lib/SimpleFTPServer/examples/Arduino_Ethernet_SdFat2/Arduino_Ethernet_SdFat2.ino @@ -0,0 +1,133 @@ +/* + * FtpServer Arduino with Ethernet library and w5100 shield + * With SdFat version > 2 (full name and more size) + * + * #ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_ARDUINO + * #define DEFAULT_FTP_SERVER_NETWORK_TYPE_ARDUINO NETWORK_W5100 + * #define DEFAULT_STORAGE_TYPE_ARDUINO STORAGE_SDFAT2 + * #endif + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#include +#include +#include +#include + +// Define Chip Select for your SD card according to hardware +// #define CS_SDCARD 4 // SD card reader of Ehernet shield +#define CS_SDCARD 4 // Chip Select for SD card reader on Due + +// Define Reset pin for W5200 or W5500 +// set to -1 for other ethernet chip or if Arduino reset board is used + #define W5x00_RESET -1 +//#define W5x00_RESET 8 // on Due +// #define W5x00_RESET 3 // on MKR + + +// Object for File system +SdFat sd; + +// Object for FtpServer +// Command port and Data port in passive mode can be defined here +// FtpServer ftpSrv( 221, 25000 ); +// FtpServer ftpSrv( 421 ); // Default data port in passive mode is 55600 +FtpServer ftpSrv; // Default command port is 21 ( !! without parenthesis !! ) + +// Mac address of ethernet adapter +// byte mac[] = { 0x90, 0xa2, 0xda, 0x00, 0x00, 0x00 }; +byte mac[] = { 0x00, 0xaa, 0xbb, 0xcc, 0xde, 0xef }; + +// IP address of FTP server +// if set to 0, use DHCP for the routeur to assign IP +// IPAddress serverIp( 192, 168, 1, 40 ); +IPAddress serverIp( 0, 0, 0, 0 ); + +// External IP address of FTP server +// In passive mode, when accessing the serveur from outside his subnet, it can be +// necessary with some clients to reply them with the server's external ip address +// IPAddress externalIp( 192, 168, 1, 2 ); + +ArduinoOutStream cout( Serial ); + + +void setup() +{ + Serial.begin( 115200 ); + cout << F("=== Test of FTP Server with SdFat ") << SD_FAT_VERSION << F(" file system ===") << endl; + + // If other chips are connected to SPI bus, set to high the pin connected + // to their CS before initializing Flash memory + pinMode( 4, OUTPUT ); + digitalWrite( 4, HIGH ); + pinMode( 10, OUTPUT ); + digitalWrite( 10, HIGH ); + + // Mount the SD card memory + cout << F("Mount the SD card memory... "); + if( ! sd.begin( CS_SDCARD, SD_SCK_MHZ( 50 ))) + { + cout << F("Unable to mount SD card") << endl; + while( true ) ; + } + pinMode( CS_SDCARD, OUTPUT ); + digitalWrite( CS_SDCARD, HIGH ); + cout << F("ok") << endl; + + // Show capacity and free space of SD card + cout << F("Capacity of card: ") << long( sd.card()->sectorCount() >> 1 ) + << F(" kBytes") << endl; + cout << F("Free space on card: ") + << long( sd.vol()->freeClusterCount() * sd.vol()->sectorsPerCluster() >> 1 ) + << F(" kBytes") << endl; + + // Send reset to Ethernet module + if( W5x00_RESET > -1 ) + { + pinMode( W5x00_RESET, OUTPUT ); + digitalWrite( W5x00_RESET, LOW ); + delay( 200 ); + digitalWrite( W5x00_RESET, HIGH ); + delay( 200 ); + } + + // Initialize the network + cout << F("Initialize ethernet module ... "); + if( serverIp[0] != 0 ) + Ethernet.begin( mac, serverIp ); + else if( Ethernet.begin( mac ) == 0 ) + { + cout << F("failed!") << endl; + while( true ) ; + } + uint16_t wizModule[] = { 0, 5100, 5200, 5500 }; + cout << F("W") << wizModule[ Ethernet.hardwareStatus()] << F(" ok") << endl; + serverIp = Ethernet.localIP(); + cout << F("IP address of server: ") + << int( serverIp[0]) << "." << int( serverIp[1]) << "." + << int( serverIp[2]) << "." << int( serverIp[3]) << endl; + + // Initialize the FTP server + ftpSrv.begin("user","password"); + // ftpSrv.init( externalIp ); + // ftpSrv.init( IPAddress( 11, 22, 33, 44 )); + + // Default username and password are set to 'arduino' and 'test' + // but can then be changed by calling ftpSrv.credentials() + // ftpSrv.credentials( "myname", "123" ); + + cout << F("Free stack: ") << FreeStack() << endl; + + cout << "Viaaa!"; +} + +void loop() +{ + ftpSrv.handleFTP(); + + // more processes... +} diff --git a/lib/SimpleFTPServer/examples/Arduino_esp32_SD/Arduino_esp32_SD.ino b/lib/SimpleFTPServer/examples/Arduino_esp32_SD/Arduino_esp32_SD.ino new file mode 100644 index 00000000..8f497a2f --- /dev/null +++ b/lib/SimpleFTPServer/examples/Arduino_esp32_SD/Arduino_esp32_SD.ino @@ -0,0 +1,96 @@ +/* + * FtpServer esp8266 and esp32 with SD + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#include +#include "SD.h" + +#include + +const char* ssid = ""; +const char* password = ""; + + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + Serial.print(">>>>>>>>>>>>>>> _callback " ); + Serial.print(ftpOperation); + /* FTP_CONNECT, + * FTP_DISCONNECT, + * FTP_FREE_SPACE_CHANGE + */ + Serial.print(" "); + Serial.print(freeSpace); + Serial.print(" "); + Serial.println(totalSpace); + + // freeSpace : totalSpace = x : 360 + + if (ftpOperation == FTP_CONNECT) Serial.println(F("CONNECTED")); + if (ftpOperation == FTP_DISCONNECT) Serial.println(F("DISCONNECTED")); +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + Serial.print(">>>>>>>>>>>>>>> _transferCallback " ); + Serial.print(ftpOperation); + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ + Serial.print(" "); + Serial.print(name); + Serial.print(" "); + Serial.println(transferredSize); +}; + +void setup(void){ + Serial.begin(115200); + WiFi.begin(ssid, password); + Serial.println(""); + + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); + + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// + SPI.begin(14, 12, 15, 13); //SCK, MISO, MOSI,SS + + if (SD.begin(13, SPI)) { + Serial.println("SD opened!"); + + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + ftpSrv.begin("esp8266","esp8266"); //username, password for ftp. (default 21, 50009 for PASV) + } +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! + // server.handleClient(); //example if running a webserver you still need to call .handleClient(); + +} diff --git a/lib/SimpleFTPServer/examples/ESP32_AP_FFAT_WiFi/ESP32_AP_FFAT_WiFi.ino b/lib/SimpleFTPServer/examples/ESP32_AP_FFAT_WiFi/ESP32_AP_FFAT_WiFi.ino new file mode 100644 index 00000000..4c7efa93 --- /dev/null +++ b/lib/SimpleFTPServer/examples/ESP32_AP_FFAT_WiFi/ESP32_AP_FFAT_WiFi.ino @@ -0,0 +1,124 @@ +/* + * FtpServer esp32 with FFat FS + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#include "Arduino.h" +#include "FS.h" +#include "FFat.h" + +#include + +#ifdef STA_MODE + const char* ssid = ""; + const char* password = ""; +#endif +const char* ssid_AP = "ESP32"; +const char* password_AP = "aabbccdd77"; + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + switch (ftpOperation) { + case FTP_CONNECT: + Serial.println(F("FTP: Connected!")); + break; + case FTP_DISCONNECT: + Serial.println(F("FTP: Disconnected!")); + break; + case FTP_FREE_SPACE_CHANGE: + Serial.printf("FTP: Free space change, free %u of %u!\n", freeSpace, totalSpace); + break; + default: + break; + } +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + switch (ftpOperation) { + case FTP_UPLOAD_START: + Serial.println(F("FTP: Upload start!")); + break; + case FTP_UPLOAD: + Serial.printf("FTP: Upload of file %s byte %u\n", name, transferredSize); + break; + case FTP_TRANSFER_STOP: + Serial.println(F("FTP: Finish transfer!")); + break; + case FTP_TRANSFER_ERROR: + Serial.println(F("FTP: Transfer error!")); + break; + default: + break; + } + + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ +}; + +void setup(void){ + Serial.begin(115200); + + //AP mode + WiFi.mode(WIFI_AP); + WiFi.softAP(ssid_AP, password_AP); + delay(1000); + //IPAddress IP = IPAddress (10, 10, 10, 1); + //IPAddress NMask = IPAddress (255, 255, 255, 0); + //WiFi.softAPConfig(IP, IP, NMask); + Serial.print("Set AP named:"); + Serial.println(ssid_AP); + IPAddress myIP = WiFi.softAPIP(); + Serial.print("AP IP address: "); + Serial.println(myIP); + +#ifdef STA_MODE + // STA mode + WiFi.begin(ssid, password); + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); +#endif + + /////FTP Setup, ensure FFAT is started before ftp; ///////// + Serial.print(F("Inizializing FS...")); + if (FFat.begin(true)){ + Serial.println(F("done.")); + }else{ + Serial.println(F("fail.")); + while (true) { delay(1000); }; + } + + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + Serial.println("Start FTP with user: user and passwd: password!"); + ftpSrv.begin("user","password"); //username, password for ftp. (default 21, 50009 for PASV) + ftpSrv.setLocalIp(myIP); +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! +} diff --git a/lib/SimpleFTPServer/examples/ESP32_FFAT_WiFi/ESP32_FFAT_WiFi.ino b/lib/SimpleFTPServer/examples/ESP32_FFAT_WiFi/ESP32_FFAT_WiFi.ino new file mode 100644 index 00000000..df97b956 --- /dev/null +++ b/lib/SimpleFTPServer/examples/ESP32_FFAT_WiFi/ESP32_FFAT_WiFi.ino @@ -0,0 +1,105 @@ +/* + * FtpServer esp32 with FFat FS + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#include "Arduino.h" +#include "FS.h" +#include "FFat.h" + +#include + +const char* ssid = ""; +const char* password = ""; + + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + switch (ftpOperation) { + case FTP_CONNECT: + Serial.println(F("FTP: Connected!")); + break; + case FTP_DISCONNECT: + Serial.println(F("FTP: Disconnected!")); + break; + case FTP_FREE_SPACE_CHANGE: + Serial.printf("FTP: Free space change, free %u of %u!\n", freeSpace, totalSpace); + break; + default: + break; + } +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + switch (ftpOperation) { + case FTP_UPLOAD_START: + Serial.println(F("FTP: Upload start!")); + break; + case FTP_UPLOAD: + Serial.printf("FTP: Upload of file %s byte %u\n", name, transferredSize); + break; + case FTP_TRANSFER_STOP: + Serial.println(F("FTP: Finish transfer!")); + break; + case FTP_TRANSFER_ERROR: + Serial.println(F("FTP: Transfer error!")); + break; + default: + break; + } + + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ +}; + +void setup(void){ + Serial.begin(115200); + WiFi.begin(ssid, password); + Serial.println(""); + + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); + + /////FTP Setup, ensure FFAT is started before ftp; ///////// + Serial.print(F("Inizializing FS...")); + if (FFat.begin(true)){ + Serial.println(F("done.")); + }else{ + Serial.println(F("fail.")); + while (true) { delay(1000); }; + } + + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + Serial.println("Start FTP with user: user and passwd: password!"); + ftpSrv.begin("user","password"); //username, password for ftp. (default 21, 50009 for PASV) + +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! +} diff --git a/lib/SimpleFTPServer/examples/ESP32_FFAT_enc28j60/ESP32_FFAT_enc28j60.ino b/lib/SimpleFTPServer/examples/ESP32_FFAT_enc28j60/ESP32_FFAT_enc28j60.ino new file mode 100644 index 00000000..78b6bb19 --- /dev/null +++ b/lib/SimpleFTPServer/examples/ESP32_FFAT_enc28j60/ESP32_FFAT_enc28j60.ino @@ -0,0 +1,130 @@ +/* + * FtpServer esp32 with FFat and EthernetENC (or UIPEthernet) + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#include "Arduino.h" +#include +#include + +#include "FS.h" +#include "FFat.h" + +#include + +#define MACADDRESS 0x00,0x01,0x02,0x03,0x04,0x05 +#define MYIPADDR 192,168,1,28 +#define MYIPMASK 255,255,255,0 +#define MYDNS 192,168,1,1 +#define MYGW 192,168,1,1 + +uint8_t macaddress[6] = {MACADDRESS}; + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + Serial.print(">>>>>>>>>>>>>>> _callback " ); + Serial.print(ftpOperation); + /* FTP_CONNECT, + * FTP_DISCONNECT, + * FTP_FREE_SPACE_CHANGE + */ + Serial.print(" "); + Serial.print(freeSpace); + Serial.print(" "); + Serial.println(totalSpace); + + // freeSpace : totalSpace = x : 360 + + if (ftpOperation == FTP_CONNECT) Serial.println(F("CONNECTED")); + if (ftpOperation == FTP_DISCONNECT) Serial.println(F("DISCONNECTED")); +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + Serial.print(">>>>>>>>>>>>>>> _transferCallback " ); + Serial.print(ftpOperation); + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ + Serial.print(" "); + Serial.print(name); + Serial.print(" "); + Serial.println(transferredSize); +}; + + +void setup(void){ + Serial.begin(115200); + + Serial.println("Begin Ethernet"); + + Ethernet.init(5); + if (Ethernet.begin(macaddress)) { // Dynamic IP setup + Serial.println("DHCP OK!"); + }else{ + Serial.println("Failed to configure Ethernet using DHCP"); + // Check for Ethernet hardware present + if (Ethernet.hardwareStatus() == EthernetNoHardware) { + Serial.println("Ethernet shield was not found. Sorry, can't run without hardware. :("); + while (true) { + delay(1); // do nothing, no point running without Ethernet hardware + } + } + if (Ethernet.linkStatus() == LinkOFF) { + Serial.println("Ethernet cable is not connected."); + } + + IPAddress ip(MYIPADDR); + IPAddress dns(MYDNS); + IPAddress gw(MYGW); + IPAddress sn(MYIPMASK); + Ethernet.begin(macaddress, ip, dns, gw, sn); + Serial.println("STATIC OK!"); + } + delay(5000); + + + Serial.print("Local IP : "); + Serial.println(Ethernet.localIP()); + Serial.print("Subnet Mask : "); + Serial.println(Ethernet.subnetMask()); + Serial.print("Gateway IP : "); + Serial.println(Ethernet.gatewayIP()); + Serial.print("DNS Server : "); + Serial.println(Ethernet.dnsServerIP()); + + Serial.println("Ethernet Successfully Initialized"); + + /////FTP Setup, ensure FFat is started before ftp; ///////// + if (FFat.begin(true)) { + Serial.println("FFat opened!"); + + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + ftpSrv.begin("user","password"); //username, password for ftp. set ports in ESP8266FtpServer.h (default 21, 50009 for PASV) + Serial.println("FTP server started!"); + } else { + Serial.println("FFat opened FAIL!!!!!"); + } + +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! + // server.handleClient(); //example if running a webserver you still need to call .handleClient(); + +} diff --git a/lib/SimpleFTPServer/examples/ESP32_SPIFM_WiFi/ESP32_SPIFM_WiFi.ino b/lib/SimpleFTPServer/examples/ESP32_SPIFM_WiFi/ESP32_SPIFM_WiFi.ino new file mode 100644 index 00000000..866d65c3 --- /dev/null +++ b/lib/SimpleFTPServer/examples/ESP32_SPIFM_WiFi/ESP32_SPIFM_WiFi.ino @@ -0,0 +1,148 @@ +/** + * ESP32 and external SPI Flash FTP server + * + * +// esp32 configuration +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP32 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_ESP32 NETWORK_ESP32 + #define DEFAULT_STORAGE_TYPE_ESP32 STORAGE_SPIFM +#endif + * + */ + +#include +#include "SdFat.h" +#include "Adafruit_SPIFlash.h" +#include +#include + +Adafruit_FlashTransport_SPI flashTransport(SS, SPI); // Set CS and SPI interface +Adafruit_SPIFlash flash(&flashTransport); + +// file system object from SdFat +FatFileSystem fatfs; + +const char* ssid = ""; +const char* password = ""; + + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + switch (ftpOperation) { + case FTP_CONNECT: + Serial.println(F("FTP: Connected!")); + break; + case FTP_DISCONNECT: + Serial.println(F("FTP: Disconnected!")); + break; + case FTP_FREE_SPACE_CHANGE: + Serial.print(F("FTP: Free space change, free ")); + Serial.print(freeSpace); + Serial.print(F(" of ")); + Serial.println(totalSpace); + break; + default: + break; + } +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + switch (ftpOperation) { + case FTP_UPLOAD_START: + Serial.println(F("FTP: Upload start!")); + break; + case FTP_UPLOAD: + Serial.print(F("FTP: Upload of file ")); + Serial.print(name); + Serial.print(F(" ")); + Serial.print(transferredSize); + Serial.println(F("bytes")); + break; + case FTP_TRANSFER_STOP: + Serial.println(F("FTP: Finish transfer!")); + break; + case FTP_TRANSFER_ERROR: + Serial.println(F("FTP: Transfer error!")); + break; + default: + break; + } + + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ +}; + +void setup() +{ + // Initialize serial port and wait for it to open before continuing. + Serial.begin(115200); + while (!Serial) { + delay(100); + } + Serial.println("FTP with external SPIFlash"); + + WiFi.begin(ssid, password); + Serial.println(""); + + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); + Serial.print("SN address: "); + Serial.println(WiFi.subnetMask()); + + if (flash.begin()) { + Serial.println(F("Device finded and supported!")); + } else { + Serial.println(F("Problem to discover and configure device, check wiring also!")); + while(1) yield(); + } + // Set 4Mhz SPI speed + flashTransport.setClockSpeed(4000000, 4000000); // added to prevent speed problem + + Serial.print("JEDEC ID: "); Serial.println(flash.getJEDECID(), HEX); + Serial.print("Flash size: "); Serial.println(flash.size()); + Serial.flush(); + + // First call begin to mount the filesystem. Check that it returns true + // to make sure the filesystem was mounted. + if (!fatfs.begin(&flash)) { + Serial.println("Error, failed to mount newly formatted filesystem!"); + Serial.println("Was the flash chip formatted with the SdFat_format example?"); + while(1) yield(); + } + Serial.println("Mounted filesystem!"); + Serial.println(); + + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + Serial.println("Starting FTP Server!"); + ftpSrv.begin("user","password"); //username, password for ftp. (default 21, 50009 for PASV) +// ftpSrv.beginAnonymous(); + +} + +// The loop function is called in an endless loop +void loop() +{ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! +} diff --git a/lib/SimpleFTPServer/examples/RaspberryPiPicoW_LittleFS_WiFi/RaspberryPiPicoW_LittleFS_WiFi.ino b/lib/SimpleFTPServer/examples/RaspberryPiPicoW_LittleFS_WiFi/RaspberryPiPicoW_LittleFS_WiFi.ino new file mode 100644 index 00000000..4cef1e97 --- /dev/null +++ b/lib/SimpleFTPServer/examples/RaspberryPiPicoW_LittleFS_WiFi/RaspberryPiPicoW_LittleFS_WiFi.ino @@ -0,0 +1,103 @@ +/* + * FtpServer Raspberry Pi Pico W with LittleFS + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#include +#include + +#include + +const char* ssid = ""; +const char* password = ""; + + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + switch (ftpOperation) { + case FTP_CONNECT: + Serial.println(F("FTP: Connected!")); + break; + case FTP_DISCONNECT: + Serial.println(F("FTP: Disconnected!")); + break; + case FTP_FREE_SPACE_CHANGE: + Serial.printf("FTP: Free space change, free %u of %u!\n", freeSpace, totalSpace); + break; + default: + break; + } +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + switch (ftpOperation) { + case FTP_UPLOAD_START: + Serial.println(F("FTP: Upload start!")); + break; + case FTP_UPLOAD: + Serial.printf("FTP: Upload of file %s byte %u\n", name, transferredSize); + break; + case FTP_TRANSFER_STOP: + Serial.println(F("FTP: Finish transfer!")); + break; + case FTP_TRANSFER_ERROR: + Serial.println(F("FTP: Transfer error!")); + break; + default: + break; + } + + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ +}; + +void setup(void){ + Serial.begin(115200); + + while (!Serial) {delay(100);}; + + WiFi.begin(ssid, password); + Serial.println(""); + + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); + + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// + if (LittleFS.begin()) { + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + Serial.println("LittleFS opened!"); + ftpSrv.begin("user","password"); //username, password for ftp. (default 21, 50009 for PASV) + } +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! +} diff --git a/lib/SimpleFTPServer/examples/STM32_SPIFlash_enc28j60/STM32_SPIFlash_enc28j60.ino b/lib/SimpleFTPServer/examples/STM32_SPIFlash_enc28j60/STM32_SPIFlash_enc28j60.ino new file mode 100644 index 00000000..761cd9a3 --- /dev/null +++ b/lib/SimpleFTPServer/examples/STM32_SPIFlash_enc28j60/STM32_SPIFlash_enc28j60.ino @@ -0,0 +1,136 @@ +/** + * SimpleFTPServer ^1.3.0 on STM32 (need FLASH > 64K) + * and ethernet w5500 + * SPI Flash with Adafruit_SPIFlash and SdFat-Adafruit-Fork library + * + +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 NETWORK_ETHERNET_ENC + #define DEFAULT_STORAGE_TYPE_STM32 STORAGE_SPIFM +#endif + + * + * @author Renzo Mischianti + * @details https://www.mischianti.org/category/my-libraries/simple-ftp-server/ + * @version 0.1 + * @date 2022-03-22 + * + * @copyright Copyright (c) 2022 + * + */ + +#include + +#include + +#include "SdFat.h" +#include "Adafruit_SPIFlash.h" + +#include + +Adafruit_FlashTransport_SPI flashTransport(SS, SPI); // Set CS and SPI interface +Adafruit_SPIFlash flash(&flashTransport); + +// file system object from SdFat +FatFileSystem fatfs; + +#define ETHERNET_CS_PIN PA3 + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +// Enter a MAC address for your controller below. +// Newer Ethernet shields have a MAC address printed on a sticker on the shield +byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED }; + +// Set the static IP address to use if the DHCP fails to assign +#define MYIPADDR 192,168,1,28 +#define MYIPMASK 255,255,255,0 +#define MYDNS 192,168,1,1 +#define MYGW 192,168,1,1 + +// Initialize the Ethernet client library +// with the IP address and port of the server +// that you want to connect to (port 80 is default for HTTP): +EthernetClient client; + +void setup() { + // Initialize serial port and wait for it to open before continuing. + Serial.begin(115200); + while (!Serial) { + delay(100); + } + Serial.println("Adafruit SPI Flash FatFs Full Usage Example"); + + // Initialize flash library and check its chip ID. + if (!flash.begin()) { + Serial.println("Error, failed to initialize flash chip!"); + while(1) yield(); + } + + Serial.print("JEDEC ID: "); Serial.println(flash.getJEDECID(), HEX); + Serial.print("Flash size: "); Serial.println(flash.size()); + Serial.flush(); + + // First call begin to mount the filesystem. Check that it returns true + // to make sure the filesystem was mounted. + if (!fatfs.begin(&flash)) { + Serial.println("Error, failed to mount newly formatted filesystem!"); + Serial.println("Was the flash chip formatted with the SdFat_format example?"); + while(1) yield(); + } + Serial.println("Mounted filesystem!"); + Serial.println(); + + + // You can use Ethernet.init(pin) to configure the CS pin + Ethernet.init(ETHERNET_CS_PIN); + + if (Ethernet.begin(mac)) { // Dynamic IP setup + Serial.println(F("DHCP OK!")); + }else{ + Serial.println(F("Failed to configure Ethernet using DHCP")); + // Check for Ethernet hardware present + if (Ethernet.hardwareStatus() == EthernetNoHardware) { + Serial.println(F("Ethernet shield was not found. Sorry, can't run without hardware. :(")); + while (true) { + delay(1); // do nothing, no point running without Ethernet hardware + } + } + if (Ethernet.linkStatus() == LinkOFF) { + Serial.println(F("Ethernet cable is not connected.")); + } + + IPAddress ip(MYIPADDR); + IPAddress dns(MYDNS); + IPAddress gw(MYGW); + IPAddress sn(MYIPMASK); + Ethernet.begin(mac, ip, dns, gw, sn); + Serial.println("STATIC OK!"); + } + delay(5000); + + + Serial.print("Local IP : "); + Serial.println(Ethernet.localIP()); + Serial.print("Subnet Mask : "); + Serial.println(Ethernet.subnetMask()); + Serial.print("Gateway IP : "); + Serial.println(Ethernet.gatewayIP()); + Serial.print("DNS Server : "); + Serial.println(Ethernet.dnsServerIP()); + + Serial.println("Ethernet Successfully Initialized"); + Serial.println(); + + + // Initialize the FTP server + ftpSrv.begin("user","password"); + Serial.println("Ftp server started!"); +} + +void loop() +{ + ftpSrv.handleFTP(); + + // more processes... +} diff --git a/lib/SimpleFTPServer/examples/STM32_SdFat_enc28j60/STM32_SdFat_enc28j60.ino b/lib/SimpleFTPServer/examples/STM32_SdFat_enc28j60/STM32_SdFat_enc28j60.ino new file mode 100644 index 00000000..614ecdd8 --- /dev/null +++ b/lib/SimpleFTPServer/examples/STM32_SdFat_enc28j60/STM32_SdFat_enc28j60.ino @@ -0,0 +1,135 @@ +/** + * SimpleFTPServer ^1.3.0 on STM32 (need FLASH > 64K) + * and ethernet w5500 + * SD connected on secondary SPI or primary + * + +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 NETWORK_W5100 + #define DEFAULT_STORAGE_TYPE_STM32 STORAGE_SDFAT2 +#endif + + * + * @author Renzo Mischianti + * @details https://www.mischianti.org/category/my-libraries/simple-ftp-server/ + * @version 0.1 + * @date 2022-03-22 + * + * @copyright Copyright (c) 2022 + * + */ +#include +#include +#include +#include + +// Ethernet CS +#define ETHERNET_CS_PIN PA3 + +// To use SD with primary SPI +#define SD_CS_PIN PA2 + +// To use SD with secondary SPI +// #define SD_CS_PIN PB12 +// static SPIClass mySPI2(PB15, PB14, PB13, SD_CS_PIN); +// #define SD2_CONFIG SdSpiConfig(SD_CS_PIN, DEDICATED_SPI, SD_SCK_MHZ(18), &mySPI2) + +SdFat sd; + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +// Enter a MAC address for your controller below. +// Newer Ethernet shields have a MAC address printed on a sticker on the shield +byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED }; + +// Set the static IP address to use if the DHCP fails to assign +#define MYIPADDR 192,168,1,28 +#define MYIPMASK 255,255,255,0 +#define MYDNS 192,168,1,1 +#define MYGW 192,168,1,1 + +// Initialize the Ethernet client library +// with the IP address and port of the server +// that you want to connect to (port 80 is default for HTTP): +EthernetClient client; + +void setup() { + Serial.begin( 115200 ); + + while (!Serial) { delay(100); } + + pinMode( SD_CS_PIN, OUTPUT ); + digitalWrite( SD_CS_PIN, HIGH ); + pinMode( ETHERNET_CS_PIN, OUTPUT ); + digitalWrite( ETHERNET_CS_PIN, HIGH ); + + Serial.print("\nInitializing SD card..."); + + // To use SD with secondary SPI + // if (!sd.begin(SD2_CONFIG)) { + // To use SD with primary SPI + if (!sd.begin(SD_CS_PIN)) { + Serial.println(F("initialization failed. Things to check:")); + Serial.println(F("* is a card inserted?")); + Serial.println(F("* is your wiring correct?")); + Serial.println(F("* did you change the chipSelect pin to match your shield or module?")); + while (1); + } else { + Serial.println(F("Wiring is correct and a card is present.")); + } + + // Show capacity and free space of SD card + Serial.print(F("Capacity of card: ")); Serial.print(long( sd.card()->sectorCount() >> 1 )); Serial.println(F(" kBytes")); + + // You can use Ethernet.init(pin) to configure the CS pin + Ethernet.init(ETHERNET_CS_PIN); + + if (Ethernet.begin(mac)) { // Dynamic IP setup + Serial.println(F("DHCP OK!")); + }else{ + Serial.println(F("Failed to configure Ethernet using DHCP")); + // Check for Ethernet hardware present + if (Ethernet.hardwareStatus() == EthernetNoHardware) { + Serial.println(F("Ethernet shield was not found. Sorry, can't run without hardware. :(")); + while (true) { + delay(1); // do nothing, no point running without Ethernet hardware + } + } + if (Ethernet.linkStatus() == LinkOFF) { + Serial.println(F("Ethernet cable is not connected.")); + } + + IPAddress ip(MYIPADDR); + IPAddress dns(MYDNS); + IPAddress gw(MYGW); + IPAddress sn(MYIPMASK); + Ethernet.begin(mac, ip, dns, gw, sn); + Serial.println("STATIC OK!"); + } + delay(5000); + + + Serial.print("Local IP : "); + Serial.println(Ethernet.localIP()); + Serial.print("Subnet Mask : "); + Serial.println(Ethernet.subnetMask()); + Serial.print("Gateway IP : "); + Serial.println(Ethernet.gatewayIP()); + Serial.print("DNS Server : "); + Serial.println(Ethernet.dnsServerIP()); + + Serial.println("Ethernet Successfully Initialized"); + Serial.println(); + + + // Initialize the FTP server + ftpSrv.begin("user","password"); + Serial.println("Ftp server started!"); +} + +void loop() +{ + ftpSrv.handleFTP(); + + // more processes... +} diff --git a/lib/SimpleFTPServer/examples/STM32_SdFat_w5500/STM32_SdFat_w5500.ino b/lib/SimpleFTPServer/examples/STM32_SdFat_w5500/STM32_SdFat_w5500.ino new file mode 100644 index 00000000..4276cf20 --- /dev/null +++ b/lib/SimpleFTPServer/examples/STM32_SdFat_w5500/STM32_SdFat_w5500.ino @@ -0,0 +1,128 @@ +/** + * SimpleFTPServer ^1.3.0 on STM32 (need FLASH > 64K) + * and ethernet w5500 + * SD connected on secondary SPI or primary + * + +#ifndef DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 + #define DEFAULT_FTP_SERVER_NETWORK_TYPE_STM32 NETWORK_W5100 + #define DEFAULT_STORAGE_TYPE_STM32 STORAGE_SDFAT2 +#endif + + * + * @author Renzo Mischianti + * @details https://www.mischianti.org/category/my-libraries/simple-ftp-server/ + * @version 0.1 + * @date 2022-03-22 + * + * @copyright Copyright (c) 2022 + * + */ +#include +#include +#include + +#include + +// To use SD with primary SPI +// #define SD_CS_PIN PA4 + +// To use SD with secondary SPI +#define SD_CS_PIN PB12 +static SPIClass mySPI2(PB15, PB14, PB13, SD_CS_PIN); +#define SD2_CONFIG SdSpiConfig(SD_CS_PIN, DEDICATED_SPI, SD_SCK_MHZ(18), &mySPI2) + +SdFat sd; + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +// Enter a MAC address for your controller below. +// Newer Ethernet shields have a MAC address printed on a sticker on the shield +byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED }; + +// Set the static IP address to use if the DHCP fails to assign +#define MYIPADDR 192,168,1,28 +#define MYIPMASK 255,255,255,0 +#define MYDNS 192,168,1,1 +#define MYGW 192,168,1,1 + +// Initialize the Ethernet client library +// with the IP address and port of the server +// that you want to connect to (port 80 is default for HTTP): +EthernetClient client; + +void setup() { + Serial.begin( 115200 ); + + while (!Serial) { delay(100); } + + Serial.print("\nInitializing SD card..."); + + // Secondary SPI for SD + if (!sd.begin(SD2_CONFIG)) { + // Primary SPI for SD + // if (!SD.begin(SD_CS_PIN)) { + Serial.println(F("initialization failed. Things to check:")); + Serial.println(F("* is a card inserted?")); + Serial.println(F("* is your wiring correct?")); + Serial.println(F("* did you change the chipSelect pin to match your shield or module?")); + while (1); + } else { + Serial.println(F("Wiring is correct and a card is present.")); + } + + // Show capacity and free space of SD card + Serial.print(F("Capacity of card: ")); Serial.print(long( sd.card()->sectorCount() >> 1 )); Serial.println(F(" kBytes")); + + // You can use Ethernet.init(pin) to configure the CS pin + Ethernet.init(PA4); + + if (Ethernet.begin(mac)) { // Dynamic IP setup + Serial.println(F("DHCP OK!")); + }else{ + Serial.println(F("Failed to configure Ethernet using DHCP")); + // Check for Ethernet hardware present + if (Ethernet.hardwareStatus() == EthernetNoHardware) { + Serial.println(F("Ethernet shield was not found. Sorry, can't run without hardware. :(")); + while (true) { + delay(1); // do nothing, no point running without Ethernet hardware + } + } + if (Ethernet.linkStatus() == LinkOFF) { + Serial.println(F("Ethernet cable is not connected.")); + } + + IPAddress ip(MYIPADDR); + IPAddress dns(MYDNS); + IPAddress gw(MYGW); + IPAddress sn(MYIPMASK); + Ethernet.begin(mac, ip, dns, gw, sn); + Serial.println("STATIC OK!"); + } + delay(5000); + + + Serial.print("Local IP : "); + Serial.println(Ethernet.localIP()); + Serial.print("Subnet Mask : "); + Serial.println(Ethernet.subnetMask()); + Serial.print("Gateway IP : "); + Serial.println(Ethernet.gatewayIP()); + Serial.print("DNS Server : "); + Serial.println(Ethernet.dnsServerIP()); + + Serial.println("Ethernet Successfully Initialized"); + Serial.println(); + + + // Initialize the FTP server + ftpSrv.begin("user","password"); + Serial.println("Ftp server started!"); +} + +void loop() +{ + ftpSrv.handleFTP(); + + // more processes... +} diff --git a/lib/SimpleFTPServer/examples/Wio_terminal/Wio_terminal.ino b/lib/SimpleFTPServer/examples/Wio_terminal/Wio_terminal.ino new file mode 100644 index 00000000..24801348 --- /dev/null +++ b/lib/SimpleFTPServer/examples/Wio_terminal/Wio_terminal.ino @@ -0,0 +1,106 @@ +/* + * FtpServer Wio Terminal + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/category/my-libraries/simple-ftp-server/ + */ + +#include +#include "SD/Seeed_SD.h" + +// #define DEFAULT_FTP_SERVER_NETWORK_TYPE_SAMD NETWORK_SEEED_RTL8720DN +// #define DEFAULT_STORAGE_TYPE_SAMD STORAGE_SEEED_SD + +#include + +#include + +FtpServer ftpSrv; + +const char *ssid = ""; +const char *password = ""; + +void listDir(const char* dirname, uint8_t levels) { + Serial.print("Listing directory: "); + Serial.println(dirname); + + File root = SD.open(dirname); + if (!root) { + Serial.println("Failed to open directory"); + return; + } + if (!root.isDirectory()) { + Serial.println("Not a directory"); + return; + } + + File file = root.openNextFile(); + while (file) { + if (file.isDirectory()) { + Serial.print(" DIR : "); + Serial.println(file.name()); + if (levels) { + listDir(file.name(), levels - 1); + } + } else { + Serial.print(" FILE: "); + Serial.print(file.name()); + Serial.print(" SIZE: "); + Serial.println(file.size()); + } + file = root.openNextFile(); + } +} + + +void setup() +{ + Serial.begin(115200); + delay(1000); + + pinMode(5, OUTPUT); + digitalWrite(5, HIGH); + + while (!SD.begin(SDCARD_SS_PIN,SDCARD_SPI,4000000UL)) { + Serial.println("Card Mount Failed"); + return; + } + + // We start by connecting to a WiFi network + Serial.println(); + Serial.println(); + Serial.print("Connecting to "); + Serial.print(ssid); + + WiFi.mode(WIFI_STA); + + WiFi.begin(ssid, password); + + while (WiFi.status() != WL_CONNECTED) + { + Serial.print("Connecting to "); + Serial.println(ssid); + WiFi.begin(ssid, password); + Serial.print("."); + delay(500); + } + + Serial.println(""); + Serial.println("WiFi connected"); + Serial.println("IP address: "); + Serial.println(WiFi.localIP()); + delay(1000); + + Serial.print("Starting SD."); + + Serial.println("finish!"); + + listDir("/", 0); + + ftpSrv.begin("esp8266","esp8266"); //username, password for ftp. + } + + void loop(void) { + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! + } diff --git a/lib/SimpleFTPServer/examples/Wio_terminal_SdFat/Wio_terminal_SdFat.ino b/lib/SimpleFTPServer/examples/Wio_terminal_SdFat/Wio_terminal_SdFat.ino new file mode 100644 index 00000000..4db9a041 --- /dev/null +++ b/lib/SimpleFTPServer/examples/Wio_terminal_SdFat/Wio_terminal_SdFat.ino @@ -0,0 +1,104 @@ +/* + * FtpServer Wio Terminal with SdFat library + * and with callbacks to the main actions of FTP server + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/category/my-libraries/simple-ftp-server/ + * + */ + +#include "SdFat.h" + +#include + +#include + +// #define DEFAULT_FTP_SERVER_NETWORK_TYPE_SAMD NETWORK_SEEED_RTL8720DN +// #define DEFAULT_STORAGE_TYPE_SAMD STORAGE_SDFAT2 + +#define SD_CONFIG SdSpiConfig(SDCARD_SS_PIN, 2) +SdFs sd; + +FtpServer ftpSrv; + +const char *ssid = ""; +const char *password = ""; + +void setup() +{ + Serial.begin(115200); + delay(1000); + + pinMode(5, OUTPUT); + digitalWrite(5, HIGH); + + // Initialize the SD. + if (!sd.begin(SD_CONFIG)) { + sd.initErrorHalt(&Serial); + } + FsFile dir; + FsFile file; + + // Open root directory + if (!dir.open("/")){ + Serial.println("dir.open failed"); + } + + // We start by connecting to a WiFi network + + Serial.println(); + Serial.println(); + Serial.print("Connecting to "); + Serial.print(ssid); + + WiFi.mode(WIFI_STA); + + + WiFi.begin(ssid, password); + + while (WiFi.status() != WL_CONNECTED) + { + Serial.print("Connecting to "); + Serial.println(ssid); + WiFi.begin(ssid, password); + Serial.print("."); + delay(500); + } + + Serial.println(""); + Serial.println("WiFi connected"); + Serial.println("IP address: "); + Serial.println(WiFi.localIP()); + delay(1000); + + Serial.print("Starting SD."); + + Serial.println("finish!"); + + while (file.openNext(&dir, O_RDONLY)) { + file.printFileSize(&Serial); + Serial.write(' '); + file.printModifyDateTime(&Serial); + Serial.write(' '); + file.printName(&Serial); + if (file.isDir()) { + // Indicate a directory. + Serial.write('/'); + } + Serial.println(); + file.close(); + } + if (dir.getError()) { + Serial.println("openNext failed"); + } else { + Serial.println("Done!"); + } + + + ftpSrv.begin("esp8266","esp8266"); //username, password for ftp. + } + + void loop(void) { + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! + } diff --git a/lib/SimpleFTPServer/examples/Wio_terminal_SdFat_TFT_monitor/Wio_terminal_SdFat_TFT_monitor.ino b/lib/SimpleFTPServer/examples/Wio_terminal_SdFat_TFT_monitor/Wio_terminal_SdFat_TFT_monitor.ino new file mode 100644 index 00000000..533944c5 --- /dev/null +++ b/lib/SimpleFTPServer/examples/Wio_terminal_SdFat_TFT_monitor/Wio_terminal_SdFat_TFT_monitor.ino @@ -0,0 +1,367 @@ +/* + * FtpServer Wio Terminal with SdFat library + * and with callbacks to the main actions of FTP server + * and a monitor on TFT + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/category/my-libraries/simple-ftp-server/ + * + */ + +#include "SdFat.h" + +#include + + +#include // Hardware-specific library +#include + +#include + +TFT_eSPI tft = TFT_eSPI(); // Invoke custom library + +#define DEG2RAD 0.0174532925 + +byte inc = 0; +unsigned int col = 0; + +#define SD_CONFIG SdSpiConfig(SDCARD_SS_PIN, 2) +SdFs sd; + +FtpServer ftpSrv; + +const char *ssid = "reef-casa-sopra"; +const char *password = "aabbccdd77"; + +#define MAIN_TOP 110 + +#define FREE_SPACE_PIE_X 80 +#define FREE_SPACE_PIE_Y MAIN_TOP+40 +#define FREE_SPACE_PIE_RADIUS 50 + +void freeSpacePieData(unsigned int freeSpace, unsigned int totalSpace) { + int pieFree = 360 - (freeSpace * 360 / totalSpace); + + fillSegment(FREE_SPACE_PIE_X, FREE_SPACE_PIE_Y, 0, pieFree, FREE_SPACE_PIE_RADIUS, TFT_RED); + fillSegment(FREE_SPACE_PIE_X, FREE_SPACE_PIE_Y, pieFree, 360 - pieFree, FREE_SPACE_PIE_RADIUS, TFT_BLUE); + + // Set "cursor" at top left corner of display (0,0) and select font 2 + // (cursor will move to next line automatically during printing with 'tft.println' + // or stay on the line is there is room for the text with tft.print) + tft.setCursor(FREE_SPACE_PIE_X + 80, MAIN_TOP, 2); + // Set the font colour to be white with a black background, set text size multiplier to 1 + tft.setTextColor(TFT_WHITE, TFT_BLACK); tft.setTextSize(1); + // We can now plot text on screen using the "print" class + Serial.print(freeSpace/1000);Serial.print("Mb/");Serial.print(String(totalSpace/1000));Serial.println("Mb"); + tft.print(freeSpace/1000);tft.print("Mb/");tft.print(String(totalSpace/1000));tft.println("Mb"); +} + +void connectedDisconnected(bool connected) { + tft.fillCircle(FREE_SPACE_PIE_X + 80 + 10, MAIN_TOP+25+7, 10, (connected)?TFT_GREEN:TFT_RED); + + tft.setCursor(FREE_SPACE_PIE_X + 80 + 25, MAIN_TOP+25, 2); + tft.println(" "); + + tft.setCursor(FREE_SPACE_PIE_X + 80 + 25, MAIN_TOP+25, 2); + (connected)?tft.println("Connected!"):tft.println("Disconnected!"); +} + +void transfer(bool transfer, bool upload) { + tft.fillCircle(FREE_SPACE_PIE_X + 80 + 10, MAIN_TOP+25+25+7, 10, (transfer)?(upload)?TFT_GREEN:TFT_BLUE:TFT_RED); + + tft.setCursor(FREE_SPACE_PIE_X + 80 + 25, MAIN_TOP+25+25, 2); + tft.println(" "); + + tft.setCursor(FREE_SPACE_PIE_X + 80 + 25, MAIN_TOP+25+25, 2); + (transfer)?tft.println((upload)?"Upload!":"Download!"):tft.println("Idle!"); +} + +//index - starting at, n- how many chars +char* subString(const char *s, int index, int n){ + char* b = (char*) malloc((strlen(s) + 1) * sizeof(char)); + strcpy(b,s); + + Serial.println("--------------------------------------"); + Serial.println(s); + Serial.println(index); + Serial.println(n); + char *res = new char[n + 1]; + Serial.println(res); + sprintf(res, "%.*s", n, b + index); + Serial.println(res); + free(b); + return res; +} + + +void fileTransfer(FtpTransferOperation ftpOperation, const char* filename, unsigned int transferredSize) { + int yoffset = 2; + + tft.setCursor(20, MAIN_TOP+(FREE_SPACE_PIE_RADIUS*2)+yoffset, 2); + tft.println(F(" ")); + + tft.setCursor(20, MAIN_TOP+(FREE_SPACE_PIE_RADIUS*2)+yoffset, 2); + int lenfile = strlen(filename); + Serial.println(lenfile); + if (lenfile>22) { + + tft.print(subString(filename, 0, 16));tft.print(F("~")); + tft.print( subString(filename, (lenfile-4), 4) ); + } else { + tft.print(filename); + } + tft.setCursor(20+160, MAIN_TOP+(FREE_SPACE_PIE_RADIUS*2)+yoffset, 2); + tft.print(F(" ")); + tft.setCursor(20+160, MAIN_TOP+(FREE_SPACE_PIE_RADIUS*2)+yoffset, 2); + tft.print(transferredSize);tft.print("Kb"); + + tft.setCursor(320-55, MAIN_TOP+(FREE_SPACE_PIE_RADIUS*2)+yoffset, 2); + switch (ftpOperation) { + case FTP_UPLOAD: + tft.setTextColor(TFT_GREEN, TFT_BLACK); + tft.print(F("Upload")); + tft.setTextColor(TFT_WHITE, TFT_BLACK); + break; + case FTP_DOWNLOAD: + tft.setTextColor(TFT_BLUE, TFT_BLACK); + tft.print(F("Down")); + tft.setTextColor(TFT_WHITE, TFT_BLACK); + + break; + case FTP_TRANSFER_STOP: + tft.setTextColor(TFT_GREEN, TFT_BLACK); + tft.print(F("OK")); + tft.setTextColor(TFT_WHITE, TFT_BLACK); + + break; + case FTP_TRANSFER_ERROR: + tft.setTextColor(TFT_RED, TFT_BLACK); + tft.print(F("Error")); + tft.setTextColor(TFT_WHITE, TFT_BLACK); + + break; + + default: + break; + } + +} + +void wifiStrenght (int8_t RSSI, bool connection = false) { + Serial.print("RSSI --> ");Serial.println(RSSI); + int marginX = 30; + + int startX = 90; + int widthW = 320-(startX+marginX); + + int startY = 60; + int heightY = 10; + + tft.setCursor(marginX, startY - 5, 2); + tft.print(F(" ")); + tft.setCursor(marginX, startY - 5, 2); + + if (connection) { + tft.print(F("Connectint to: ")); + tft.print(ssid); + }else{ + tft.println("WiFi str: "); + + // 120 : 120-RSSI = 300 : x + + tft.drawRoundRect(startX, startY, widthW, heightY, 5, TFT_WHITE); + + uint32_t colorRSSI = TFT_GREEN; + if (abs(RSSI)<55) { + colorRSSI = TFT_GREEN; + } else if (abs(RSSI)<75) { + colorRSSI = TFT_YELLOW; + } else if (abs(RSSI)<75) { + colorRSSI = TFT_RED; + } + + tft.fillRoundRect(startX+1, startY+1, (120+RSSI)*widthW/120, heightY-2, 5, colorRSSI); + + tft.setCursor(marginX, startY + 15, 2); + + tft.print("IP: "); + tft.println(WiFi.localIP()); + } +} + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + Serial.print(">>>>>>>>>>>>>>> _callback " ); + Serial.print(ftpOperation); + /* FTP_CONNECT, + * FTP_DISCONNECT, + * FTP_FREE_SPACE_CHANGE + */ + Serial.print(" "); + Serial.print(freeSpace); + Serial.print(" "); + Serial.println(totalSpace); + + // freeSpace : totalSpace = x : 360 + + freeSpacePieData(freeSpace, totalSpace); + + if (ftpOperation == FTP_CONNECT) connectedDisconnected(true); + if (ftpOperation == FTP_DISCONNECT) connectedDisconnected(false); +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + Serial.print(">>>>>>>>>>>>>>> _transferCallback " ); + Serial.print(ftpOperation); + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ + Serial.print(" "); + Serial.print(name); + Serial.print(" "); + Serial.println(transferredSize); + + (ftpOperation==FTP_UPLOAD || ftpOperation==FTP_DOWNLOAD)?transfer(true, ftpOperation==FTP_UPLOAD):transfer(false, false); + + fileTransfer(ftpOperation, name, transferredSize); +}; + + +void setup() +{ + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + Serial.begin(115200); + delay(1000); + + tft.init(); + + tft.begin(); + + tft.setRotation(3); + + tft.fillScreen(TFT_BLACK); + + tft.setCursor(0, 0); + + tft.setTextColor(TFT_BLACK, TFT_WHITE); tft.setTextSize(2); + + tft.fillRoundRect(3, 3, 320-6, 40, 5, TFT_WHITE); + + tft.drawCentreString("www.mischianti.org", 160, 14,1); + tft.setTextColor(TFT_WHITE, TFT_BLACK); + + + freeSpacePieData(0, 0); + connectedDisconnected(false); + transfer(false, false); + + wifiStrenght(0, true); + + Serial.println(); + Serial.println(); + Serial.print("Connecting to "); + Serial.print(ssid); + + WiFi.mode(WIFI_STA); + + + WiFi.begin(ssid, password); + + while (WiFi.status() != WL_CONNECTED) + { + Serial.print("."); + WiFi.begin(ssid, password); + Serial.print("."); + tft.print(F(".")); + delay(500); + } + + Serial.println(""); + Serial.println("WiFi connected"); + Serial.println("IP address: "); + Serial.println(WiFi.localIP()); + + wifiStrenght(WiFi.RSSI()); + + delay(1000); + + if (!sd.begin(SD_CONFIG)) { + sd.initErrorHalt(&Serial); + } + FsFile dir; + FsFile file; + + // Open root directory + if (!dir.open("/")){ + Serial.println("dir.open failed"); + } + + ftpSrv.begin("wioterminal","wioterminal"); //username, password for ftp. + +} + +void loop() { + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! +} + + +// ######################################################################### +// Draw circle segments +// ######################################################################### + +// x,y == coords of centre of circle +// start_angle = 0 - 359 +// sub_angle = 0 - 360 = subtended angle +// r = radius +// colour = 16 bit colour value + +int fillSegment(int x, int y, int start_angle, int sub_angle, int r, unsigned int colour) { + // Calculate first pair of coordinates for segment start + float sx = cos((start_angle - 90) * DEG2RAD); + float sy = sin((start_angle - 90) * DEG2RAD); + uint16_t x1 = sx * r + x; + uint16_t y1 = sy * r + y; + + // Draw colour blocks every inc degrees + for (int i = start_angle; i < start_angle + sub_angle; i++) { + + // Calculate pair of coordinates for segment end + int x2 = cos((i + 1 - 90) * DEG2RAD) * r + x; + int y2 = sin((i + 1 - 90) * DEG2RAD) * r + y; + + tft.fillTriangle(x1, y1, x2, y2, x, y, colour); + + // Copy segment end to sgement start for next segment + x1 = x2; + y1 = y2; + } +} + + +// ######################################################################### +// Return the 16 bit colour with brightness 0-100% +// ######################################################################### +unsigned int brightness(unsigned int colour, int brightness) { + byte red = colour >> 11; + byte green = (colour & 0x7E0) >> 5; + byte blue = colour & 0x1F; + + blue = (blue * brightness) / 100; + green = (green * brightness) / 100; + red = (red * brightness) / 100; + + return (red << 11) + (green << 5) + blue; +} + diff --git a/lib/SimpleFTPServer/examples/esp8266_esp32_LittleFS/esp8266_esp32_LittleFS.ino b/lib/SimpleFTPServer/examples/esp8266_esp32_LittleFS/esp8266_esp32_LittleFS.ino new file mode 100644 index 00000000..0d82e764 --- /dev/null +++ b/lib/SimpleFTPServer/examples/esp8266_esp32_LittleFS/esp8266_esp32_LittleFS.ino @@ -0,0 +1,110 @@ +/* + * FtpServer esp8266 and esp32 with LittleFS + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#ifdef ESP8266 +#include +#include +#elif defined ESP32 +#include +#include +#include +#endif + +#include + +const char* ssid = ""; +const char* password = ""; + + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + switch (ftpOperation) { + case FTP_CONNECT: + Serial.println(F("FTP: Connected!")); + break; + case FTP_DISCONNECT: + Serial.println(F("FTP: Disconnected!")); + break; + case FTP_FREE_SPACE_CHANGE: + Serial.printf("FTP: Free space change, free %u of %u!\n", freeSpace, totalSpace); + break; + default: + break; + } +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + switch (ftpOperation) { + case FTP_UPLOAD_START: + Serial.println(F("FTP: Upload start!")); + break; + case FTP_UPLOAD: + Serial.printf("FTP: Upload of file %s byte %u\n", name, transferredSize); + break; + case FTP_TRANSFER_STOP: + Serial.println(F("FTP: Finish transfer!")); + break; + case FTP_TRANSFER_ERROR: + Serial.println(F("FTP: Transfer error!")); + break; + default: + break; + } + + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ +}; + +void setup(void){ + Serial.begin(115200); + WiFi.begin(ssid, password); + Serial.println(""); + + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); + + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// +#ifdef ESP32 //esp32 we send true to format spiffs if cannot mount + if (LittleFS.begin(true)) { +#elif defined ESP8266 + if (LittleFS.begin()) { +#endif + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + Serial.println("LittleFS opened!"); + ftpSrv.begin("user","password"); //username, password for ftp. (default 21, 50009 for PASV) + } +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! +} diff --git a/lib/SimpleFTPServer/examples/esp8266_esp32_SPIFFS/esp8266_esp32_SPIFFS.ino b/lib/SimpleFTPServer/examples/esp8266_esp32_SPIFFS/esp8266_esp32_SPIFFS.ino new file mode 100644 index 00000000..c67a1ade --- /dev/null +++ b/lib/SimpleFTPServer/examples/esp8266_esp32_SPIFFS/esp8266_esp32_SPIFFS.ino @@ -0,0 +1,110 @@ +/* + * FtpServer esp8266 and esp32 with SPIFFS + * + * AUTHOR: Renzo Mischianti + * + * https://www.mischianti.org/2020/02/08/ftp-server-on-esp8266-and-esp32 + * + */ + +#ifdef ESP8266 +#include +#elif defined ESP32 +#include +#include "SPIFFS.h" +#endif + +#include + +const char* ssid = "YOUR_SSID"; +const char* password = "YOUR_PASS"; + + +FtpServer ftpSrv; //set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial + +void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace){ + switch (ftpOperation) { + case FTP_CONNECT: + Serial.println(F("FTP: Connected!")); + break; + case FTP_DISCONNECT: + Serial.println(F("FTP: Disconnected!")); + break; + case FTP_FREE_SPACE_CHANGE: + Serial.printf("FTP: Free space change, free %u of %u!\n", freeSpace, totalSpace); + break; + default: + break; + } +}; +void _transferCallback(FtpTransferOperation ftpOperation, const char* name, unsigned int transferredSize){ + switch (ftpOperation) { + case FTP_UPLOAD_START: + Serial.println(F("FTP: Upload start!")); + break; + case FTP_UPLOAD: + Serial.printf("FTP: Upload of file %s byte %u\n", name, transferredSize); + break; + case FTP_TRANSFER_STOP: + Serial.println(F("FTP: Finish transfer!")); + break; + case FTP_TRANSFER_ERROR: + Serial.println(F("FTP: Transfer error!")); + break; + default: + break; + } + + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ +}; + +void setup(void){ + Serial.begin(115200); + WiFi.begin(ssid, password); + Serial.println(""); + + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println(""); + Serial.print("Connected to "); + Serial.println(ssid); + Serial.print("IP address: "); + Serial.println(WiFi.localIP()); + + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// + + /////FTP Setup, ensure SPIFFS is started before ftp; ///////// +#ifdef ESP32 //esp32 we send true to format spiffs if cannot mount + if (SPIFFS.begin(true)) { +#elif defined ESP8266 + if (SPIFFS.begin()) { +#endif + ftpSrv.setCallback(_callback); + ftpSrv.setTransferCallback(_transferCallback); + + Serial.println("SPIFFS opened!"); + ftpSrv.begin("esp8266","esp8266"); //username, password for ftp. (default 21, 50009 for PASV) + } +} +void loop(void){ + ftpSrv.handleFTP(); //make sure in loop you call handleFTP()!! + // server.handleClient(); //example if running a webserver you still need to call .handleClient(); + +} diff --git a/lib/SimpleFTPServer/keywords.txt b/lib/SimpleFTPServer/keywords.txt new file mode 100644 index 00000000..5cc521ad --- /dev/null +++ b/lib/SimpleFTPServer/keywords.txt @@ -0,0 +1,17 @@ +####################################### +# Datatypes (KEYWORD1) +####################################### + +SimpleFtpServer KEYWORD1 + +####################################### +# Methods and Functions (KEYWORD2) +####################################### + +begin KEYWORD2 +end KEYWORD2 +setLocalIp KEYWORD2 +credentials KEYWORD2 +setCallback KEYWORD2 +setTransferCallback KEYWORD2 +handleFTP KEYWORD2 \ No newline at end of file diff --git a/lib/SimpleFTPServer/library.json b/lib/SimpleFTPServer/library.json new file mode 100644 index 00000000..9ad3dfc9 --- /dev/null +++ b/lib/SimpleFTPServer/library.json @@ -0,0 +1,21 @@ +{ + "name": "SimpleFTPServer", + "description": "Simple FTP Server for using esp8266, esp32, STM32, Arduino", + "keywords": "arduino, esp8266, esp32, stm32, rp2040, Raspberry Pi, ftp, FtpServer, spiffs, Fat, LittleFS, Ethernet, WiFi, WiFiNINA", + "homepage": "https://www.mischianti.org/category/my-libraries/simple-ftp-server/", + "authors": + { + "name": "Renzo Mischianti", + "email": "renzo.mischianti@gmail.com", + "url": "https://www.mischianti.org" + }, + "repository": + { + "type": "git", + "url": "https://github.com/xreef/SimpleFTPServer" + }, + "url": "https://www.mischianti.org", + "frameworks": "Arduino", + "version": "2.1.6", + "platforms": "*" +} diff --git a/lib/SimpleFTPServer/library.properties b/lib/SimpleFTPServer/library.properties new file mode 100644 index 00000000..e94fd11b --- /dev/null +++ b/lib/SimpleFTPServer/library.properties @@ -0,0 +1,11 @@ +name=SimpleFTPServer +version=2.1.6 +author=Renzo Mischianti +maintainer=Renzo Mischianti +sentence=Simple FTP server for esp8266, esp32, STM32, Raspberry Pi Pico and Arduino +paragraph=Simple FTP server for Raspberry Pi Pico W (LittleFS), esp8266 (SPIFFS and LittleFS or SD, SdFat 2.x), esp32 (SPIFFS, LittleFS and FFAT or SD, SdFat 2.x) and Arduino (SdFat, SD basic lib with 8.3 file format), Wio Terminal (Seed_SD, SdFat 2.x), Arduino MKR (SdFat 2), STM32 (Flash >64K SdFat 2.x and SPI Flash). Support w5500, w5100 and enc28j60. With internal callback to chck the phase of communication. +category=Communication +url=https://www.mischianti.org/category/my-libraries/simple-ftp-server/ +repository=https://github.com/xreef/SimpleFTPServer.git +architectures=* +includes=SimpleFTPServer.h \ No newline at end of file diff --git a/myProfile.json b/myProfile.json index 01261688..bfac01c1 100644 --- a/myProfile.json +++ b/myProfile.json @@ -118,7 +118,7 @@ }, { "path": "src/modules/sensors/Ds2423", - "active": true + "active": false }, { "path": "src/modules/sensors/Emon", @@ -170,7 +170,7 @@ }, { "path": "src/modules/sensors/Ntc", - "active": true + "active": false }, { "path": "src/modules/sensors/Pzem004t", @@ -190,7 +190,7 @@ }, { "path": "src/modules/sensors/Scd40", - "active": true + "active": false }, { "path": "src/modules/sensors/Sds011", @@ -238,6 +238,10 @@ "path": "src/modules/exec/EspCam", "active": false }, + { + "path": "src/modules/exec/Ftp", + "active": false + }, { "path": "src/modules/exec/HttpGet", "active": false diff --git a/src/modules/exec/Ftp/Ftp.cpp b/src/modules/exec/Ftp/Ftp.cpp new file mode 100644 index 00000000..fb0cdc57 --- /dev/null +++ b/src/modules/exec/Ftp/Ftp.cpp @@ -0,0 +1,97 @@ +#include "Global.h" +#include "classes/IoTItem.h" + +#include + +#define DEFAULT_STORAGE_TYPE_ESP32 STORAGE_LITTLEFS + +class FTPModule : public IoTItem +{ +private: + String login; + String pass; + FtpServer ftpSrv; // set #define FTP_DEBUG in ESP8266FtpServer.h to see ftp verbose on serial +public: + FTPModule(String parameters) : IoTItem(parameters) + { + jsonRead(parameters, F("login"), login); + jsonRead(parameters, F("pass"), pass); + ftpSrv.setCallback(FTPModule::_callback); + ftpSrv.setTransferCallback(FTPModule::_transferCallback); + ftpSrv.begin(login.c_str(), pass.c_str(), "Welcome IoTManager FTP server"); // username, password for ftp. (default 21, 50009 for PASV) + SerialPrint("I", "FtpServer " + (String)_id, "begin"); + } + + void loop() + { + ftpSrv.handleFTP(); + } + + static void _callback(FtpOperation ftpOperation, unsigned int freeSpace, unsigned int totalSpace) + { + switch (ftpOperation) + { + case FTP_CONNECT: + SerialPrint("i", "FTP", F("Connected!")); + + break; + case FTP_DISCONNECT: + SerialPrint("i", "FTP", F("Disconnected!")); + break; + case FTP_FREE_SPACE_CHANGE: + SerialPrint("i", "FTP", "Free space change, free " + (String)freeSpace + " of " + (String)totalSpace); + break; + default: + break; + } + } + + static void _transferCallback(FtpTransferOperation ftpOperation, const char *name, unsigned int transferredSize) + { + switch (ftpOperation) + { + case FTP_UPLOAD_START: + SerialPrint("i","FTP", F("Upload start!")); + break; + case FTP_UPLOAD: + SerialPrint("i","FTP", "Upload of file " + (String)name + " byte " + (String)transferredSize); + break; + case FTP_TRANSFER_STOP: + SerialPrint("i","FTP", F("Finish transfer!")); + break; + case FTP_TRANSFER_ERROR: + SerialPrint("E","FTP", F("Transfer error!")); + break; + default: + break; + } + + /* FTP_UPLOAD_START = 0, + * FTP_UPLOAD = 1, + * + * FTP_DOWNLOAD_START = 2, + * FTP_DOWNLOAD = 3, + * + * FTP_TRANSFER_STOP = 4, + * FTP_DOWNLOAD_STOP = 4, + * FTP_UPLOAD_STOP = 4, + * + * FTP_TRANSFER_ERROR = 5, + * FTP_DOWNLOAD_ERROR = 5, + * FTP_UPLOAD_ERROR = 5 + */ + } + + ~FTPModule(){}; +}; + +void *getAPI_FTPModule(String subtype, String param) +{ + if (subtype == F("ftp")) + { + return new FTPModule(param); + } + //} + + return nullptr; +} diff --git a/src/modules/exec/Ftp/modinfo.json b/src/modules/exec/Ftp/modinfo.json new file mode 100644 index 00000000..03412d91 --- /dev/null +++ b/src/modules/exec/Ftp/modinfo.json @@ -0,0 +1,68 @@ +{ + "menuSection": "Исполнительные устройства", + "configItem": [ + { + "global": 0, + "name": "FTP сервер", + "type": "Reading", + "subtype": "ftp", + "id": "ftp", + "widget": "nil", + "page": "", + "descr": "FTP сервер", + "login": "admin", + "pass": "admin" + + + } + + + ], + "about": { + "authorName": "Bubnov Mikhail", + "authorContact": "https://t.me/Mit4bmw", + "authorGit": "https://github.com/Mit4el", + "specialThanks": "", + "moduleName": "FTPModule", + "moduleVersion": "0.1", + "usedRam": { + "esp32_4mb": 15, + "esp8266_4mb": 15 + }, + "title": "FTP-сервер", + "moduleDesc": "Запускает FTP-сервер на плате esp", + "propInfo": { + "login": "Логин FTP сервера", + "pass": "Пароль FTP сервера" + } + }, + "defActive": false, + "usedLibs": { + "esp32_4mb": [ + "https://github.com/xreef/SimpleFTPServer" + ], + "esp32s2_4mb": [], + "esp8266_4mb": [ + "https://github.com/xreef/SimpleFTPServer" + ], + "esp8266_1mb": [ + "https://github.com/xreef/SimpleFTPServer" + ], + "esp8266_1mb_ota": [ + "https://github.com/xreef/SimpleFTPServer" + ], + "esp8285_1mb": [ + "https://github.com/xreef/SimpleFTPServer" + ], + "esp8285_1mb_ota": [ + "https://github.com/xreef/SimpleFTPServer" + ], + "esp8266_2mb": [ + "https://github.com/xreef/SimpleFTPServer" + ], + "esp8266_2mb_ota": [ + "https://github.com/xreef/SimpleFTPServer" + ] + } +} + diff --git a/src/modules/sensors/Scd40/Scd40.cpp b/src/modules/sensors/Scd40/Scd40.cpp index eb93da2c..df4a4476 100644 --- a/src/modules/sensors/Scd40/Scd40.cpp +++ b/src/modules/sensors/Scd40/Scd40.cpp @@ -30,7 +30,7 @@ void printSerialNumber(uint16_t serial0, uint16_t serial1, uint16_t serial2) } // Функция инициализации библиотечного класса, возвращает Единстрвенный указать на библиотеку -SensirionI2CScd4x *instance() +SensirionI2CScd4x *instanceScd4x() { if (!scd4x) { // Если библиотека ранее инициализировалась, т о просто вернем указатель @@ -41,7 +41,7 @@ SensirionI2CScd4x *instance() //Останавливаем периодический опрос датчика вбиблиотеке для запроса Сер.номера (на всякий случай) // stop potentially previously started measurement - errorCodeScd4x = instance()->stopPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->stopPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute stopPeriodicMeasurement(): "); @@ -52,7 +52,7 @@ SensirionI2CScd4x *instance() uint16_t serial0; uint16_t serial1; uint16_t serial2; - errorCodeScd4x = instance()->getSerialNumber(serial0, serial1, serial2); + errorCodeScd4x = instanceScd4x()->getSerialNumber(serial0, serial1, serial2); if (errorCodeScd4x) { Serial.print("Error trying to execute getSerialNumber(): "); @@ -66,7 +66,7 @@ SensirionI2CScd4x *instance() //Обратно стартуем периодический опрос датчика библиотекой (по описанию библиотеки каждые 5сек) // Start Measurement - errorCodeScd4x = instance()->startPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->startPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute startPeriodicMeasurement(): "); @@ -102,7 +102,7 @@ public: float humidity = 0.0f; bool isDataReady = false; //Запрашиваем библиотеку о готовности отправить запрос - errorCodeScd4x = instance()->getDataReadyFlag(isDataReady); + errorCodeScd4x = instanceScd4x()->getDataReadyFlag(isDataReady); if (errorCodeScd4x) { Serial.print("Error trying to execute getDataReadyFlag(): "); @@ -115,7 +115,7 @@ public: return; } //Если все нормально забираем у библиотеки данные - errorCodeScd4x = instance()->readMeasurement(co2, temperature, humidity); + errorCodeScd4x = instanceScd4x()->readMeasurement(co2, temperature, humidity); if (errorCodeScd4x) { Serial.print("Error trying to execute readMeasurement(): "); @@ -158,7 +158,7 @@ public: { //Останавливаем периодический опрос датчика вбиблиотеке для запроса Сер.номера (на всякий случай) // stop potentially previously started measurement - errorCodeScd4x = instance()->stopPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->stopPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute stopPeriodicMeasurement(): "); @@ -166,7 +166,7 @@ public: } delay(500); // Из описания performForcedRecalibration 2. Stop periodic measurement. Wait 500 ms. uint16_t frcCorrection; - errorCodeScd4x = instance()->performForcedRecalibration(targetCo2, frcCorrection); + errorCodeScd4x = instanceScd4x()->performForcedRecalibration(targetCo2, frcCorrection); if (errorCodeScd4x) { @@ -182,7 +182,7 @@ public: //Обратно стартуем периодический опрос датчика библиотекой (по описанию библиотеки каждые 5сек) // Start Measurement - errorCodeScd4x = instance()->startPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->startPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute startPeriodicMeasurement(): "); @@ -197,14 +197,14 @@ public: { //Останавливаем периодический опрос датчика вбиблиотеке для запроса Сер.номера (на всякий случай) // stop potentially previously started measurement - errorCodeScd4x = instance()->stopPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->stopPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute stopPeriodicMeasurement(): "); Serial.println(errorMessageScd4x); } - errorCodeScd4x = instance()->startLowPowerPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->startLowPowerPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute startLowPowerPeriodicMeasurement(): "); @@ -216,7 +216,7 @@ public: Serial.println("startLowPowerPeriodicMeasurement(): OK!"); } - errorCodeScd4x = instance()->setAutomaticSelfCalibration((uint16_t)autoCalibration); + errorCodeScd4x = instanceScd4x()->setAutomaticSelfCalibration((uint16_t)autoCalibration); if (errorCodeScd4x) { Serial.print("Error trying to execute setAutomaticSelfCalibration(): "); @@ -230,7 +230,7 @@ public: //Обратно стартуем периодический опрос датчика библиотекой (по описанию библиотеки каждые 5сек) // Start Measurement - errorCodeScd4x = instance()->startPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->startPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute startPeriodicMeasurement(): "); @@ -262,7 +262,7 @@ public: float temperature = 0.0f; float humidity = 0.0f; bool isDataReady = false; - errorCodeScd4x = instance()->getDataReadyFlag(isDataReady); + errorCodeScd4x = instanceScd4x()->getDataReadyFlag(isDataReady); if (errorCodeScd4x) { Serial.print("Error trying to execute getDataReadyFlag(): "); @@ -274,7 +274,7 @@ public: { return; } - errorCodeScd4x = instance()->readMeasurement(co2, temperature, humidity); + errorCodeScd4x = instanceScd4x()->readMeasurement(co2, temperature, humidity); if (errorCodeScd4x) { Serial.print("errorCodeScd4x trying to execute readMeasurement(): "); @@ -308,14 +308,14 @@ public: { //Останавливаем периодический опрос датчика вбиблиотеке для запроса Сер.номера (на всякий случай) // stop potentially previously started measurement - errorCodeScd4x = instance()->stopPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->stopPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute stopPeriodicMeasurement(): "); Serial.println(errorMessageScd4x); } - errorCodeScd4x = instance()->setTemperatureOffset((uint16_t)offsetT); + errorCodeScd4x = instanceScd4x()->setTemperatureOffset((uint16_t)offsetT); if (errorCodeScd4x) { Serial.print("Error trying to execute setTemperatureOffset(): "); @@ -329,7 +329,7 @@ public: //Обратно стартуем периодический опрос датчика библиотекой (по описанию библиотеки каждые 5сек) // Start Measurement - errorCodeScd4x = instance()->startPeriodicMeasurement(); + errorCodeScd4x = instanceScd4x()->startPeriodicMeasurement(); if (errorCodeScd4x) { Serial.print("Error trying to execute startPeriodicMeasurement(): "); @@ -357,7 +357,7 @@ public: float temperature = 0.0f; float humidity = 0.0f; bool isDataReady = false; - errorCodeScd4x = instance()->getDataReadyFlag(isDataReady); + errorCodeScd4x = instanceScd4x()->getDataReadyFlag(isDataReady); if (errorCodeScd4x) { Serial.print("Error trying to execute getDataReadyFlag(): "); @@ -369,7 +369,7 @@ public: { return; } - errorCodeScd4x = instance()->readMeasurement(co2, temperature, humidity); + errorCodeScd4x = instanceScd4x()->readMeasurement(co2, temperature, humidity); if (errorCodeScd4x) { Serial.print("Error trying to execute readMeasurement(): ");