mirror of
https://github.com/pelgraine/Meck.git
synced 2026-05-01 19:12:35 +02:00
add serial command support for regions, update serial settings guide
This commit is contained in:
@@ -57,7 +57,6 @@ All commands follow a simple pattern: `get` to read, `set` to write.
|
||||
| `get radio` | All radio params in one line |
|
||||
| `get utc` | UTC offset (hours) |
|
||||
| `get notify` | Keyboard flash notification (on/off) |
|
||||
| `get largefont` | Larger font mode (on/off) |
|
||||
| `get gps` | GPS status and interval |
|
||||
| `get pin` | BLE pairing PIN |
|
||||
| `get path.hash.mode` | Path hash size (0=1-byte, 1=2-byte, 2=3-byte) |
|
||||
@@ -65,10 +64,12 @@ All commands follow a simple pattern: `get` to read, `set` to write.
|
||||
| `get af` | Airtime factor |
|
||||
| `get multi.acks` | Redundant ACKs (0 or 1) |
|
||||
| `get int.thresh` | Interference threshold (0=disabled) |
|
||||
| `get tx.fail.reset` | TX fail reset threshold (0=disabled, default 3) |
|
||||
| `get rx.fail.reboot` | RX stuck reboot threshold (0=disabled, default 3) |
|
||||
| `get tx.fail.threshold` | TX fail reset threshold (0=disabled, default 3) |
|
||||
| `get rx.fail.threshold` | RX stuck reboot threshold (0=disabled, default 3) |
|
||||
| `get gps.baud` | GPS baud rate (0=compile-time default) |
|
||||
| `get channels` | List all channels with index numbers |
|
||||
| `get region` | Default region scope (e.g. `au-nsw`, or `none`) |
|
||||
| `get channels` | List all channels with index numbers and region scopes |
|
||||
| `get channel.scope <idx>` | Show region scope for a specific channel |
|
||||
| `get presets` | List all radio presets with parameters |
|
||||
| `get pubkey` | Device public key (hex) |
|
||||
| `get firmware` | Firmware version string |
|
||||
@@ -167,15 +168,6 @@ set notify on
|
||||
set notify off
|
||||
```
|
||||
|
||||
#### Larger Font Mode
|
||||
|
||||
Toggle larger text on channel messages, contacts, DM inbox, and repeater admin screens:
|
||||
|
||||
```
|
||||
set largefont on
|
||||
set largefont off
|
||||
```
|
||||
|
||||
#### BLE PIN
|
||||
|
||||
```
|
||||
@@ -248,8 +240,8 @@ Values: 0 (disabled, default) or 14+ (14 is the typical setting). Values between
|
||||
Automatically resets the radio hardware after this many consecutive failed transmission attempts. This recovers from "zombie radio" states where the SX1262 stops responding to send commands.
|
||||
|
||||
```
|
||||
set tx.fail.reset 3
|
||||
set tx.fail.reset 0
|
||||
set tx.fail.threshold 3
|
||||
set tx.fail.threshold 0
|
||||
```
|
||||
|
||||
Values: 0 (disabled) or 1–10 (default: 3). After the threshold is reached, the radio is reset and the failed packet is re-queued.
|
||||
@@ -259,8 +251,8 @@ Values: 0 (disabled) or 1–10 (default: 3). After the threshold is reached, the
|
||||
Automatically reboots the device after this many consecutive RX-stuck recovery failures. An RX-stuck event occurs when the radio is not in receive mode for 8 seconds despite automatic recovery attempts.
|
||||
|
||||
```
|
||||
set rx.fail.reboot 3
|
||||
set rx.fail.reboot 0
|
||||
set rx.fail.threshold 3
|
||||
set rx.fail.threshold 0
|
||||
```
|
||||
|
||||
Values: 0 (disabled) or 1–10 (default: 3). A full device reboot is a last resort — this should only trigger in rare cases of persistent radio hardware malfunction.
|
||||
@@ -276,6 +268,18 @@ set gps.baud 0
|
||||
|
||||
Valid rates: 0 (default), 4800, 9600, 19200, 38400, 57600, 115200.
|
||||
|
||||
#### Backlight (T5S3 E-Paper Pro Only)
|
||||
|
||||
Control the front-light on the T5S3 display:
|
||||
|
||||
```
|
||||
set backlight on
|
||||
set backlight off
|
||||
set backlight 128
|
||||
```
|
||||
|
||||
Values: `on`, `off`, or a brightness level from 0–255.
|
||||
|
||||
### Channel Management
|
||||
|
||||
#### List Channels
|
||||
@@ -287,11 +291,13 @@ get channels
|
||||
Output:
|
||||
|
||||
```
|
||||
[0] #public
|
||||
[1] #meck-test
|
||||
[2] #local-group
|
||||
[0] #public [*]
|
||||
[1] #meck-test [au-nsw]
|
||||
[2] #local-group [*]
|
||||
```
|
||||
|
||||
Each channel shows its region scope in brackets. `[*]` means the channel uses the device default region (or unscoped if no default is set). A specific name like `[au-nsw]` means that channel has its own region override.
|
||||
|
||||
#### Add a Hashtag Channel
|
||||
|
||||
```
|
||||
@@ -308,6 +314,82 @@ set channel.del 2
|
||||
|
||||
Channels are referenced by their index number (shown in `get channels`). Channel 0 (public) cannot be deleted. Remaining channels are automatically compacted after deletion.
|
||||
|
||||
### Region Scope
|
||||
|
||||
Regions limit how far your flood messages propagate through the mesh. When you set a region, outgoing messages are tagged with a transport code that repeaters use to decide whether to forward them. Messages sent without a region reach all repeaters via the default wildcard, same as always.
|
||||
|
||||
Meck does not pre-set any region on a fresh flash. Region names are determined by your local mesh community — check with your local group for the names in use. Common patterns follow ISO 3166 country/subdivision codes (e.g. `au` for Australia, `gb-eng` for England, `us-ca` for California), but communities may also use custom names for their area.
|
||||
|
||||
Region names must be lowercase alphanumeric characters and hyphens only, max 29 characters.
|
||||
|
||||
#### View Default Region
|
||||
|
||||
```
|
||||
get region
|
||||
```
|
||||
|
||||
Output:
|
||||
|
||||
```
|
||||
> au-nsw
|
||||
```
|
||||
|
||||
Or if no region is set:
|
||||
|
||||
```
|
||||
> (none — unscoped)
|
||||
```
|
||||
|
||||
#### Set Default Region
|
||||
|
||||
```
|
||||
set region au-nsw
|
||||
```
|
||||
|
||||
This applies to all channels and DMs unless a channel has its own region override.
|
||||
|
||||
#### Clear Default Region
|
||||
|
||||
```
|
||||
set region none
|
||||
```
|
||||
|
||||
Returns to unscoped mode — messages reach all repeaters.
|
||||
|
||||
#### View Channel Region
|
||||
|
||||
```
|
||||
get channel.scope 2
|
||||
```
|
||||
|
||||
Output:
|
||||
|
||||
```
|
||||
> #local-group scope: au-syd
|
||||
```
|
||||
|
||||
Or if the channel uses the device default:
|
||||
|
||||
```
|
||||
> #local-group scope: (device default)
|
||||
```
|
||||
|
||||
#### Set Channel Region
|
||||
|
||||
```
|
||||
set channel.scope 2 au-syd
|
||||
```
|
||||
|
||||
This overrides the device default for that specific channel.
|
||||
|
||||
#### Clear Channel Region
|
||||
|
||||
```
|
||||
set channel.scope 2 none
|
||||
```
|
||||
|
||||
Returns the channel to using the device default region.
|
||||
|
||||
### 4G Modem (4G Variant Only)
|
||||
|
||||
#### Enable / Disable Modem
|
||||
@@ -420,7 +502,7 @@ set channel.add local-group
|
||||
get all
|
||||
```
|
||||
|
||||
### Switching to a New Region
|
||||
### Switching to a Different Radio Preset
|
||||
|
||||
Moving from Australia to the US? One command:
|
||||
|
||||
@@ -434,6 +516,27 @@ Verify with:
|
||||
get radio
|
||||
```
|
||||
|
||||
### Setting Up Regions
|
||||
|
||||
Check with your local mesh community for the region names in use, then set your device default:
|
||||
|
||||
```
|
||||
set region au-nsw
|
||||
```
|
||||
|
||||
If you want a specific channel to use a different region (e.g. a nationwide channel):
|
||||
|
||||
```
|
||||
set channel.scope 1 au
|
||||
```
|
||||
|
||||
Verify everything:
|
||||
|
||||
```
|
||||
get region
|
||||
get channels
|
||||
```
|
||||
|
||||
### Custom Radio Configuration
|
||||
|
||||
If none of the presets match your local group or you need specific parameters, set them directly. You can do it all in one command:
|
||||
|
||||
@@ -2453,6 +2453,26 @@ void MyMesh::checkCLIRescueCmd() {
|
||||
Serial.printf(" > %lu (effective: %lu)\n",
|
||||
(unsigned long)_prefs.gps_baudrate, (unsigned long)effective);
|
||||
|
||||
} else if (strcmp(key, "region") == 0) {
|
||||
if (_prefs.default_scope_name[0]) {
|
||||
Serial.printf(" > %s\n", _prefs.default_scope_name);
|
||||
} else {
|
||||
Serial.println(" > (none — unscoped)");
|
||||
}
|
||||
} else if (memcmp(key, "channel.scope ", 14) == 0) {
|
||||
int idx = atoi(&key[14]);
|
||||
if (idx >= 0 && idx < MAX_GROUP_CHANNELS) {
|
||||
ChannelDetails ch;
|
||||
if (getChannel(idx, ch) && ch.name[0] != '\0') {
|
||||
Serial.printf(" > %s scope: %s\n", ch.name,
|
||||
ch.scope_name[0] ? ch.scope_name : "(device default)");
|
||||
} else {
|
||||
Serial.printf(" Error: channel %d is empty\n", idx);
|
||||
}
|
||||
} else {
|
||||
Serial.println(" Error: invalid channel index");
|
||||
}
|
||||
|
||||
} else if (strcmp(key, "radio") == 0) {
|
||||
Serial.printf(" > freq=%.3f bw=%.1f sf=%d cr=%d tx=%d\n",
|
||||
_prefs.freq, _prefs.bw, _prefs.sf, _prefs.cr, _prefs.tx_power_dbm);
|
||||
@@ -2467,7 +2487,11 @@ void MyMesh::checkCLIRescueCmd() {
|
||||
for (uint8_t i = 0; i < MAX_GROUP_CHANNELS; i++) {
|
||||
ChannelDetails ch;
|
||||
if (getChannel(i, ch) && ch.name[0] != '\0') {
|
||||
Serial.printf(" [%d] %s\n", i, ch.name);
|
||||
if (ch.scope_name[0]) {
|
||||
Serial.printf(" [%d] %s [%s]\n", i, ch.name, ch.scope_name);
|
||||
} else {
|
||||
Serial.printf(" [%d] %s [*]\n", i, ch.name);
|
||||
}
|
||||
found = true;
|
||||
} else {
|
||||
break;
|
||||
@@ -2514,6 +2538,8 @@ void MyMesh::checkCLIRescueCmd() {
|
||||
uint32_t eff_baud = _prefs.gps_baudrate ? _prefs.gps_baudrate : GPS_BAUDRATE;
|
||||
Serial.printf(" gps.baud: %lu\n", (unsigned long)eff_baud);
|
||||
}
|
||||
Serial.printf(" region: %s\n",
|
||||
_prefs.default_scope_name[0] ? _prefs.default_scope_name : "(none — unscoped)");
|
||||
#ifdef HAS_4G_MODEM
|
||||
Serial.printf(" modem: %s\n", ModemManager::loadEnabledConfig() ? "on" : "off");
|
||||
Serial.printf(" apn: %s\n", modemManager.getAPN());
|
||||
@@ -2548,7 +2574,11 @@ void MyMesh::checkCLIRescueCmd() {
|
||||
for (uint8_t i = 0; i < MAX_GROUP_CHANNELS; i++) {
|
||||
ChannelDetails ch;
|
||||
if (getChannel(i, ch) && ch.name[0] != '\0') {
|
||||
Serial.printf(" [%d] %s\n", i, ch.name);
|
||||
if (ch.scope_name[0]) {
|
||||
Serial.printf(" [%d] %s [%s]\n", i, ch.name, ch.scope_name);
|
||||
} else {
|
||||
Serial.printf(" [%d] %s [*]\n", i, ch.name);
|
||||
}
|
||||
chFound = true;
|
||||
} else {
|
||||
break;
|
||||
@@ -2942,6 +2972,54 @@ void MyMesh::checkCLIRescueCmd() {
|
||||
Serial.println(" Error: use 0 (default), 4800, 9600, 19200, 38400, 57600, or 115200");
|
||||
}
|
||||
|
||||
// Region scope commands
|
||||
} else if (memcmp(config, "region ", 7) == 0) {
|
||||
const char* name = &config[7];
|
||||
if (strcmp(name, "none") == 0 || strcmp(name, "clear") == 0 || name[0] == '\0') {
|
||||
memset(_prefs.default_scope_name, 0, sizeof(_prefs.default_scope_name));
|
||||
memset(_prefs.default_scope_key, 0, sizeof(_prefs.default_scope_key));
|
||||
savePrefs();
|
||||
Serial.println(" > region cleared (unscoped)");
|
||||
} else if (strlen(name) < 31) {
|
||||
strncpy(_prefs.default_scope_name, name, sizeof(_prefs.default_scope_name));
|
||||
_prefs.default_scope_name[30] = '\0';
|
||||
TransportKey key;
|
||||
deriveScopeKey(name, key);
|
||||
memcpy(_prefs.default_scope_key, key.key, sizeof(_prefs.default_scope_key));
|
||||
savePrefs();
|
||||
Serial.printf(" > region = %s\n", _prefs.default_scope_name);
|
||||
} else {
|
||||
Serial.println(" Error: region name too long (max 29 chars)");
|
||||
}
|
||||
} else if (memcmp(config, "channel.scope ", 14) == 0) {
|
||||
// set channel.scope <idx> <name> (or "set channel.scope <idx> none" to clear)
|
||||
int idx = atoi(&config[14]);
|
||||
const char* rest = strchr(&config[14], ' ');
|
||||
if (idx >= 0 && idx < MAX_GROUP_CHANNELS && rest) {
|
||||
rest++; // skip space
|
||||
ChannelDetails ch;
|
||||
if (getChannel(idx, ch) && ch.name[0] != '\0') {
|
||||
if (strcmp(rest, "none") == 0 || strcmp(rest, "clear") == 0) {
|
||||
memset(ch.scope_name, 0, sizeof(ch.scope_name));
|
||||
} else if (strlen(rest) < 31) {
|
||||
strncpy(ch.scope_name, rest, sizeof(ch.scope_name));
|
||||
ch.scope_name[30] = '\0';
|
||||
} else {
|
||||
Serial.println(" Error: scope name too long (max 29 chars)");
|
||||
cli_command[0] = 0;
|
||||
return;
|
||||
}
|
||||
setChannel(idx, ch);
|
||||
saveChannels();
|
||||
Serial.printf(" > %s scope = %s\n", ch.name,
|
||||
ch.scope_name[0] ? ch.scope_name : "(device default)");
|
||||
} else {
|
||||
Serial.printf(" Error: channel %d is empty\n", idx);
|
||||
}
|
||||
} else {
|
||||
Serial.println(" Usage: set channel.scope <idx> <name|none>");
|
||||
}
|
||||
|
||||
// Backlight control (T5S3 E-Paper Pro only)
|
||||
} else if (memcmp(config, "backlight ", 10) == 0) {
|
||||
#if defined(LilyGo_T5S3_EPaper_Pro)
|
||||
@@ -3046,6 +3124,12 @@ void MyMesh::checkCLIRescueCmd() {
|
||||
Serial.println(" set preset <name|num> Apply radio preset");
|
||||
Serial.println(" set channel.add <name> Add hashtag channel");
|
||||
Serial.println(" set channel.del <idx> Delete channel by index");
|
||||
Serial.println("");
|
||||
Serial.println(" Regions:");
|
||||
Serial.println(" get/set region Device default region (e.g. au-nsw)");
|
||||
Serial.println(" set region none Clear default region (unscoped)");
|
||||
Serial.println(" get channel.scope <i> Show scope for channel i");
|
||||
Serial.println(" set channel.scope <i> <name|none>");
|
||||
#ifdef HAS_4G_MODEM
|
||||
Serial.println("");
|
||||
Serial.println(" 4G modem:");
|
||||
|
||||
Reference in New Issue
Block a user