For some reason, your WMR200 emitted a clock packet with some bad numbers
in it. Try this version of wmr100.py. It catches and logs the error, rather
than crashing. Let me know how it works.

-tk


On Tue, Jun 22, 2021 at 3:19 AM markusk <[email protected]> wrote:

>
> weewx with WMRS200 running fine since january - anyone got an idea why i
> got this tonight and/or how to avoid it?:
>
>
> Jun 22 00:35:16 raspberrypi weewx[21495] INFO weewx.manager: Added record
> 2021-06-22 00:35:00 CEST (1624314900) to database 'weewx.sdb'
> Jun 22 00:35:16 raspberrypi weewx[21495] INFO weewx.manager: Added record
> 2021-06-22 00:35:00 CEST (1624314900) to daily summary in 'weewx.sdb'
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:15:19: New
> client connected from 172.18.0.1 as weewx_698a3f6e (p2, c1, k60).
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:15:19: Client
> weewx_698a3f6e disconnected.
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:20:17: New
> client connected from 172.18.0.1 as weewx_4b5f85fb (p2, c1, k60).
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:20:17: Client
> weewx_4b5f85fb disconnected.
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:25:18: New
> client connected from 172.18.0.1 as weewx_e92e6132 (p2, c1, k60).
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:25:18: Client
> weewx_e92e6132 disconnected.
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:30:20: New
> client connected from 172.18.0.1 as weewx_8d9f7aa4 (p2, c1, k60).
> Jun 22 00:35:16 raspberrypi 34110015c165[628]: 2021-06-21T22:30:20: Client
> weewx_8d9f7aa4 disconnected.
> Jun 22 00:35:16 raspberrypi weewx[21495] INFO weewx.restx: MQTT: Published
> record 2021-06-22 00:35:00 CEST (1624314900)
> Jun 22 00:35:16 raspberrypi weewx[21495] INFO weewx.restx: OWM: Published
> record 2021-06-22 00:35:00 CEST (1624314900)
> Jun 22 00:35:16 raspberrypi weewx[21495] INFO weewx.restx: PWSWeather:
> Published record 2021-06-22 00:35:00 CEST (1624314900)
> Jun 22 00:35:17 raspberrypi weewx[21495] INFO weewx.cheetahgenerator:
> Generated 8 files for report SeasonsReport in 1.60 seconds
> Jun 22 00:35:18 raspberrypi weewx[21495] INFO weewx.imagegenerator:
> Generated 16 images for report SeasonsReport in 0.69 seconds
> Jun 22 00:35:18 raspberrypi weewx[21495] INFO weewx.reportengine: Copied 0
> files to /var/www/html/weewx
> Jun 22 00:35:18 raspberrypi weewx[21495] INFO weewx.cheetahgenerator:
> Generated 1 files for report MobileReport in 0.03 seconds
> Jun 22 00:35:18 raspberrypi weewx[21495] INFO weewx.imagegenerator:
> Generated 4 images for report MobileReport in 0.13 seconds
> Jun 22 00:35:18 raspberrypi weewx[21495] INFO weewx.reportengine: Copied 0
> files to /var/www/html/weewx/mobile
> Jun 22 00:35:20 raspberrypi weewx[21495] INFO weewx.cheetahgenerator:
> Generated 11 files for report Belchertown in 1.88 seconds
> Jun 22 00:35:20 raspberrypi weewx[21495] INFO weewx.reportengine: Copied 2
> files to /var/www/html/weewx/belchertown
> Jun 22 00:36:07 raspberrypi kernel: [9277996.404380] brcmfmac:
> brcmf_cfg80211_set_power_mgmt: power save enabled
> Jun 22 00:40:24 raspberrypi weewx[21495] INFO weewx.manager: Added record
> 2021-06-22 00:40:00 CEST (1624315200) to database 'weewx.sdb'
> Jun 22 00:40:24 raspberrypi weewx[21495] INFO weewx.manager: Added record
> 2021-06-22 00:40:00 CEST (1624315200) to daily summary in 'weewx.sdb'
> Jun 22 00:40:24 raspberrypi weewx[21495] INFO weewx.restx: MQTT: Published
> record 2021-06-22 00:40:00 CEST (1624315200)
> Jun 22 00:40:24 raspberrypi weewx[21495] INFO weewx.restx: OWM: Published
> record 2021-06-22 00:40:00 CEST (1624315200)
> Jun 22 00:40:24 raspberrypi weewx[21495] INFO weewx.restx: PWSWeather:
> Published record 2021-06-22 00:40:00 CEST (1624315200)
> Jun 22 00:40:25 raspberrypi weewx[21495] INFO weewx.cheetahgenerator:
> Generated 8 files for report SeasonsReport in 1.54 seconds
> Jun 22 00:40:26 raspberrypi weewx[21495] INFO weewx.imagegenerator:
> Generated 16 images for report SeasonsReport in 0.72 seconds
> Jun 22 00:40:26 raspberrypi weewx[21495] INFO weewx.reportengine: Copied 0
> files to /var/www/html/weewx
> Jun 22 00:40:26 raspberrypi weewx[21495] INFO weewx.cheetahgenerator:
> Generated 1 files for report MobileReport in 0.03 seconds
> Jun 22 00:40:26 raspberrypi weewx[21495] INFO weewx.imagegenerator:
> Generated 4 images for report MobileReport in 0.13 seconds
> Jun 22 00:40:26 raspberrypi weewx[21495] INFO weewx.reportengine: Copied 0
> files to /var/www/html/weewx/mobile
> Jun 22 00:40:28 raspberrypi weewx[21495] INFO weewx.cheetahgenerator:
> Generated 11 files for report Belchertown in 1.63 seconds
> Jun 22 00:40:28 raspberrypi weewx[21495] INFO weewx.reportengine: Copied 2
> files to /var/www/html/weewx/belchertown
> Jun 22 00:41:07 raspberrypi weewx[21495] INFO weewx.engine: Main loop
> exiting. Shutting engine down.
> Jun 22 00:41:07 raspberrypi weewx[21495] INFO weewx.engine: Shutting down
> StdReport thread
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__: Caught
> unrecoverable exception:
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> mktime argument out of range
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> Traceback (most recent call last):
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> File "/usr/share/weewx/weewxd", line 157, in main
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> engine.run()
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> File "/usr/share/weewx/weewx/engine.py", line 208, in run
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> for packet in self.console.genLoopPackets():
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> File "/usr/share/weewx/weewx/drivers/wmr100.py", line 200, in genLoopPackets
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> _raw = WMR100._dispatch_dict[_packet_type](self, _packet)
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> File "/usr/share/weewx/weewx/drivers/wmr100.py", line 407, in _clock_packet
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> self.last_time = time.mktime(tt)
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> OverflowError: mktime argument out of range
> Jun 22 00:41:07 raspberrypi weewx[21495] CRITICAL __main__:     ****
> Exiting.
>
> --
> You received this message because you are subscribed to the Google Groups
> "weewx-user" group.
> To unsubscribe from this group and stop receiving emails from it, send an
> email to [email protected].
> To view this discussion on the web visit
> https://groups.google.com/d/msgid/weewx-user/f0b0c511-4e20-484c-85f0-e495de5aa2ben%40googlegroups.com
> <https://groups.google.com/d/msgid/weewx-user/f0b0c511-4e20-484c-85f0-e495de5aa2ben%40googlegroups.com?utm_medium=email&utm_source=footer>
> .
>

