Mesh Bot for Network Testing and BBS Activities
Mesh Bot is a feature-rich Python bot designed to enhance your Meshtastic network experience. It provides powerful tools for network testing, messaging, games, and more—all via text-based message delivery. Whether you want to test your mesh, send messages, or play games, mesh_bot.py has you covered.
TLDR
Key Features
Intelligent Keyword Responder
- Automated Responses: Detects keywords like "ping" and replies with "pong" in direct messages (DMs) or group channels.
- Customizable Triggers: Monitors group channels for specific keywords and sends custom responses.
- Emergency Detection: Watches for emergency-related keywords and alerts a wider audience.
- New Node Greetings: Automatically welcomes new nodes joining the mesh.
Network Tools
- Mesh Testing: Use
pingto test message delivery with realistic packets. - Hardware Testing: The
testcommand sends incrementally sized data to test radio buffer limits. - Network Monitoring: Alerts for noisy nodes, tracks node locations, and suggests optimal relay placement.
Multi-Radio/Node Support
- Simultaneous Monitoring: Observe up to nine networks at once.
- Flexible Messaging: Send mail and messages between networks.
Advanced Messaging Capabilities
- Mail Messaging: Leave messages for other devices; delivered as DMs when the device is next seen. Use
bbspost @nodeNumber #messageorbbspost @nodeShortName #message. - Message Scheduler: Automate messages such as weather updates or net reminders.
- Store and Forward: Retrieve missed messages with the
messagescommand; optionally log messages to disk. - BBS Linking: Connect multiple bots to expand BBS coverage.
- E-Mail/SMS Integration: Send mesh messages to email or SMS for broader reach.
- New Node Greetings: Automatically greet new nodes via text.
Interactive AI and Data Lookup
- Weather, Earthquake, River, and Tide Data: Get local alerts and info from NOAA/USGS; uses Open-Meteo for areas outside NOAA coverage.
- Wikipedia Search: Retrieve summaries from Wikipedia.
- Ollama LLM Integration: Query the Ollama AI for advanced responses.
- Satellite Passes: Find upcoming satellite passes for your location.
- GeoMeasuring Tools: Calculate distances and midpoints using collected GPS data; supports Fox & Hound direction finding.
Proximity Alerts
- Location-Based Alerts: Get notified when members arrive at a configured latitude/longitude—ideal for campsites, geo-fences, or remote locations. Optionally, trigger scripts, send emails, or automate actions (e.g., change node config, turn on lights, or drop an
alert.txtfile to start a survey or game). - Customizable Triggers: Use proximity events for creative applications like "king of the hill" or 🧭 geocache games by adjusting the alert cycle.
- High Flying Alerts: Receive notifications when nodes with high altitude are detected on the mesh.
- Voice/Command Triggers: Activate bot functions using keywords or voice commands (see Voice Commands for "Hey Chirpy!" support).
Radio Frequency Monitoring
- SNR RF Activity Alerts: Monitor radio frequencies and receive alerts when high SNR (Signal-to-Noise Ratio) activity is detected.
- Hamlib Integration: Use Hamlib (rigctld) to monitor the S meter on a connected radio.
- Speech-to-Text Broadcasting: Convert received audio to text using Vosk and broadcast it to the mesh.
Check-In / Check-Out & Asset Tracking
- Asset Tracking: Maintain a check-in/check-out list for nodes or assets—ideal for accountability of people and equipment (e.g., Radio-Net, FEMA, trailhead groups).
Fun and Games
- Built-in Games: Play classic games like DopeWars, Lemonade Stand, BlackJack, and Video Poker directly via DM.
- FCC ARRL QuizBot: Practice for the ham radio exam with the integrated quiz bot.
- Command-Based Gameplay: Use the
gamescommand to view available games and start playing. - Telemetry Leaderboard: Compete for fun stats like lowest battery or coldest temperature.
QuizMaster
- Group Quizzes: Admins can start and stop quiz games for groups.
- Player Participation: Players join with
q: join, leave withq: leave, and answer questions by prefixing their answer withq:, e.g.,q: 42. - Scoring & Leaderboards: Check your score with
q: scoreand see the top performers withq: top. - Admin Controls: QuizMasters (from
bbs_admin_list) can useq: start,q: stop, andq: broadcast <message>to manage games.
Survey Module
- Custom Surveys: Create and manage surveys by editing JSON files in
data/survey. Multiple surveys are supported (e.g.,survey snow). - User Feedback: Users participate via DM; responses are logged for review.
- Reporting: Retrieve survey results with
survey reportorsurvey report <surveyname>.
EAS Alerts
- FEMA iPAWS/EAS Alerts: Receive Emergency Alerts from FEMA via API on internet-connected nodes.
- NOAA EAS Alerts: Get Emergency Alerts from NOAA via API.
- USGS Volcano Alerts: Receive volcano alerts from USGS via API.
- Offline EAS Alerts: Report EAS alerts over the mesh using external tools, even without internet.
- NINA Alerts (Germany): Receive emergency alerts from the xrepository.de feed for Germany.
File Monitor Alerts
- File Monitoring: Watch a text file for changes and broadcast updates to the mesh channel.
- News File Access: Retrieve the contents of a news file on request; supports multiple news sources or files.
- Shell Command Access: Execute shell commands via DM with replay protection (admin only).
Data Reporting
- HTML Reports: Visualize bot traffic and data flows with a built-in HTML generator. See data reporting for details.
- RSS & News Feeds: Receive news and data from multiple sources directly on the mesh.
Robust Message Handling
- Automatic Message Chunking: Messages over 160 characters are automatically split to ensure reliable delivery across multiple hops.
Getting Started
This project is developed on Linux (specifically a Raspberry Pi) but should work on any platform where the Meshtastic protobuf API modules are supported, and with any compatible Meshtastic hardware. For pico or low-powered devices, see projects for embedding, armbian or buildroot, also see femtofox for running on luckfox hardware. If you need a local console consider the firefly project.
🥔 Please use responsibly and follow local rulings for such equipment. This project captures packets, logs them, and handles over the air communications which can include PII such as GPS locations.
Quick Setup
Clone the Repository
If you dont have git you will need it sudo apt-get install git
git clone https://github.com/spudgunman/meshing-around
- Automated Installation:
install.shwill automate optional venv and requirements installation. - Launch Script:
launch.shonly used in a venv install, to launch the bot and the report generator.
Full list of commands for the bot
Games (via DM only)
QuizMaster
To use QuizMaster the bbs_admin_list is the QuizMaster, who can q: start and q: stop to start and stop the game, q: broadcast <message> to send a message to all players.
Players can q: join to join the game, q: leave to leave the game, q: score to see their score, and q: top to see the top 3 players.
To Answer a question, just type the answer prefixed with q: <answer>
Survey
To use the Survey feature edit the json files in data/survey multiple surveys are possible such as survey snow you can pull data back with survey report or survey report snow
Other Install Options
Docker Installation - handy for windows
See further info on the docker.md
Manual Install
Install the required dependencies using pip:
pip install -r requirements.txt
Copy the configuration template to config.ini and edit it to suit your needs:
cp config.template config.ini
Configuration Guide
The following is documentation for the config.ini file
If you have not done so, or want to 'factory reset', copy the config.template to config.ini and set the appropriate interface for your method (serial/ble/tcp). While BLE and TCP will work, they are not as reliable as serial connections. There is a watchdog to reconnect TCP if possible. To get the BLE MAC address, use:
meshtastic --ble-scan
Note: The code has been tested with a single BLE device and is written to support only one BLE port.
# config.ini
# type can be serial, tcp, or ble.
# port is the serial port to use; commented out will try to auto-detect
# hostname is the IP/DNS and port for tcp type default is host:4403
# mac is the MAC address of the device to connect to for BLE type
[interface]
type = serial
# port = '/dev/ttyUSB0'
# hostname = 192.168.0.1
# mac = 00:11:22:33:44:55
# Additional interface for dual radio support. See config.template for more.
[interface2]
enabled = False
General Settings
The following settings determine how the bot responds. By default, the bot will not spam the default channel. Setting respond_by_dm_only to True will force all messages to be sent via DM, which may not be desired. Setting it to [False] will allow responses in the channel for all to see. If you have no default channel you can set this value to -1 or any unused channel index. You can also have the bot ignore the defaultChannel for any commands, but still observe the channel.
[general]
respond_by_dm_only = True
defaultChannel = 0
ignoreDefaultChannel = False # ignoreDefaultChannel, the bot will ignore the default channel set above
ignoreChannels = # ignoreChannels is a comma separated list of channels to ignore, e.g. 4,5
cmdBang = False # require ! to be the first character in a command
explicitCmd = True # require explicit command, the message will only be processed if it starts with a command word disable to get more activity
Location Settings
The weather forecasting defaults to NOAA, for locations outside the USA, you can set UseMeteoWxAPI to True, to use a global weather API. The lat and lon are default values when a node has no location data, as well as the default for all NOAA, repeater lookup. It is also the center of radius for Sentry.
[location]
enabled = True
lat = 48.50
lon = -123.0
# To fuzz the location of the above
fuzzConfigLocation = True
# Fuzz all values in all data
fuzzItAll = False
UseMeteoWxAPI = True
coastalEnabled = False # NOAA Coastal Data Enable NOAA Coastal Waters Forecasts and Tide
# Find the correct coastal weather directory at https://tgftp.nws.noaa.gov/data/forecasts/marine/coastal/
# this map can help https://www.weather.gov/marine select location and then look at the 'Forecast-by-Zone Map'
myCoastalZone = https://tgftp.nws.noaa.gov/data/forecasts/marine/coastal/pz/pzz135.txt # myCoastalZone is the .txt file with the forecast data
coastalForecastDays = 3 # number of data points to return, default is 3
Module Settings
Modules can be enabled or disabled as needed. They are essentally larger functions of code which you may not want on your mesh or in memory space.
[bbs]
enabled = False
[general]
DadJokes = False
StoreForward = False
History
The history command shows the last commands the user ran, and [lheard] reflects the last users on the bot.
enableCmdHistory = True # history command enabler
lheardCmdIgnoreNodes = # command history ignore list ex: 2813308004,4258675309
Sentry Settings
Sentry Bot detects anyone coming close to the bot-node. uses the Location Lat/Lon value.
SentryEnabled = True # detect anyone close to the bot
emailSentryAlerts = True # if SMTP enabled send alert to sysop email list
SentryRadius = 100 # radius in meters to detect someone close to the bot
SentryChannel = 9 # holdoff time multiplied by seconds(20) of the watchdog
SentryHoldoff = 2 # channel to send a message to when the watchdog is triggered
sentryIgnoreList = # list of ignored nodes numbers ex: 2813308004,4258675309
highFlyingAlert = True # HighFlying Node alert
highFlyingAlertAltitude = 2000 # Altitude in meters to trigger the alert
highflyOpenskynetwork = True # check with OpenSkyNetwork if highfly detected for aircraft
E-Mail / SMS Settings
To enable connectivity with SMTP allows messages from meshtastic into SMTP. The term SMS here is for connection via carrier email
[smtp]
# enable or disable the SMTP module, minimum required for outbound notifications
enableSMTP = True # enable or disable the IMAP module for inbound email, not implemented yet
enableImap = False # list of Sysop Emails separate with commas, used only in emergency responder currently
sysopEmails =
# See config.template for all the SMTP settings
SMTP_SERVER = smtp.gmail.com
SMTP_AUTH = True
EMAIL_SUBJECT = Meshtastic✉️
Emergency Response Handler
Traps the following ("emergency", "911", "112", "999", "police", "fire", "ambulance", "rescue") keywords. Responds to the user, and calls attention to the text message in logs and via another network or channel.
[emergencyHandler]
enabled = True # enable or disable the emergency response handler
alert_channel = 2 # channel to send a message to when the emergency handler is triggered
alert_interface = 1
EAS Alerting
To Alert on Mesh with the EAS API you can set the channels and enable, checks every 20min.
FEMA iPAWS/EAS and NINA
This uses USA: SAME, FIPS, to locate the alerts in the feed. By default ignoring Test messages.
eAlertBroadcastEnabled = False # Goverment IPAWS/CAP Alert Broadcast
eAlertBroadcastCh = 2,3 # Goverment Emergency IPAWS/CAP Alert Broadcast Channels
ignoreFEMAenable = True # Ignore any headline that includes followig word list
ignoreFEMAwords = test,exercise
# comma separated list of FIPS codes to trigger local alert. find your FIPS codes at https://en.wikipedia.org/wiki/Federal_Information_Processing_Standard_state_code
myFIPSList = 57,58,53
# find your SAME https://www.weather.gov/nwr/counties comma separated list of SAME code to further refine local alert.
mySAMEList = 053029,053073
# To use other country services enable only a single optional serivce
enableDEalerts = False # Use DE Alert Broadcast Data see template for filters
myRegionalKeysDE = 110000000000,120510000000
NOAA EAS
This uses the defined lat-long of the bot for collecting of data from the API. see File-Monitoring for ideas to collect EAS alerts from a RTL-SDR.
wxAlertBroadcastEnabled = True # EAS Alert Broadcast
wxAlertBroadcastCh = 2,4 # EAS Alert Broadcast Channels
ignoreEASenable = True # Ignore any headline that includes followig word list
ignoreEASwords = test,advisory
USGS River flow data and Volcano alerts
Using the USGS water data page locate a water flow device, for example Columbia River at Vancouver, WA - USGS-14144700
Volcano Alerts use lat/long to determine ~1000km radius
[location]
# USGS Hydrology unique identifiers, LID or USGS ID https://waterdata.usgs.gov
riverList = 14144700 # example Mouth of Columbia River
# USGS Volcano alerts Enable USGS Volcano Alert Broadcast
volcanoAlertBroadcastEnabled = False
volcanoAlertBroadcastCh = 2
Repeater Settings
A repeater function for two different nodes and cross-posting messages. The repeater_channels is a list of repeater channels that will be consumed and rebroadcast on the same number channel on the other device, node, or interface. Each node should have matching channel numbers. The channel names and PSK do not need to be the same on the nodes. Use this feature responsibly to avoid creating a feedback loop.
[repeater] # repeater module
enabled = True
repeater_channels = [2, 3]
Ollama (LLM/AI) Settings
For Ollama to work, the command line ollama run 'model' needs to work properly. Ensure you have enough RAM and your GPU is working as expected. The default model for this project is set to gemma3:270m. Ollama can be remote Ollama Server works on a pi58GB with 40 second or less response time.
# Enable ollama LLM see more at https://ollama.com
ollama = True # Ollama model to use (defaults to gemma2:2b)
ollamaModel = gemma3:latest # Ollama model to use (defaults to gemma3:270m)
ollamaHostName = http://localhost:11434 # server instance to use (defaults to local machine install)
Also see llm.py for changing the defaults of:
# LLM System Variables
rawQuery = True # if True, the input is sent raw to the LLM if False, it is processed by the meshBotAI template
# Used in the meshBotAI template (legacy)
llmEnableHistory = True # enable history for the LLM model to use in responses adds to compute time
llmContext_fromGoogle = True # enable context from google search results helps with responses accuracy
googleSearchResults = 3 # number of google search results to include in the context more results = more compute time
Note for LLM in docker with NVIDIA. Needed for the container with ollama running.
Wikipedia Search Settings
The Wikipedia search module can use either the online Wikipedia API or a local Kiwix server for offline wiki access. Kiwix is especially useful for mesh networks operating in remote or offline environments.
# Enable or disable the wikipedia search module
wikipedia = True
# Use local Kiwix server instead of online Wikipedia
# Set to False to use online Wikipedia (default)
useKiwixServer = False
# Kiwix server URL (only used if useKiwixServer is True)
kiwixURL = http://127.0.0.1:8080
# Kiwix library name (e.g., wikipedia_en_100_nopic_2024-06)
# Find available libraries at https://library.kiwix.org/
kiwixLibraryName = wikipedia_en_100_nopic_2024-06
To set up a local Kiwix server:
- Install Kiwix tools: https://kiwix.org/en/
sudo apt install kiwix-tools -y - Download a Wikipedia ZIM file to
data/: https://library.kiwix.org/wget https://download.kiwix.org/zim/wikipedia/wikipedia_en_100_nopic_2025-09.zim - Run the server:
kiwix-serve --port 8080 wikipedia_en_100_nopic_2025-09.zim - Set
useKiwixServer = Truein your config.ini
The bot will automatically extract and truncate content to fit Meshtastic's message size limits (~500 characters).
Radio Monitoring
A module allowing a Hamlib compatible radio to connect to the bot. When functioning, it will message the configured channel with a message of in use. Requires hamlib/rigctld to be running as a service.
[radioMon]
enabled = True
rigControlServerAddress = localhost:4532
sigWatchBroadcastCh = 2 # channel to broadcast to can be 2,3
signalDetectionThreshold = -10 # minimum SNR as reported by radio via hamlib
signalHoldTime = 10 # hold time for high SNR
signalCooldown = 5 # the following are combined to reset the monitor
signalCycleLimit = 5
File Monitoring
Some dev notes for ideas of use
[fileMon]
filemon_enabled = True
file_path = alert.txt # text file to monitor for changes
broadcastCh = 2 # channel to send the message to can be 2,3 multiple channels comma separated
enable_read_news = False # news command will return the contents of a text file
news_file_path = news.txt
news_random_line = False # only return a single random line from the news file
enable_runShellCmd = False # enable the use of exernal shell commands, this enables more data in `sysinfo` DM
# if runShellCmd and you think it is safe to allow the x: command to run
# direct shell command handler the x: command in DMs user must be in bbs_admin_list
allowXcmd = True
Offline EAS
To Monitor EAS with no internet connection see the following notes
- samedec rust decoder much like multimon-ng
- sameold rust SAME message translator much like EAS2Text and dsame3
no examples yet for these tools
- EAS2Text
- depends on multimon-ng, direwolf, samedec rust decoder much like multimon-ng
- dsame3
- has a sample .ogg file for testing alerts
The following example shell command can pipe the data using etc/eas_alert_parser.py to alert.txt
sox -t ogg WXR-RWT.ogg -esigned-integer -b16 -r 22050 -t raw - | multimon-ng -a EAS -v 1 -t raw - | python eas_alert_parser.py
The following example shell command will pipe rtl_sdr to alert.txt
rtl_fm -f 162425000 -s 22050 | multimon-ng -t raw -a EAS /dev/stdin | python eas_alert_parser.py
Newspaper on mesh
Maintain multiple news sources. Each source should be a file named {source}_news.txt in the data/ directory (for example, data/mesh_news.txt).
- To read the default news, use the
readnewscommand (reads fromdata/news.txt. - To read a specific source, use
readnews abcto read fromdata/abc_news.txt.
This allows you to organize and access different news feeds or categories easily.
External scripts can update these files as needed, and the bot will serve the latest content on request.
Greet new nodes QRZ module
This isnt QRZ.com this is Q code for who is calling me, this will track new nodes and say hello
[qrz]
enabled = True # QRZ Hello to new nodes
qrz_hello_string = "send CMD or DM me for more info." # will be sent to all heard nodes once
training = True # Training mode will not send the hello message to new nodes, use this to build up database
Scheduler
In the config.ini enable the module
[scheduler]
enabled = False # enable or disable the scheduler module
interface = 1 # channel to send the message to
channel = 2
message = "MeshBot says Hello! DM for more info."
value = # value can be min,hour,day,mon,tue,wed,thu,fri,sat,sun.
# value can also be joke (everyXmin) or weather (hour) for special scheduled messages
# custom for module/scheduler.py custom schedule examples
interval = # interval to use when time is not set (e.g. every 2 days)
time = # time of day in 24:00 hour format when value is 'day' and interval is not set
The basic brodcast message can be setup in condig.ini. For advanced, See the modules/scheduler.py to edit the schedule. See schedule documentation for more. Recomend to backup changes so they dont get lost.
#Send WX every Morning at 08:00 using handle_wxc function to channel 2 on device 1
schedule.every().day.at("08:00").do(lambda: send_message(handle_wxc(0, 1, 'wx'), 2, 0, 1))
#Send a Net Starting Now Message Every Wednesday at 19:00 using send_message function to channel 2 on device 1
schedule.every().wednesday.at("19:00").do(lambda: send_message("Net Starting Now", 2, 0, 1))
BBS Link
The scheduler also handles the BBS Link Broadcast message, this would be an example of a mesh-admin channel on 8 being used to pass BBS post traffic between two bots as the initiator, one direction pull. The message just needs to have bbslink
# Send bbslink looking for peers every other day at 10:00 using send_message function to channel 8 on device 1
schedule.every(2).days.at("10:00").do(lambda: send_message("bbslink MeshBot looking for peers", 8, 0, 1))
bbslink_enabled = True
bbslink_whitelist = # list of whitelisted nodes numbers ex: 2813308004,4258675309 empty list allows all
Firmware 2.6 DM Key, and 2.7 CLIENT_BASE Favorite Nodes
Firmware 2.6 introduced PKC, enabling secure private messaging by adding necessary keys to each node. To fully utilize this feature, you should add favorite nodes—such as BBS admins—to your node’s favorites list to ensure their keys are retained. A helper script is provided to simplify this process:
- Run the helper script from the main program directory:
python3 script/addFav.py - By default, this script adds nodes from
bbs_admin_listandbbslink_whitelist - If using a virtual environment, run:
launch.sh addfav
To configure favorite nodes, add their numbers to your config file:
[general]
favoriteNodeList = # list of favorite nodes numbers ex: 2813308004,4258675309 used by script/addFav.py
MQTT Notes
There is no direct support for MQTT in the code, however, reports from Discord are that using meshtasticd with no radio and attaching the bot to the software node, which is MQTT-linked, allows routing. Tested working fully Firmware:2.6.11 with mosquitto.
There also seems to be a quicker way to enable MQTT by having your bot node with the enabled serial module with echo enabled and MQTT uplink and downlink. These two
Recognition
I used ideas and snippets from other responder bots and want to call them out!
Inspiration and Code Snippets
Games Ported From
- Lemonade Stand
- Drug Wars
- BlackJack
- Video Poker Terminal Game
- Python Mastermind
- Golf
- ARRL Question Pool Data from https://github.com/russolsen/ham_radio_question_pool
Special Thanks
- xdep: For the reporting tools.
- Nestpebble: For new ideas and enhancements.
- mrpatrick1991: For Docker configurations.
- https://github.com/A-c0rN: Assistance with iPAWS and EAS
- Mike O'Connell/skrrt: For eas_alert_parser enhanced by sheer.cold
- PiDiBi: For looking at test functions and other suggestions like wxc, CPU use, and alerting ideas.
- WH6GXZ nurse dude: For bashing on installer, Volcano Alerts 🌋
- Josh: For more bashing on installer!
- dj505: trying it on windows!
- mikecarper: ideas, and testing. hamtest
- c.merphy360: high altitude alerts
- Iris: testing and finding 🐞
- FJRPiolt: testing bugs out!!
- Cisien, bitflip, Woof, propstg, snydermesh, trs2982, F0X, mesb1, and Hailo1999: For testing and feature ideas on Discord and GitHub.
- Meshtastic Discord Community: For tossing out ideas and testing code.
Tools
- Node Backup Management: Node Slurper
Requirements
Python 3.8? or later is needed (docker on 3.13). The following can be installed with pip install -r requirements.txt or using the install.sh script for venv and automation:
pip install meshtastic
pip install pubsub
Mesh-bot enhancements:
pip install pyephem
pip install requests
pip install geopy
pip install maidenhead
pip install beautifulsoup4
pip install dadjokes
pip install schedule
pip install wikipedia
For the Ollama LLM:
pip install googlesearch-python
To enable emoji in the Debian console, install the fonts:
sudo apt-get install fonts-noto-color-emoji
Meshtastic® is a registered trademark of Meshtastic LLC. Meshtastic software components are released under various licenses, see GitHub for details. No warranty is provided - use at your own risk.
