Customizing the WeeWX service engine¶
This is an advanced topic intended for those who wish to try their hand at extending the internal engine in WeeWX. Before attempting these examples, you should be reasonably proficient with Python.
Warning
Please note that the API to the service engine may change in future versions!
At a high level, WeeWX consists of an engine that is responsible for managing a set of services. A service consists of a Python class which binds its member functions to various events. The engine arranges to have the bound member function called when a specific event happens, such as a new LOOP packet arriving.
The services are specified in lists in the
[Engine][[Services]]
stanza of the configuration file. The [[Services]]
section
lists all the services to be run, broken up into different service
lists.
These lists are designed to orchestrate the data as it flows through the WeeWX engine. For example,
you want to make sure that data has been processed by the quality control service, StdQC
, before
putting them in the database. Similarly, the reporting system must come after the data has been
put in the database. These groups ensure that things happen in the proper sequence.
See the table The standard WeeWX services for a list of the services that are normally run.
Modifying an existing service¶
The service weewx.engine.StdPrint
prints out new LOOP and
archive packets to the console when they arrive. By default, it prints
out the entire record, which generally includes a lot of possibly
distracting information and can be rather messy. Suppose you do not like
this, and want it to print out only the time, barometer reading, and the
outside temperature whenever a new LOOP packet arrives.
This could be done by subclassing the default print service StdPrint
and overriding member
function new_loop_packet()
.
Create the file user/myprint.py
:
from weewx.engine import StdPrint
from weeutil.weeutil import timestamp_to_string
class MyPrint(StdPrint):
# Override the default new_loop_packet member function:
def new_loop_packet(self, event):
packet = event.packet
print("LOOP: ", timestamp_to_string(packet['dateTime']),
"BAR=", packet.get('barometer', 'N/A'),
"TEMP=", packet.get('outTemp', 'N/A'))
This service substitutes a new implementation for the member function
new_loop_packet
. This implementation prints out the time, then
the barometer reading (or N/A
if it is not available) and the
outside temperature (or N/A
).
You then need to specify that your print service class should be loaded
instead of the default StdPrint
service. This is done by
substituting your service name for StdPrint
in
service_list
, located in [Engine]/[[Services]]
:
[Engine]
[[Services]]
...
report_services = user.myprint.MyPrint, weewx.engine.StdReport
Note that the report_services
must be all on one line.
Unfortunately, the parser ConfigObj
does not allow options to
be continued on to following lines.
Creating a new service¶
Suppose there is no service that can be easily customized for your
needs. In this case, a new one can easily be created by subclassing off
the abstract base class StdService
, and then adding the
functionality you need. Here is an example that implements an alarm,
which sends off an email when an arbitrary expression evaluates
True
.
This example is included in the standard distribution as
examples/alarm.py:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 |
|
This service expects all the information it needs to be in the
configuration file weewx.conf
in a new section called
[Alarm]
. So, add the following lines to your configuration
file:
[Alarm]
expression = "outTemp < 40.0"
time_wait = 3600
smtp_host = smtp.example.com
smtp_user = myusername
smtp_password = mypassword
mailto = auser@example.com, anotheruser@example.com
from = me@example.com
subject = "Alarm message from WeeWX!"
There are three important points to be noted in this example, each
marked with a NOTE
flag in the code.
- (Line 40) Here is where the binding happens between an event,
weewx.NEW_ARCHIVE_RECORD
in this example, and a member function,self.new_archive_record
. When the eventNEW_ARCHIVE_RECORD
occurs, the functionself.new_archive_record
will be called. There are many other events that can be intercepted. Look in the fileweewx/_init_.py
. - (Line 56) Some hardware do not emit all possible observation types in every
record, so it's possible that a record may be missing some types
that are used in the expression. This try block will catch the
NameError
exception that would be raised should this occur. - (Line 59) This is where the test is done for whether to sound the
alarm. The
[Alarm]
configuration options specify that the alarm be sounded whenoutTemp < 40.0
evaluatesTrue
, that is when the outside temperature is below 40.0 degrees. Any valid Python expression can be used, although the only variables available are those in the current archive record.
Another example expression could be:
expression = "outTemp < 32.0 and windSpeed > 10.0"
In this case, the alarm is sounded if the outside temperature drops below freezing and the wind speed is greater than 10.0.
Note that units must be the same as whatever is being used in your
database, that is, the same as what you specified in option
target_unit
.
Option time_wait
is used to avoid a flood of nearly identical
emails. The new service will wait this long before sending another email
out.
Email will be sent through the SMTP host specified by option
smtp_host
. The recipient(s) are specified by the comma
separated option mailto
.
Many SMTP hosts require user login. If this is the case, the user and
password are specified with options smtp_user
and
smtp_password
, respectively.
The last two options, from
and subject
are optional.
If not supplied, WeeWX will supply something sensible. Note, however,
that some mailers require a valid "from" email address and the one
WeeWX supplies may not satisfy its requirements.
To make this all work, you must first copy the alarm.py
file to
the user
directory. Then tell the engine to load this new
service by adding the service name to the list report_services
,
located in [Engine]/[[Services]]
:
[Engine]
[[Services]]
report_services = weewx.engine.StdPrint, weewx.engine.StdReport, user.alarm.MyAlarm
Again, note that the option report_services
must be all on one
line — the ConfigObj
parser does not allow options to be
continued on to following lines.
In addition to this example, the distribution also includes a
low-battery alarm (lowBattery.py
), which is similar, except
that it intercepts LOOP events instead of archiving events.
Adding a second data source¶
A very common problem is wanting to augment the data from your weather station with data from some other device. Generally, you have two approaches for how to handle this:
- Run two instances of WeeWX, each using its own database and
weewx.conf
configuration file. The results are then combined in a final report, using WeeWX's ability to use more than one database. See the Wiki entry How to run multiple instances of WeeWX for details on how to do this. - Run one instance, but use a custom WeeWX service to augment the records coming from your weather station with data from the other device.
This section covers the latter approach.
Suppose you have installed an electric meter at your house and you wish to correlate electrical usage with the weather. The meter has some sort of connection to your computer, allowing you to download the total power consumed. At the end of every archive interval you want to calculate the amount of power consumed during the interval, then add the results to the record coming off your weather station. How would you do this?
Here is the outline of a service that retrieves the electrical
consumption data and adds it to the archive record. It assumes that you
already have a function download_total_power()
that, somehow,
downloads the amount of power consumed since time zero.
File user/electricity.py
import weewx
from weewx.engine import StdService
class AddElectricity(StdService):
def __init__(self, engine, config_dict):
# Initialize my superclass first:
super(AddElectricity, self).__init__(engine, config_dict)
# Bind to any new archive record events:
self.bind(weewx.NEW_ARCHIVE_RECORD, self.new_archive_record)
self.last_total = None
def new_archive_record(self, event):
total_power = download_total_power()
if self.last_total:
net_consumed = total_power - self.last_total
event.record['electricity'] = net_consumed
self.last_total = total_power
This adds a new key electricity
to the record dictionary and
sets it equal to the difference between the amount of power currently
consumed and the amount consumed at the last archive record. Hence, it
will be the amount of power consumed over the archive interval. The unit
should be Watt-hours.
As an aside, it is important that the function
download_total_power()
does not delay very long because it will
sit right in the main loop of the WeeWX engine. If it's going to cause
a delay of more than a couple seconds you might want to put it in a
separate thread and feed the results to AddElectricity
through
a queue.
To make sure your service gets run, you need to add it to one of the
service lists in weewx.conf
, section [Engine]
,
subsection [[Services]]
.
In our case, the obvious place for our new service is in
data_services
. When you're done, your section
[Engine]
will look something like this:
# This section configures the internal WeeWX engine.
[Engine]
[[Services]]
# This section specifies the services that should be run. They are
# grouped by type, and the order of services within each group
# determines the order in which the services will be run.
xtype_services = weewx.wxxtypes.StdWXXTypes, weewx.wxxtypes.StdPressureCooker, weewx.wxxtypes.StdRainRater, weewx.wxxtypes.StdDelta
prep_services = weewx.engine.StdTimeSynch
data_services = user.electricity.AddElectricity
process_services = weewx.engine.StdConvert, weewx.engine.StdCalibrate, weewx.engine.StdQC, weewx.wxservices.StdWXCalculate
archive_services = weewx.engine.StdArchive
restful_services = weewx.restx.StdStationRegistry, weewx.restx.StdWunderground, weewx.restx.StdPWSweather, weewx.restx.StdCWOP, weewx.restx.StdWOW, weewx.restx.StdAWEKAS
report_services = weewx.engine.StdPrint, weewx.engine.StdReport