-- 
You received this message because you are subscribed to the Google Groups 
"weewx-user" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to [email protected].
To view this discussion on the web visit 
https://groups.google.com/d/msgid/weewx-user/CAPq0zECkuJU00T%2B%2BHGrurqbYAV9EtDPuMUFHsqeNZF8wRs6WHA%40mail.gmail.com.
#
#    Copyright (c) 2009-2015 Tom Keffer <[email protected]>
#
#    See the file LICENSE.txt for your full rights.
#
"""Classees and functions for interfacing with an Oregon Scientific WMR100
station.  The WMRS200 reportedly works with this driver (NOT the WMR200, which
is a different beast).

The wind sensor reports wind speed, wind direction, and wind gust.  It does
not report wind gust direction.

WMR89:
 - data logger
 - up to 3 channels
 - protocol 3 sensors
 - THGN800, PRCR800, WTG800

WMR86:
 - no data logger
 - protocol 3 sensors
 - THGR800, WGR800, PCR800, UVN800

The following references were useful for figuring out the WMR protocol:
    
From Per Ejeklint:
  https://github.com/ejeklint/WLoggerDaemon/blob/master/Station_protocol.md

From Rainer Finkeldeh:
  http://www.bashewa.com/wmr200-protocol.php

The WMR driver for the wfrog weather system:
  http://code.google.com/p/wfrog/source/browse/trunk/wfdriver/station/wmrs200.py

Unfortunately, there is no documentation for PyUSB v0.4, so you have to back
it out of the source code, available at:
  https://pyusb.svn.sourceforge.net/svnroot/pyusb/branches/0.4/pyusb.c  
"""

