5#include "ustd_platform.h"
10#ifdef USTD_FEATURE_FILESYSTEM
11#include "filesystem.h"
22#if defined(__ESP__) || defined(__UNIXOID__)
23typedef std::function<void(String command, String args, Print *printer)>
T_COMMANDFN;
25typedef ustd::function<void(String command, String args, Print *printer)>
T_COMMANDFN;
90#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
96 ustd::array<T_COMMAND> commands;
97 int commandHandle = 0;
104 ustd::array<int> subsub;
110 : printer(printer), name(name) {
118#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
119 for (
unsigned int i = 0; i < commands.length(); i++) {
120 free(commands[i].command);
135#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
146 cmd.id = ++commandHandle;
148 cmd.command = (
char *)malloc(command.length() + 1);
149 strcpy(cmd.command, command.c_str());
150 return commands.add(cmd) != -1 ? commandHandle : -1;
158 for (
unsigned int i = 0; i < commands.length(); i++) {
159 if (command == (
const char *)commands[i].command) {
160 return commands.erase(i);
171 for (
unsigned int i = 0; i < commands.length(); i++) {
172 if (commands[i].
id == commandHandle) {
173 return commands.erase(i);
181 virtual void prompt() {
182 printer->print(
"\rmuwerk> ");
183 printer->print(args);
186 size_t outputf(
const char *format, ...) {
188 va_start(arg, format);
191 size_t len = vsnprintf(temp,
sizeof(temp), format, arg);
193 if (len >
sizeof(temp) - 1) {
194 buffer =
new char[len + 1];
198 va_start(arg, format);
199 vsnprintf(buffer, len + 1, format, arg);
202 len = printer->write((
const uint8_t *)buffer, len);
203 if (buffer != temp) {
219 void commandparser() {
220 String cmd = pullArg();
224 }
else if (cmd ==
"reboot") {
226 }
else if (cmd ==
"wifi") {
229 }
else if (cmd ==
"uname") {
231 }
else if (cmd ==
"uptime") {
233 }
else if (cmd ==
"info") {
235#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
236 }
else if (cmd ==
"mem") {
239#ifdef USTD_FEATURE_CLK_READ
240 }
else if (cmd ==
"date") {
243 }
else if (cmd ==
"sub") {
245 }
else if (cmd ==
"pub") {
247#ifdef USTD_FEATURE_FILESYSTEM
248 }
else if (cmd ==
"ls") {
250 }
else if (cmd ==
"rm") {
252 }
else if (cmd ==
"cat") {
254 }
else if (cmd ==
"jf") {
257 }
else if (!cmd_custom(cmd)) {
258 printer->println(
"-mush: " + cmd +
": command not found");
263 String help =
"commands: help, pub, sub, uname, uptime, info";
264#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
267#ifdef USTD_FEATURE_CLK_READ
270#ifdef USTD_FEATURE_FILESYSTEM
271 help +=
", ls, rm, cat, jf";
274 help +=
", debug, wifi, reboot";
276#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
277 for (
unsigned int i = 0; i < commands.length(); i++) {
279 help += commands[i].command;
282 printer->println(help);
286 String arg = pullArg();
287 if (arg ==
"-h" || arg ==
"-H") {
288 printer->println(
"usage: sub [all | none]");
289 printer->println(
"usage: sub topic [topic [..]]");
291 }
else if (arg ==
"none") {
293 }
else if (arg ==
"all") {
295 }
else if (arg.length()) {
302 }
while (arg.length());
304 if (subsub.length()) {
306 printer->println(
"All topics subscribed");
308 outputf(
"%i subscriptions\r\n", subsub.length());
311 printer->println(
"No subscriptions");
316 String arg = pullArg();
317 if (arg ==
"-h" || arg ==
"-H" || arg ==
"") {
318 printer->println(
"usage: pub <topic> <message>");
321 pSched->
publish(arg, args, name);
326 printer->println(
"Restarting...");
331 printer->println(
"WiFi Information:");
332 printer->println(
"-----------------");
333 WiFi.printDiag(*printer);
338 unsigned long uptime = pSched->
getUptime();
339 unsigned long days = uptime / 86400;
340 unsigned long hours = (uptime % 86400) / 3600;
341 unsigned long minutes = (uptime % 3600) / 60;
342 unsigned long seconds = uptime % 60;
344 printer->print(
"up ");
346 printer->print(days);
348 printer->print(
" days, ");
350 printer->print(
" day, ");
353 outputf(
"%2.2lu:%2.2lu:%2.2lu\r\n", hours, minutes, seconds);
356#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
360#if defined(__ESP32__) || defined(__ESP32_RISC__)
361 printer->println(
"Internal Ram:");
362 printer->println(
"-------------");
363 outputf(
"Size: %u B\r\n", (
unsigned int)ESP.getHeapSize());
364 outputf(
"Free: %u B\r\n", (
unsigned int)ESP.getFreeHeap());
365 outputf(
"Used: %u B\r\n", (
unsigned int)ESP.getHeapSize() - ESP.getFreeHeap());
366 outputf(
"Peak: %u B\r\n", (
unsigned int)ESP.getHeapSize() - ESP.getMinFreeHeap());
367 outputf(
"MaxB: %u B\r\n", (
unsigned int)ESP.getMaxAllocHeap());
370 printer->println(
"SPI Ram:");
371 printer->println(
"--------");
372 outputf(
"Size: %u B\r\n", (
unsigned int)ESP.getPsramSize());
373 outputf(
"Free: %u B\r\n", (
unsigned int)ESP.getFreePsram());
374 outputf(
"Used: %u B\r\n", (
unsigned int)ESP.getPsramSize() - ESP.getFreePsram());
375 outputf(
"Peak: %u B\r\n", (
unsigned int)ESP.getPsramSize() - ESP.getMinFreePsram());
376 outputf(
"MaxB: %u B\r\n", (
unsigned int)ESP.getMaxAllocPsram());
379 printer->println(
"Internal Ram:");
380 printer->println(
"-------------");
381 outputf(
"Free: %u B\r\n", (
unsigned int)ESP.getFreeHeap());
382 outputf(
"Fragmentation: %u%%\r\n", (
unsigned int)ESP.getHeapFragmentation());
383 outputf(
"Largest Free Block: %u B\r\n", (
unsigned int)ESP.getMaxFreeBlockSize());
388 printer->println(
"Memory:");
389 printer->println(
"-------");
390 printer->print(
"Free: ");
391 printer->print(freeMemory());
392 printer->println(
" B");
395 printer->println(
"No information available");
405#if defined(__ESP32__) || defined(__ESP32_RISC__)
406 printer->println(
"ESP32 Information:");
407 printer->println(
"------------------");
409 outputf(
"Silicon type: RISC-V");
411 outputf(
"Silicon type: Tensilica");
413 outputf(
"Chip Verion: %u\r\n", (
unsigned int)ESP.getChipRevision());
414 outputf(
"CPU Frequency: %u MHz\r\n", (
unsigned int)ESP.getCpuFreqMHz());
415 outputf(
"SDK Version: %s\r\n", ESP.getSdkVersion());
416 outputf(
"Program Size: %u B\r\n", (
unsigned int)ESP.getSketchSize());
417 outputf(
"Program Free: %u B\r\n", (
unsigned int)ESP.getFreeSketchSpace());
418 outputf(
"Flash Chip Size: %u B\r\n", (
unsigned int)ESP.getFlashChipSize());
419 outputf(
"Flash Chip Speed: %.2f MHz\r\n", (
float)ESP.getFlashChipSpeed() / 1000000.0);
422 printer->println(
"ESP Information:");
423 printer->println(
"----------------");
424 outputf(
"Chip ID: %u\r\n", (
unsigned int)ESP.getChipId());
425 outputf(
"Chip Version: %s\r\n", ESP.getCoreVersion().c_str());
426 outputf(
"SDK Version: %s\r\n", ESP.getSdkVersion());
427 outputf(
"CPU Frequency: %u MHz\r\n", (
unsigned int)ESP.getCpuFreqMHz());
428 outputf(
"Program Size: %u B\r\n", (
unsigned int)ESP.getSketchSize());
429 outputf(
"Program Free: %u B\r\n", (
unsigned int)ESP.getFreeSketchSpace());
430 outputf(
"Flash Chip ID: %u\r\n", (
unsigned int)ESP.getFlashChipId());
431 outputf(
"Flash Chip Size: %u B\r\n", (
unsigned int)ESP.getFlashChipSize());
432 outputf(
"Flash Chip Real Size: %u B\r\n", (
unsigned int)ESP.getFlashChipRealSize());
433 outputf(
"Flash Chip Speed: %.2f MHz\r\n", (
float)ESP.getFlashChipSpeed() / 1000000.0);
434 outputf(
"Last Reset Reason: %s\r\n", ESP.getResetReason().c_str());
439#if USTD_FEATURE_MEMORY < USTD_FEATURE_MEM_8K
440 printer->print(
"mem ");
441 printer->print(freeMemory());
442 printer->print(
", ");
443 printer->print((
float)(F_CPU) / 1000000.0, 2);
444 printer->println(
" MHz");
447 printer->print(
"CPU Frequency: ");
448 printer->print((
float)(F_CPU) / 1000000.0, 2);
449 printer->println(
" MHz");
450 outputf(
"Free Memory: %u B\r\n", (
unsigned int)freeMemory());
454#if USTD_FEATURE_MEMORY < USTD_FEATURE_MEM_8K
455 printer->println(
"No info");
457 printer->println(
"No information available");
464 void cmd_uname(
char opt =
'\0',
bool crlf =
true) {
469 switch (arg.length()) {
484 printer->print(
"munix");
488 cmd_uname(
's',
false);
490 cmd_uname(
'n',
false);
492 cmd_uname(
'v',
false);
496#if defined(__ESP32__) || defined(__ESP32_RISC__)
497 printer->print(WiFi.getHostname());
499 printer->print(WiFi.hostname());
502 printer->print(
"localhost");
507 printer->print(ESP.getSdkVersion());
509 printer->print(
"unknown");
514#if defined(__ESP32__)
515 printer->print(
"ESP32 (Tensilica)");
516#elif defined(__ESP32_RISC__)
517 printer->print(
"ESP32 (RISC-V)");
519 printer->print(
"ESP");
523 printer->print(
"Arduino");
525 printer->print(
"Unknown");
530 cmd_uname(
'p',
false);
531 printer->print(
" Version ");
532 cmd_uname(
'r',
false);
533 printer->print(
": " __DATE__
" " __TIME__);
535 printer->print(
"; PlatformIO " + String(PLATFORMIO));
536#ifdef ARDUINO_VARIANT
537 printer->print(
", " ARDUINO_VARIANT);
543 printer->println(
"usage: uname [-amnprsv]");
551#ifdef USTD_FEATURE_CLK_READ
553 String arg = pullArg();
556#ifdef USTD_FEATURE_CLK_SET
557 printer->println(
"usage: date [[YYYY-MM-DD] hh:mm:ss]");
559 printer->println(
"usage: date");
563 time_t t = time(
nullptr);
564 struct tm *lt = localtime(&t);
566 printer->println(
"error: current time cannot be determined");
570 outputf(
"%4.4i-%2.2i-%2.2i %2.2i:%2.2i:%2.2i - epoch %lu\r\n", (
int)lt->tm_year + 1900,
571 (
int)lt->tm_mon + 1, (
int)lt->tm_mday, (
int)lt->tm_hour, (
int)lt->tm_min,
574#ifdef USTD_FEATURE_CLK_SET
577 int i = sscanf(arg.c_str(),
"%4i-%2i-%2i", <->tm_year, <->tm_mon, <->tm_mday);
578 if (i != 3 || lt->tm_year < 1970 || lt->tm_year > 2038 || lt->tm_mon < 1 ||
579 lt->tm_mon > 11 || lt->tm_mday < 1 || lt->tm_mday > 31) {
580 printer->println(
"error: " + arg +
" is not a date");
588 int i = sscanf(arg.c_str(),
"%2i:%2i:%2i", <->tm_hour, <->tm_min, <->tm_sec);
589 if (i != 3 || lt->tm_hour < 0 || lt->tm_hour > 23 || lt->tm_min < 0 ||
590 lt->tm_min > 59 || lt->tm_sec < 0 || lt->tm_sec > 59) {
591 printer->println(
"error: " + arg +
" is not a time");
594 time_t newt = mktime(lt);
596 DBGF(
"Setting date to: %4.4i-%2.2i-%2.2i %2.2i:%2.2i:%2.2i - epoch %lu\n",
597 lt->tm_year + 1900, lt->tm_mon + 1, lt->tm_mday, lt->tm_hour, lt->tm_min,
598 lt->tm_sec, (
unsigned long)newt);
603 printer->print(
"Date set to: ");
610#ifdef USTD_FEATURE_FILESYSTEM
612 ustd::array<String> paths;
613 bool extended =
false;
615 for (String arg = pullArg(); arg.length(); arg = pullArg()) {
616 if (arg ==
"-h" || arg ==
"-H") {
617 printer->println(
"\rusage: ls [-l] <path> [<path> [...]]");
619 }
else if (arg ==
"-l" || arg ==
"-L" || arg ==
"-la") {
626 if (paths.length() == 0) {
631 for (
unsigned int i = 0; i < paths.length(); i++) {
635 outputf(
"%crw-rw-rw- root root %10u ", (dir.isDirectory() ?
'd' :
'-'),
637 time_t tt = dir.fileTime();
638 struct tm *lt = localtime(&tt);
640 outputf(
"%4.4i-%2.2i-%2.2i %2.2i:%2.2i:%2.2i ", lt->tm_year + 1900,
641 lt->tm_mon + 1, lt->tm_mday, lt->tm_hour, lt->tm_min, lt->tm_sec);
644 printer->println(dir.fileName());
650 String arg = pullArg();
651 if (arg ==
"-h" || arg ==
"-H") {
652 printer->println(
"usage: rm <filename>");
656#ifndef USE_SERIAL_DBG
657 printer->println(
"error: File " + arg +
" can't be deleted.");
663 String arg = pullArg();
664 if (arg ==
"-h" || arg ==
"-H") {
665 printer->println(
"usage: cat <filename>");
669 fs::File f =
fsOpen(arg,
"r");
671#ifndef USE_SERIAL_DBG
672 printer->println(
"error: File " + arg +
" can't be opened.");
676 if (!f.available()) {
680 while (f.available()) {
682 printer->println(f.readStringUntil(
'\n'));
688 String arg = pullArg();
691 if (arg ==
"-h" || arg ==
"") {
693 }
else if (arg ==
"get") {
695 }
else if (arg ==
"set") {
697 }
else if (arg ==
"del") {
700 printer->println(
"error: bad command " + arg +
"specified.");
706 printer->println(
"usage: jf get <jsonpath>");
707 printer->println(
"usage: jf set <jsonpath> <jsonvalue>");
708 printer->println(
"usage: jf del <jsonpath>");
712 String arg = pullArg();
714 if (arg ==
"-h" || arg ==
"-H" || arg ==
"") {
723#ifndef USE_SERIAL_DBG
724 printer->println(
"error: Cannot read value " + arg);
730 String type = JSON.typeof(value);
731 printer->print(
": " + type);
732 if (type ==
"unknown") {
733 printer->println(
"");
735 printer->println(
", " + JSON.stringify(value));
740 String arg = pullArg();
742 if (arg ==
"-h" || arg ==
"-H" || arg ==
"" || args ==
"") {
748 JSONVar value = JSON.parse(args);
749 if (JSON.typeof(value) ==
"undefined") {
750 printer->println(
"error: Cannot parse value " + args);
754#ifndef USE_SERIAL_DBG
755 printer->println(
"error: Failed to write value " + arg);
761 String arg = pullArg();
763 if (arg ==
"-h" || arg ==
"-H" || arg ==
"") {
770#ifndef USE_SERIAL_DBG
771 printer->println(
"error: Failed to delete value " + arg);
777 bool cmd_custom(String &cmd) {
778#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
779 for (
unsigned int i = 0; i < commands.length(); i++) {
780 if (cmd == commands[i].command) {
781 commands[i].fn(cmd, args, printer);
789 bool addsub(String topic) {
792 }
else if (topic ==
"#" && subsub.length()) {
796 pSched->
subscribe(tID, topic, [
this](String topic, String msg, String originator) {
797 printer->print(
"\r>> ");
798#if USTD_FEATURE_MEMORY >= USTD_FEATURE_MEM_8K
799 if (originator.length()) {
801 printer->print(originator);
802 printer->print(
"] ");
805 printer->print(topic);
807 printer->println(msg);
818 if (subsub.length()) {
819 for (
unsigned int i = 0; i < subsub.length(); i++) {
827 String pullArg(String defValue =
"") {
874#if USTD_FEATURE_MEMORY < USTD_FEATURE_MEM_8K
875#define MU_SERIAL_BUF_SIZE 0
878#define MU_SERIAL_BUF_SIZE 2
882#ifndef MU_SERIAL_BUF_SIZE
883#define MU_SERIAL_BUF_SIZE 16
885#ifndef MU_SERIAL_CHUNK_SIZE
886#define MU_SERIAL_CHUNK_SIZE 32
892#if MU_SERIAL_BUF_SIZE > 0
893 char buffer[MU_SERIAL_BUF_SIZE];
902#if MU_SERIAL_BUF_SIZE > 0
904 memset(buffer, 0,
sizeof(buffer) /
sizeof(
char));
908 void begin(
Scheduler *_pSched, String initialCommand =
"",
unsigned long pollRate = 60) {
919 begin(_pSched, &Serial, initialCommand, pollRate);
923 unsigned long pollRate = 60) {
937 }
else if (pollRate > 1000) {
944 tID = pSched->
add([
this]() { this->loop(); }, name, 60000);
951#if MU_SERIAL_BUF_SIZE > 0
952 virtual void prompt() {
954 printer->print(buffer);
960 bool changed =
false;
963 while ((incomingByte = ((Stream *)printer)->read()) != -1 && count < MU_SERIAL_CHUNK_SIZE) {
965 switch (incomingByte) {
970#if MU_SERIAL_BUF_SIZE > 0
974 }
else if (args.length()) {
975 args.remove(args.length() - 1, 1);
979 args.remove(args.length() - 1, 1);
986#if MU_SERIAL_BUF_SIZE > 0
996#if MU_SERIAL_BUF_SIZE > 0
997 *pcur = (char)incomingByte;
999 if (pcur == buffer + (
sizeof(buffer) /
sizeof(
char)) - 1) {
1004 args += (char)incomingByte;
1016#if MU_SERIAL_BUF_SIZE > 0
1022 memset(buffer, 0,
sizeof(buffer) /
sizeof(
char));
muwerk Console Class
Definition console.h:88
bool unextend(int commandHandle)
Definition console.h:166
bool unextend(String command)
Definition console.h:153
int extend(String command, T_COMMANDFN handler)
Definition console.h:136
bool execute(String command)
Definition console.h:126
Console(String name, Print *printer)
Definition console.h:109
muwerk Scheduler Class
Definition scheduler.h:199
int add(T_TASK task, String name, unsigned long minMicroSecs=100000L, T_PRIO prio=PRIO_NORMAL)
Definition scheduler.h:475
unsigned long getUptime()
Definition scheduler.h:554
bool publish(String topic, String msg="", String originator="")
Definition scheduler.h:367
int subscribe(int taskID, String topic, T_SUBS subs, String originator="")
Definition scheduler.h:395
bool unsubscribe(int subscriptionHandle)
Definition scheduler.h:428
muwerk Serial Console Class
Definition console.h:890
void begin(Scheduler *_pSched, String initialCommand="", unsigned long pollRate=60)
Definition console.h:908
void begin(Scheduler *_pSched, Stream *pSerial, String initialCommand="", unsigned long pollRate=60)
Definition console.h:922
SerialConsole()
Definition console.h:898
muwerk JSON File Class
Definition jsonfile.h:67
bool remove(String key)
Definition jsonfile.h:206
bool writeJsonVar(String key, JSONVar &value)
Definition jsonfile.h:685
bool readJsonVar(String key, JSONVar &value)
Definition jsonfile.h:228
The muwerk namespace.
Definition console.h:15
fs::File fsOpen(String filename, String mode)
Definition filesystem.h:113
fs::Dir fsOpenDir(String path)
Definition filesystem.h:134
String shift(String &src, char delimiter=' ', String defValue="")
Definition muwerk.h:94
bool fsDelete(String filename)
Definition filesystem.h:97
std::function< void(String command, String args, Print *printer)> T_COMMANDFN
Console Extension Function.
Definition console.h:23