Introduction: Davis Van ISS Weather Station With Raspbe
In this project I'm going to document my journey trying to set up a Davis Vantage Vue weather station continuously updating data on the internet with the help of a Raspberry Pi 4. You may say: there are a lot of projects like that on the internet! And you may be wrong :)
In this project we're going to use ONLY the Integrated Sensor Suite (ISS) of the Vantage Vue (which is sold separately for about 100 euros less than the kit including the console) and a RTL-SDR type receiver to get the sensor data. This way we'll be able to save a lot of money as a normal setup would need to have at least the console (to receive the data) and the USB datalogger from Davis (to interface with any software).
We're going to use Weewx (https://www.weewx.com/) to manage the weather station data. It is an amazing piece of open-source software which lets you connect basically every weather station that you can find on the market. Notably, it lets you use an RTL-SDR dongle to intercept data transmitted from a Davis Station (but really you could also get any data transmitted by a "normal" weather station) on the 868 MHz band (I'm based in Europe but it will work in a similar way for people in the US).
Supplies
- NooElec NSDR Nano2+ RTL receiver - 35 EUR (but you can use whatever you have available)
- Davis Vantage Vue ISS - 269 EUR
- Raspberry Pi 4 4GB, Power Adapter, SD Card - I paid 80 EUR for a complete set also with case
- BME280 Pressure and temperature sensor (optional) - 15 EUR
- Lots of patience
Step 1: Install WeeWx
NOTE: this assumes that you have a working installation of Debian or Ubuntu on the Raspberry PI. If that's not the case you can just check one of the thousand guides that explain how to perform a first setup. I will mainly work for the terminal so you can perform the following operations either connected via SSH or directly on the raspberry terminal.
Installing WeeWx is fairly easy. I followed the instructions for Debian (https://www.weewx.com/docs/debian.htm) and used the installation method with apt. Take note of the installation folders because we'll need that later.
executable: /usr/bin/weewxd configuration file: /etc/weewx/weewx.conf skins and templates: /etc/weewx/skins sqlite databases: /var/lib/weewx/ generated web pages and images: /var/www/html/weewx/ documentation: /usr/share/doc/weewx/ examples: /usr/share/doc/weewx/examples/ utilities: /usr/bin/wee_*
You can use the "Simulator" station to start with while we set up the rtldavis executable, which will take most of our time.
Once finished the installation a daemon will already be running in the background: you can check that by doing
systemctl status weewx
In order to reach the webpage of the station you either need to configure a web server/reverse proxy (to access directly the web page to the same machine where weewx is running) or just set up a copy of the files into a FTP server (optional step). The HTML pages created by weewx are nothing but pure static content created on a user-defined interval so you can put them wherever you want.
I only had to install lighttpd to see the webpage
sudo apt install lighttpd
it will look for files into /var/www/html and, as weewx copies its files into /var/www/html/weewx, you'll be able to reach the webpage at raspberry-pi-local-ip/weewx.
You'll see the data update after the first 5 minutes. Don't worry, we'll change this ugly skin in a minute :)
Step 2: Install RTL Drivers
This step really depends on the system that you're using, but in theory you should only need to install some packages from apt and make some correction to the udev rules. The final goal is to have a working installation of RTL-SDR which can recognize your dongle.
NOTE: you don't have to use the Noeelec NANO: any RTL-SDR dongle with the RTL2832U chip will work. This happened to be the smallest and cheapest receiver that I could find in Germany.
Start by installing the drivers.
sudo apt install rtl-sdr
Now we have to set some rules.
echo 'blacklist dvb_usb_rtl28xxu' | sudo tee – append /etc/modprobe.d/blacklist-dvb_usb_rtl28xxu.conf
Then go into /etc/udev/rules.d and create a new file (with sudo) called rtl-sdr.rules, with the content from this file https://osmocom.org/projects/rtl-sdr/repository/rtl-sdr/revisions/master/entry/rtl-sdr.rules.
It should be everything; if you now execute rtl_test into the command line you should be able to see something like this
$ rtl_test Found 1 device(s): 0: Realtek, RTL2838UHIDIR, SN: 00000001 Using device 0: Generic RTL2832U OEM Found Rafael Micro R820T tuner Supported gain values (29): 0.0 0.9 1.4 2.7 3.7 7.7 8.7 12.5 14.4 15.7 16.6 19.7 20.7 22.9 25.4 28.0 29.7 32.8 33.8 36.4 37.2 38.6 40.2 42.1 43.4 43.9 44.5 48.0 49.6 [R82XX] PLL not locked! Sampling at 2048000 S/s. Info: This tool will continuously read from the device, and report if samples get lost. If you observe no further output, everything is fine. Reading samples in async mode...
If you have troubles make sure to follow the RLT-SDR blog installation here https://www.rtl-sdr.com/tag/install-guide/ (scroll to Install on Linux)
Step 3: Install Rtldavis
This is the step that took me the longest, as it is poorly documented. rtldavis is a program initially created by
Douglas Hall (https://github.com/bemasher/rtldavis) who unfortunately hasn't been able to maintain it. Luckily
Luc Heijst forked the repo (https://github.com/lheijst/rtldavis) and added some new corrections and the European frequencies, which we're going to need. That's the one that we're going to use, together with the driver for weewx (weewx-rtldavis), which is needed to translate the data read by rtldavis into a LOOP packet that weewx can use.
We'll start by installing rtldavis and testing it .
First of all install golang
sudo apt-get install golang git cmake
Then install librtlsdr-dev manually: this is a requirement, otherwise rtldavis will not work.
cd /home/pi git clone https://github.com/steve-m/librtlsdr.git cd librtlsdr mkdir build cd build cmake ../ -DINSTALL_UDEV_RULES=ON make sudo make install sudo ldconfig
Change some lines into your profile file
sudo vim ~/.profile export GOROOT=/usr/lib/go export GOPATH=$HOME export PATH=$PATH:$GOROOT/bin:$GOPATH/bin source ~/.profile
Now get the rtldavis package
cd /home/pi go get -v github.com/lheijst/rtldavis
and compile the source
cd $GOPATH/src/github.com/lheijst/rtldavis git submodule init git submodule update go install -v .
If everything goes well you should find a new folder in your home with the executable inside /home/pi/bin/rtldavis.
Make sure to install librtlsdr-dev as showed before; if you use the one packaged in your system you'll not see an error, but instead a message (allocating zero copy buffer) and the executable will not produce anything.
Step 4: Set Up Rtldavis
We're going to test the executable alone to see that everything's working fine. Most importantly we NEED to find the right frequencies. The go package has some EU frequencies hardcoded (868077250, 868197250, 868317250, 868437250, 868557250) but those will likely not work for you out of the box because of some discrepancies.
Go into /home/pi/bin/ and tun this command
./rtldavis -startfreq 868000000 -endfreq 868600000 -stepfreq 10000
This will scan the entire frequency spectrum where the station should transmit (you already turned the ISS sensors on, haven't you?). It will take a while but you should see some frequencies where the transmission is
10:10:31.123820 rtldavis.go VERSION=0.15 10:10:31.124130 tr=1 fc=0 ppm=0 gain=0 maxmissed=51 ex=0 receiveWindow=300 actChan=[0] maxChan=1 10:10:31.124179 undefined=false verbose=false disableAfc=false deviceString=0 10:10:31.124216 TEST: startFreq=868000000 endFreq=868600000 stepFreq=10000 10:10:31.124687 BitRate: 19200 10:10:31.124738 SymbolLength: 14 10:10:31.124773 SampleRate: 268800 10:10:31.124807 Preamble: 1100101110001001 10:10:31.124841 PreambleSymbols: 16 10:10:31.124875 PreambleLength: 224 10:10:31.124909 PacketSymbols: 80 10:10:31.124943 PacketLength: 1120 10:10:31.124977 BlockSize: 512 10:10:31.125010 BufferLength: 2048 Found Rafael Micro R820T tuner 10:10:31.525099 Hop: {ChannelIdx:0 ChannelFreq:868077250 FreqError:0 Transmitter:0} Exact sample rate is: 268800.001367 Hz 10:10:31.698617 GetTunerGain: 0 Db 10:10:31.698697 SetFreqCorrection 0 ppm Successful 10:10:31.702138 Init channels: wait max 17 seconds for a message of each transmitter 10:11:07.580455 TESTFREQ 1: Frequency 868000000: NOK 10:11:25.519440 TESTFREQ 2: Frequency 868010000: NOK 10:11:43.458467 TESTFREQ 3: Frequency 868020000: NOK 10:12:01.397429 TESTFREQ 4: Frequency 868030000: NOK 10:12:19.337436 TESTFREQ 5: Frequency 868040000: NOK 10:12:37.276475 TESTFREQ 6: Frequency 868050000: NOK 10:12:55.215505 TESTFREQ 7: Frequency 868060000: NOK 10:13:13.154774 TESTFREQ 8: Frequency 868070000: NOK 10:13:31.093651 TESTFREQ 9: Frequency 868080000: NOK 10:13:49.032647 TESTFREQ 10: Frequency 868090000: NOK 10:14:06.971697 TESTFREQ 11: Frequency 868100000: NOK 10:14:24.910614 TESTFREQ 12: Frequency 868110000: NOK 10:14:27.099174 TESTFREQ 13: Frequency 868120000 (freqCorr=0): OK, msg.data: 80014D272907E6A4 10:14:39.912796 TESTFREQ 14: Frequency 868130000 (freqCorr=1031): OK, msg.data: 90058B09F39525CB 10:14:57.851600 TESTFREQ 15: Frequency 868140000: NOK 10:15:15.790642 TESTFREQ 16: Frequency 868150000: NOK 10:15:33.729646 TESTFREQ 17: Frequency 868160000: NOK 10:15:51.668677 TESTFREQ 18: Frequency 868170000: NOK 10:16:09.607740 TESTFREQ 19: Frequency 868180000: NOK 10:16:27.546769 TESTFREQ 20: Frequency 868190000: NOK 10:16:45.486136 TESTFREQ 21: Frequency 868200000: NOK 10:17:03.425139 TESTFREQ 22: Frequency 868210000: NOK 10:17:21.364168 TESTFREQ 23: Frequency 868220000: NOK 10:17:39.303185 TESTFREQ 24: Frequency 868230000: NOK 10:17:46.975373 TESTFREQ 25: Frequency 868240000 (freqCorr=1170): OK, msg.data: E0045ACB030E7D35 10:17:59.788122 TESTFREQ 26: Frequency 868250000 (freqCorr=2130): OK, msg.data: 50075FFF730024FC 10:18:17.727103 TESTFREQ 27: Frequency 868260000: NOK 10:18:35.666147 TESTFREQ 28: Frequency 868270000: NOK 10:18:53.605043 TESTFREQ 29: Frequency 868280000: NOK 10:19:11.544189 TESTFREQ 30: Frequency 868290000: NOK 10:19:29.483189 TESTFREQ 31: Frequency 868300000: NOK 10:19:47.422126 TESTFREQ 32: Frequency 868310000: NOK 10:20:05.361277 TESTFREQ 33: Frequency 868320000: NOK 10:20:23.300354 TESTFREQ 34: Frequency 868330000: NOK 10:20:41.239652 TESTFREQ 35: Frequency 868340000: NOK 10:20:59.178850 TESTFREQ 36: Frequency 868350000: NOK 10:21:06.849820 TESTFREQ 37: Frequency 868360000 (freqCorr=2310): OK, msg.data: 80046326B9066F1D 10:21:06.851418 TESTFREQ 38: Frequency 868370000 (freqCorr=3070): OK, msg.data: 80046326B9066F1D 10:21:24.789923 TESTFREQ 39: Frequency 868380000: NOK 10:21:42.729097 TESTFREQ 40: Frequency 868390000: NOK 10:22:00.668286 TESTFREQ 41: Frequency 868400000: NOK 10:22:18.607461 TESTFREQ 42: Frequency 868410000: NOK 10:22:36.546700 TESTFREQ 43: Frequency 868420000: NOK 10:22:54.484625 TESTFREQ 44: Frequency 868430000: NOK 10:23:12.423892 TESTFREQ 45: Frequency 868440000: NOK 10:23:30.363085 TESTFREQ 46: Frequency 868450000: NOK 10:23:48.302206 TESTFREQ 47: Frequency 868460000: NOK 10:24:06.241397 TESTFREQ 48: Frequency 868470000: NOK 10:24:13.915688 TESTFREQ 49: Frequency 868480000 (freqCorr=3727): OK, msg.data: A00552752B01A3E9 10:24:26.727222 TESTFREQ 50: Frequency 868490000 (freqCorr=4422): OK, msg.data: E00455CB010B9F1C 10:24:44.666628 TESTFREQ 51: Frequency 868500000: NOK 10:25:02.605734 TESTFREQ 52: Frequency 868510000: NOK 10:25:20.544182 TESTFREQ 53: Frequency 868520000: NOK 10:25:38.483333 TESTFREQ 54: Frequency 868530000: NOK 10:25:56.422456 TESTFREQ 55: Frequency 868540000: NOK 10:26:14.361597 TESTFREQ 56: Frequency 868550000: NOK 10:26:32.300674 TESTFREQ 57: Frequency 868560000: NOK 10:26:50.240205 TESTFREQ 58: Frequency 868570000: NOK 10:27:08.179294 TESTFREQ 59: Frequency 868580000: NOK 10:27:26.118451 TESTFREQ 60: Frequency 868590000: NOK 10:27:33.790629 TESTFREQ 61: Frequency 868600000 (freqCorr=3955): OK, msg.data: 500565FF720677CA 10:27:33.790832 Test reached endfreq; test ended
Results will probably vary for you, in my case I immediately saw that there was a frequency shift with respect to the "official" EU frequency of about 50000 Hz, and this was the reason why running rtldavis directly didn't work.
Anyway, after finding your offset you can re-rung rtldavis in receiving mode
./rtldavis -fc 56000 -tf EU -tr 1 -ppm 1
and you should see messages being received. You won't see any data from the sensors yet, but you should see something being printed like this
10:31:27.364138 rtldavis.go VERSION=0.15 10:31:27.365227 tr=1 fc=56000 ppm=1 gain=0 maxmissed=51 ex=0 receiveWindow=300 actChan=[0] maxChan=1 10:31:27.365752 undefined=false verbose=false disableAfc=false deviceString=0 10:31:27.367250 BitRate: 19200 10:31:27.367777 SymbolLength: 14 10:31:27.368256 SampleRate: 268800 10:31:27.368757 Preamble: 1100101110001001 10:31:27.369228 PreambleSymbols: 16 10:31:27.369697 PreambleLength: 224 10:31:27.370163 PacketSymbols: 80 10:31:27.370701 PacketLength: 1120 10:31:27.371174 BlockSize: 512 10:31:27.371642 BufferLength: 2048 Found Rafael Micro R820T tuner 10:31:27.811079 Hop: {ChannelIdx:0 ChannelFreq:868077250 FreqError:0 Transmitter:0} Exact sample rate is: 268800.001367 Hz 10:31:27.988707 GetTunerGain: 0 Db 10:31:28.049017 SetFreqCorrection 1 ppm Successful 10:31:28.053112 Init channels: wait max 17 seconds for a message of each transmitter 10:31:32.105955 TRANSMITTER 0 SEEN 10:31:32.106099 Hop: {ChannelIdx:2 ChannelFreq:868317250 FreqError:0 Transmitter:0} 10:31:34.667998 300567FFC1807FAA 2 0 0 0 0 msg.ID=0 10:31:34.668201 Hop: {ChannelIdx:4 ChannelFreq:868557250 FreqError:0 Transmitter:0} 10:31:37.229970 E00367CB03079EA7 3 0 0 0 0 msg.ID=0 10:31:37.230392 Hop: {ChannelIdx:1 ChannelFreq:868197250 FreqError:0 Transmitter:0} 10:31:39.793944 50035FFF7101DBB9 4 0 0 0 0 msg.ID=0 10:31:39.794112 Hop: {ChannelIdx:3 ChannelFreq:868437250 FreqError:0 Transmitter:0} 10:31:42.355950 80035E2679071BD3 5 0 0 0 0 msg.ID=0 10:31:42.356254 Hop: {ChannelIdx:0 ChannelFreq:868077250 FreqError:274 Transmitter:0}
You're ready to go to the next step
Step 5: Install Weewx-rtldavis
Now we can install the rtldavis driver for weewx. Make sure to stop weewx before proceeding further
sudo systemctl stop weewx wget -O weewx-rtldavis-master.zip https://github.com/lheijst/weewx-rtldavis/archive/master.zip sudo wee_extension --install weewx-rtldavis-master.zip sudo wee_config --reconfigure --driver=user.rtldavis --no-prompt
We first need to do some modifications to the hardcoded folder of the binary:
sudo vim /usr/share/weewx/user/rtldavis.py
and modify all the occurrences of /home/pi/work/bin/rtldavis with /home/pi/bin/rtldavis.
Now try to run the driver
sudo PYTHONPATH=/usr/share/weewx python /usr/share/weewx/user/rtldavis.py
and you should see some output.
Now let's configure the driver in weewx.
sudo vim /etc/weewx/weewx.conf [Rtldavis] cmd = /home/pi/bin/rtldavis -fc 55500 -ppm 1 transceiver_frequency = EU iss_channel = 1 # The values below only apply for Vantage Pro or Pro2 anemometer_channel = 0 leaf_soil_channel = 0 temp_hum_1_channel = 0 temp_hum_2_channel = 0 # rain bucket type (0: 0.01 inch, 1: 0.2 mm) rain_bucket_type = 1 # Print debug messages # 0=no logging; 1=minimum logging; 2=normal logging; 3=detailed logging debug_parse = 0 debug_rain = 0 debug_rtld = 1 # rtldavis logging: 1=inf; 2=(1)+data+chan; 3=(2)+pkt # The pct_good per transmitter can be saved to the database # This has only effect with 2 transmitters or more save_pct_good_per_transmitter = True log_unknown_sensors = True log_unmapped_sensors = True # The driver to use: driver = user.rtldavis
Remember to change the station type to rtldavis in [Station] before rebooting weewx. You can also turn debug=1 and debug_rtld=3 to check that everything is working properly.
Step 6: Optional: Add BME280 Sensors for Pressure and Inside Temperature
The ISS does not have sensors for pressure, nor it can measure inside temperature and humidity (obviously), so to replicate the original Davis console we need to add these data manually. While Weewx can easily continue running without inside temperature and humidity, pressure is such an important weather variable that it would be a pity not having it into the DB. Luckily you can use a cheap BME280 sensor to enhance the database with pressure, inside temperature and humidity.
First of all we need to turn of the inside temperature and humidity captured by rtldavis
sudo vim /usr/share/weewx/user/rtldavis.py
look for DEFAULT_SENSOR_MAP and comment these lines
#'inTemp': 'temp_in', # temperature of optional BMP280 module #'inHumidity': 'humidity_in',
Then connect the BME280 sensor with the right pins (see attached image) and make sure that I2C is enabled in raspi-config (https://www.raspberrypi-spy.co.uk/2014/11/enabling-the-i2c-interface-on-the-raspberry-pi/).
Now you can install i2cdetect
sudo apt install i2c-tools
if everything went well you should be able to run the following command and see the output
i2cdetect -y 1 0 1 2 3 4 5 6 7 8 9 a b c d e f 00: -- -- -- -- -- -- -- -- 10: -- -- -- -- -- -- -- -- -- -- 1a -- -- -- -- -- 20: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 30: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 40: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 50: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 70: -- -- -- -- -- -- -- 77
77 is the address of our sensor, which we'll need later.
Now that the sensor is working we need to (you say it) install a weewx driver that can parse this data. Luckily there's already a driver ready to be downloaded: https://gitlab.com/mfraz74/bme280wx/ (note, I use a fork of the original project that reduces logging).
Download and install the driver with
sudo systemctl stop weewx wget https://gitlab.com/mfraz74/bme280wx/-/archive/master/bme280wx-master.zip sudo wee_extension --install bme280wx-master.zip
Then go into the configuration file /etc/weewx/weewx.conf and modify some lines
[Bme280wx] i2c_port = 1 i2c_address = 0x77 usUnits = US temperatureKeys = inTemp temperature_must_have = "" pressureKeys = pressure pressure_must_have = outTemp humidityKeys = inHumidity humidity_must_have = ""
you'll notice we use the same address found with i2cdetect: in case you have a different address you of course need to change it. You can leave the rest as it is. Just make sure that the Engine / Services part of the configuration file has data_services = user.bme280wx.Bme280wx (should be added automatically by the installation).
Restart weewx and you should see the internal temperature and humidity, together with the pressure, being updated.
Step 7: (Optional) Change Skin and Configuration
I used the skin shown in the picture which includes data and plots of different periods.
You can install it easily by following the instructions here https://neoground.com/docs/neowx-material/installation
Once you installed the skin make sure you change your weewx.conf to only produce HTML pages for the neowx skin:
[StdReport] ... [[SeasonsReport]] skin = Seasons enable = false [[SmartphoneReport]] skin = Smartphone enable = false [[MobileReport]] skin = Mobile enable = false [[StandardReport]] skin = Standard enable = false [[Neow]] skin = neowx-material enable = true
Restart again weewx and everything should work as expected.
You can modify settings of the skin here /etc/weewx/skins/neowx-material. Note that if you press on the "telemetry" link (lower left) you can see the frequency errors, which clearly depend on the temperature, so that you can check if the frequency offset that you chose at the beginning needs to be tuned. You can even check if the battery of the capacitor in the Sensor suite is low or not.
See the screenshot for the values that I have,
Step 8: (Optional) Where Is the Database/Modifying Values
Unfortunately weewx does not give you the possibility to modify the data directly in the web interface.
However, the data is stored into a simple database file that you can open, view and modify: it can be found in /var/lib/weewx/
here is a sample Python script to read the data
import sqlite3 import pandas as pd con = sqlite3.connect('/var/lib/weewx/weewx.sdb') df = pd.read_sql_query("SELECT * from archive", con, index_col='dateTime')#.dropna(axis=1) df['datetime'] = pd.to_datetime(df.index, unit='s', utc=True) df['outTemp'] = df['outTemp'].apply(lambda x: (x-32.) * (5./9.) ) df['inTemp'] = df['inTemp'].apply(lambda x: (x-32.) * (5./9.) ) df['appTemp'] = df['outTemp'].apply(lambda x: (x-32.) * (5./9.) ) df['heatindex'] = df['heatindex'].apply(lambda x: (x-32.) * (5./9.) ) df['humidex'] = df['humidex'].apply(lambda x: (x-32.) * (5./9.) ) df['barometer']=df['barometer'].apply(lambda x: x * 33.86389 ) df['pressure']=df['pressure'].apply(lambda x: x * 33.86389 ) df['altimeter']=df['altimeter'].apply(lambda x: x * 33.86389 ) df['dewpoint'] = df['dewpoint'].apply(lambda x: (x-32.) * (5./9.) ) df['inDewpoint'] = df['dewpoint'].apply(lambda x: (x-32.) * (5./9.) )
To modify the data see the article https://github.com/weewx/weewx/wiki/Cleaning-up-old-'bad'-data
Step 9: (Optional) Upload Data Via FTP to a Server
You can use the [[FTP]] skin to upload the data to your website so that you can access the data also on the go, not only in your local LAN.
Just set the settings accordingly
[[FTP]] skin = Ftp enable = true user = ... password = .... server = .... # The ftp server name, e.g, www.myserver.org path = .... # The destination directory, e.g., /weather secure_ftp = False port = 21 passive = 1 ftp_encoding = latin-1
Note that I used ftp_encoding = latin-1 because some servers only accept this encoding. if that's not the case for you just comment this line.
Step 10: (Optional) Backup the Database
Coming soon...