from __future__ import absolute_import
from __future__ import print_function
import logging
import time
import operator
from functools import reduce

import usb

import weewx.drivers
import weewx.wxformulas
import weeutil.weeutil

log = logging.getLogger(__name__)

DRIVER_NAME = 'WMR100'
DRIVER_VERSION = "3.5.0"

def loader(config_dict, engine):  # @UnusedVariable
    return WMR100(**config_dict[DRIVER_NAME])    

def confeditor_loader():
    return WMR100ConfEditor()


class WMR100(weewx.drivers.AbstractDevice):
    """Driver for the WMR100 station."""

    DEFAULT_MAP = {
        'pressure': 'pressure',
        'windSpeed': 'wind_speed',
        'windDir': 'wind_dir',
        'windGust': 'wind_gust',
        'windBatteryStatus': 'battery_status_wind',
        'inTemp': 'temperature_0',
        'outTemp': 'temperature_1',
        'extraTemp1': 'temperature_2',
        'extraTemp2': 'temperature_3',
        'extraTemp3': 'temperature_4',
        'extraTemp4': 'temperature_5',
        'extraTemp5': 'temperature_6',
        'extraTemp6': 'temperature_7',
        'extraTemp7': 'temperature_8',
        'inHumidity': 'humidity_0',
        'outHumidity': 'humidity_1',
        'extraHumid1': 'humidity_2',
        'extraHumid2': 'humidity_3',
        'extraHumid3': 'humidity_4',
        'extraHumid4': 'humidity_5',
        'extraHumid5': 'humidity_6',
        'extraHumid6': 'humidity_7',
        'extraHumid7': 'humidity_8',
        'inTempBatteryStatus': 'battery_status_0',
        'outTempBatteryStatus': 'battery_status_1',
        'extraBatteryStatus1': 'battery_status_2',
        'extraBatteryStatus2': 'battery_status_3',
        'extraBatteryStatus3': 'battery_status_4',
        'extraBatteryStatus4': 'battery_status_5',
        'extraBatteryStatus5': 'battery_status_6',
        'extraBatteryStatus6': 'battery_status_7',
        'extraBatteryStatus7': 'battery_status_8',
        'rain': 'rain',
        'rainTotal': 'rain_total',
        'rainRate': 'rain_rate',
        'hourRain': 'rain_hour',
        'rain24': 'rain_24',
        'rainBatteryStatus': 'battery_status_rain',
        'UV': 'uv',
        'uvBatteryStatus': 'battery_status_uv'}
    
    def __init__(self, **stn_dict):
        """Initialize an object of type WMR100.
        
        NAMED ARGUMENTS:
        
        model: Which station model is this?
        [Optional. Default is 'WMR100']

        timeout: How long to wait, in seconds, before giving up on a response
        from the USB port.
        [Optional. Default is 15 seconds]
        
        wait_before_retry: How long to wait before retrying.
        [Optional. Default is 5 seconds]

        max_tries: How many times to try before giving up.
        [Optional. Default is 3]
        
        vendor_id: The USB vendor ID for the WMR
        [Optional. Default is 0xfde]
        
        product_id: The USB product ID for the WM
        [Optional. Default is 0xca01]
        
        interface: The USB interface
        [Optional. Default is 0]
        
        IN_endpoint: The IN USB endpoint used by the WMR.
        [Optional. Default is usb.ENDPOINT_IN + 1]
        """

        log.info('Driver version is %s' % DRIVER_VERSION)
        self.model = stn_dict.get('model', 'WMR100')
        # TODO: Consider putting these in the driver loader instead:
        self.record_generation = stn_dict.get('record_generation', 'software')
        self.timeout = float(stn_dict.get('timeout', 15.0))
        self.wait_before_retry = float(stn_dict.get('wait_before_retry', 5.0))
        self.max_tries = int(stn_dict.get('max_tries', 3))
        self.vendor_id = int(stn_dict.get('vendor_id', '0x0fde'), 0)
        self.product_id = int(stn_dict.get('product_id', '0xca01'), 0)
        self.interface = int(stn_dict.get('interface', 0))
        self.IN_endpoint = int(stn_dict.get('IN_endpoint', usb.ENDPOINT_IN + 1))
        self.sensor_map = dict(self.DEFAULT_MAP)
        if 'sensor_map' in stn_dict:
            self.sensor_map.update(stn_dict['sensor_map'])
        log.info('Sensor map is %s' % self.sensor_map)
        self.last_rain_total = None
        self.devh = None
        self.openPort()

    def openPort(self):
        dev = self._findDevice()
        if not dev:
            log.error("Unable to find USB device (0x%04x, 0x%04x)"
                      % (self.vendor_id, self.product_id))
            raise weewx.WeeWxIOError("Unable to find USB device")
        self.devh = dev.open()
        # Detach any old claimed interfaces
        try:
            self.devh.detachKernelDriver(self.interface)
        except usb.USBError:
            pass
        try:
            self.devh.claimInterface(self.interface)
        except usb.USBError as e:
            self.closePort()
            log.error("Unable to claim USB interface: %s" % e)
            raise weewx.WeeWxIOError(e)
        
    def closePort(self):
        try:
            self.devh.releaseInterface()
        except usb.USBError:
            pass
        try:
            self.devh.detachKernelDriver(self.interface)
        except usb.USBError:
            pass
        
    def genLoopPackets(self):
        """Generator function that continuously returns loop packets"""
        
        # Get a stream of raw packets, then convert them, depending on the
        # observation type.
        
        for _packet in self.genPackets():
            try:
                _packet_type = _packet[1]
                if _packet_type in WMR100._dispatch_dict:
                    # get the observations from the packet
                    _raw = WMR100._dispatch_dict[_packet_type](self, _packet)
                    if _raw is not None:
                        # map the packet labels to schema fields
                        _record = dict()
                        for k in self.sensor_map:
                            if self.sensor_map[k] in _raw:
                                _record[k] = _raw[self.sensor_map[k]]
                        # if there are any observations, add time and units
                        if _record:
                            for k in ['dateTime', 'usUnits']:
                                _record[k] = _raw[k]
                            yield _record
            except IndexError:
                log.error("Malformed packet: %s" % _packet)
                
    def genPackets(self):
        """Generate measurement packets. These are 8 to 17 byte long packets containing
        the raw measurement data.
        
        For a pretty good summary of what's in these packets see
        https://github.com/ejeklint/WLoggerDaemon/blob/master/Station_protocol.md
        """
        
        # Wrap the byte generator function in GenWithPeek so we 
        # can peek at the next byte in the stream. The result, the variable
        # genBytes, will be a generator function.
        genBytes = weeutil.weeutil.GenWithPeek(self._genBytes_raw())

        # Start by throwing away any partial packets:
        for ibyte in genBytes:
            if genBytes.peek() != 0xff:
                break
        
        buff = []
        # March through the bytes generated by the generator function genBytes:
        for ibyte in genBytes:
            # If both this byte and the next one are 0xff, then we are at the end of a record
            if ibyte == 0xff and genBytes.peek() == 0xff:
                # We are at the end of a packet.
                # Compute its checksum. This can throw an exception if the packet is empty.
                try:
                    computed_checksum = reduce(operator.iadd, buff[:-2])
                except TypeError as e:
                    log.debug("Exception while calculating checksum: %s" % e)
                else:
                    actual_checksum = (buff[-1] << 8) + buff[-2]
                    if computed_checksum == actual_checksum:
                        # Looks good. Yield the packet
                        yield buff
                    else:
                        log.debug("Bad checksum on buffer of length %d" % len(buff))
                # Throw away the next character (which will be 0xff):
                next(genBytes)
                # Start with a fresh buffer
                buff = []
            else:
                buff.append(ibyte)
             
    @property
    def hardware_name(self):
        return self.model
        
    #===============================================================================
    #                         USB functions
    #===============================================================================

    def _findDevice(self):
        """Find the given vendor and product IDs on the USB bus"""
        for bus in usb.busses():
            for dev in bus.devices:
                if dev.idVendor == self.vendor_id and dev.idProduct == self.product_id:
                    return dev

    def _genBytes_raw(self):
        """Generates a sequence of bytes from the WMR USB reports."""
        
        try:
            # Only need to be sent after a reset or power failure of the station:
            self.devh.controlMsg(usb.TYPE_CLASS + usb.RECIP_INTERFACE,       # requestType
                                 0x0000009,                                  # request
                                 [0x20,0x00,0x08,0x01,0x00,0x00,0x00,0x00],  # buffer
                                 0x0000200,                                  # value
                                 0x0000000,                                  # index
                                 1000)                                       # timeout
        except usb.USBError as e:
            log.error("Unable to send USB control message: %s" % e)
            # Convert to a Weewx error:
            raise weewx.WakeupError(e)
            
        nerrors = 0
        while True:
            try:
                # Continually loop, retrieving "USB reports". They are 8 bytes long each.
                report = self.devh.interruptRead(self.IN_endpoint,
                                                 8, # bytes to read
                                                 int(self.timeout * 1000))
                # While the report is 8 bytes long, only a smaller, variable portion of it
                # has measurement data. This amount is given by byte zero. Return each
                # byte, starting with byte one:
                for i in range(1, report[0] + 1):
                    yield report[i]
                nerrors = 0
            except (IndexError, usb.USBError) as e:
                log.debug("Bad USB report received: %s" % e)
                nerrors += 1
                if nerrors > self.max_tries:
                    log.error("Max retries exceeded while fetching USB reports")
                    raise weewx.RetriesExceeded("Max retries exceeded while fetching USB reports")
                time.sleep(self.wait_before_retry)
    
    # =========================================================================
    # LOOP packet decoding functions
    #==========================================================================

    def _rain_packet(self, packet):
        
        # NB: in my experiments with the WMR100, it registers in increments of
        # 0.04 inches. Per Ejeklint's notes have you divide the packet values
        # by 10, but this would result in an 0.4 inch bucket --- too big. So,
        # I'm dividing by 100.
        _record = {
            'rain_rate'          : ((packet[3] << 8) + packet[2]) / 100.0,
            'rain_hour'          : ((packet[5] << 8) + packet[4]) / 100.0,
            'rain_24'            : ((packet[7] << 8) + packet[6]) / 100.0,
            'rain_total'         : ((packet[9] << 8) + packet[8]) / 100.0,
            'battery_status_rain': packet[0] >> 4,
            'dateTime': int(time.time() + 0.5),
            'usUnits': weewx.US}

        # Because the WMR does not offer anything like bucket tips, we must
        # calculate it by looking for the change in total rain. Of course, this
        # won't work for the very first rain packet.
        _record['rain'] = weewx.wxformulas.calculate_rain(
            _record['rain_total'], self.last_rain_total)
        self.last_rain_total = _record['rain_total']
        return _record

    def _temperature_packet(self, packet):
        _record = {'dateTime': int(time.time() + 0.5),
                   'usUnits': weewx.METRIC}
        # Per Ejeklint's notes don't mention what to do if temperature is
        # negative. I think the following is correct. Also, from experience, we
        # know that the WMR has problems measuring dewpoint at temperatures
        # below about 20F. So ignore dewpoint and let weewx calculate it.
        T = (((packet[4] & 0x7f) << 8) + packet[3]) / 10.0
        if packet[4] & 0x80:
            T = -T
        R = float(packet[5])
        channel = packet[2] & 0x0f
        _record['temperature_%d' % channel] = T
        _record['humidity_%d' % channel] = R
        _record['battery_status_%d' % channel] = (packet[0] & 0x40) >> 6
        return _record
        
    def _temperatureonly_packet(self, packet):
        # function added by fstuyk to manage temperature-only sensor THWR800
        _record = {'dateTime': int(time.time() + 0.5),
                   'usUnits': weewx.METRIC}
        # Per Ejeklint's notes don't mention what to do if temperature is
        # negative. I think the following is correct. 
        T = (((packet[4] & 0x7f) << 8) + packet[3])/10.0
        if packet[4] & 0x80:
            T = -T
        channel = packet[2] & 0x0f
        _record['temperature_%d' % channel] = T
        _record['battery_status_%d' % channel] = (packet[0] & 0x40) >> 6
        return _record

    def _pressure_packet(self, packet):
        # Although the WMR100 emits SLP, not all consoles in the series
        # (notably, the WMRS200) allow the user to set altitude.  So we
        # record only the station pressure (raw gauge pressure).
        SP = float(((packet[3] & 0x0f) << 8) + packet[2])
        _record = {'pressure': SP,
                   'dateTime': int(time.time() + 0.5),
                   'usUnits': weewx.METRIC}
        return _record
        
    def _uv_packet(self, packet):
        _record = {'uv': float(packet[3]),
                   'battery_status_uv': packet[0] >> 4,
                   'dateTime': int(time.time() + 0.5),
                   'usUnits': weewx.METRIC}
        return _record

    def _wind_packet(self, packet):
        """Decode a wind packet. Wind speed will be in kph"""

        _record = {
            'wind_speed': ((packet[6] << 4) + ((packet[5]) >> 4)) / 10.0,
            'wind_gust': (((packet[5] & 0x0f) << 8) + packet[4]) / 10.0,
            'wind_dir': (packet[2] & 0x0f) * 360.0 / 16.0,
            'battery_status_wind': (packet[0] >> 4),
            'dateTime': int(time.time() + 0.5),
            'usUnits': weewx.METRICWX}

        # Sometimes the station emits a wind gust that is less than the
        # average wind. If this happens, ignore it.
        if _record['wind_gust'] < _record['wind_speed']:
            _record['wind_gust'] = None

        return _record
    
    def _clock_packet(self, packet):
        """The clock packet is not used by weewx.  However, the last time is
        saved in case getTime() is called."""
        tt = (2000 + packet[8], packet[7], packet[6], packet[5], packet[4], 0, 0, 0, -1)
        try:
            self.last_time = time.mktime(tt)
        except OverflowError:
            log.error("Bad clock packet: %s", packet)
            log.error("**** ignored.")
        return None
    
    # Dictionary that maps a measurement code, to a function that can decode it
    _dispatch_dict = {0x41: _rain_packet,
                      0x42: _temperature_packet,
                      0x46: _pressure_packet,
                      0x47: _uv_packet,
                      0x48: _wind_packet,
                      0x60: _clock_packet,
                      0x44: _temperatureonly_packet}


class WMR100ConfEditor(weewx.drivers.AbstractConfEditor):
    @property
    def default_stanza(self):
        return """
[WMR100]
    # This section is for the Oregon Scientific WMR100

    # The driver to use
    driver = weewx.drivers.wmr100

    # The station model, e.g., WMR100, WMR100N, WMRS200
    model = WMR100
"""

    def modify_config(self, config_dict):
        print("""
Setting rainRate calculation to hardware.""")
        config_dict.setdefault('StdWXCalculate', {})
        config_dict['StdWXCalculate'].setdefault('Calculations', {})
        config_dict['StdWXCalculate']['Calculations']['rainRate'] = 'hardware'

Reply via email to