| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952 |
- ###############################################################################
- #
- # Developed with Kate
- #
- # (c) 2017 Copyright: Marko Oldenburg (leongaultier at gmail dot com)
- # All rights reserved
- #
- # Special thanks goes to comitters:
- # - maddhin(FHEM Forum) Thanks for Readings and Commandref
- #
- #
- # This script is free software; you can redistribute it and/or modify
- # it under the terms of the GNU General Public License as published by
- # the Free Software Foundation; either version 2 of the License, or
- # any later version.
- #
- # The GNU General Public License can be found at
- # http://www.gnu.org/copyleft/gpl.html.
- # A copy is found in the textfile GPL.txt and important notices to the license
- # from the author is found in LICENSE.txt distributed with these scripts.
- #
- # This script is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- #
- # $Id: 46_Aqicn.pm 17701 2018-11-07 12:45:10Z CoolTux $
- #
- ###############################################################################
- ##
- ##
- ## Das JSON Modul immer in einem eval aufrufen
- # $data = eval{decode_json($data)};
- #
- # if($@){
- # Log3($SELF, 2, "$TYPE ($SELF) - error while request: $@");
- #
- # readingsSingleUpdate($hash, "state", "error", 1);
- #
- # return;
- # }
- #
- #######
- #######
- #
- ##
- ##
- package main;
- use strict;
- use warnings;
- my $version = "0.4.2";
- sub Aqicn_Initialize($) {
- my ($hash) = @_;
- # Consumer
- $hash->{GetFn} = "Aqicn::Get";
- $hash->{DefFn} = "Aqicn::Define";
- $hash->{UndefFn} = "Aqicn::Undef";
- $hash->{NotifyFn} = "Aqicn::Notify";
- $hash->{AttrFn} = "Aqicn::Attr";
- $hash->{AttrList} =
- "interval " . "disable:1 " . "language:de,en " . $readingFnAttributes;
- foreach my $d ( sort keys %{ $modules{Aqicn}{defptr} } ) {
- my $hash = $modules{Aqicn}{defptr}{$d};
- $hash->{VERSION} = $version;
- }
- }
- package Aqicn;
- use strict;
- use warnings;
- use HttpUtils;
- use GPUtils qw(GP_Import)
- ; # wird für den Import der FHEM Funktionen aus der fhem.pl benötigt
- my $missingModul = "";
- eval "use Encode qw(encode encode_utf8 decode_utf8);1"
- or $missingModul .= "Encode ";
- eval "use JSON;1" or $missingModul .= "JSON ";
- ## Import der FHEM Funktionen
- BEGIN {
- GP_Import(
- qw(readingsSingleUpdate
- readingsBulkUpdate
- readingsBulkUpdateIfChanged
- readingsBeginUpdate
- readingsEndUpdate
- defs
- modules
- Log3
- CommandAttr
- AttrVal
- IsDisabled
- deviceEvents
- init_done
- gettimeofday
- InternalTimer
- RemoveInternalTimer
- urlEncode
- HttpUtils_NonblockingGet
- makeDeviceName
- asyncOutput)
- );
- }
- ### Air Quality Index scale
- my %AQIS = (
- 1 => {
- 'i18nde' => 'Gut',
- 'i18nen' => 'Good',
- 'bgcolor' => '#009966',
- 'font color' => '#FFFFFF'
- },
- 2 => {
- 'i18nde' => 'Moderat',
- 'i18nen' => 'Moderate',
- 'bgcolor' => '#ffde33',
- 'font color' => '#000000'
- },
- 3 => {
- 'i18nde' => 'Ungesund für empfindliche Personengruppen',
- 'i18nen' => 'Unhealthy for Sensitive Groups',
- 'bgcolor' => '#ff9933',
- 'font color' => '#000000'
- },
- 4 => {
- 'i18nde' => 'Ungesund',
- 'i18nen' => 'Unhealthy',
- 'bgcolor' => '#cc0033',
- 'font color' => '#FFFFFF'
- },
- 5 => {
- 'i18nde' => 'Sehr ungesund',
- 'i18nen' => 'Very Unhealthy',
- 'bgcolor' => '#660099',
- 'font color' => '#FFFFFF'
- },
- 6 => {
- 'i18nde' => 'Gefährlich',
- 'i18nen' => 'Hazardous',
- 'bgcolor' => '#7e0023',
- 'font color' => '#FFFFFF'
- },
- );
- my %paths = (
- 'statussoe' => 'system_status/soe',
- 'aggregates' => 'meters/aggregates',
- 'siteinfo' => 'site_info',
- 'sitemaster' => 'sitemaster',
- 'powerwalls' => 'powerwalls',
- 'registration' => 'customer/registration',
- 'status' => 'status'
- );
- sub Define($$) {
- my ( $hash, $def ) = @_;
- my @a = split( "[ \t][ \t]*", $def );
- if ( $a[2] =~ /^token=/ ) {
- $a[2] =~ m/token=([^\s]*)/;
- $hash->{TOKEN} = $1;
- }
- else {
- $hash->{UID} = $a[2];
- }
- return "Cannot define a Aqicn device. Perl modul $missingModul is missing."
- if ($missingModul);
- return "too few parameters: define <name> Aqicn <OPTION-PARAMETER>"
- if ( @a != 3 );
- return "too few parameters: define <name> Aqicn token=<TOKEN-KEY>"
- if ( not defined( $hash->{TOKEN} )
- and not defined( $modules{Aqicn}{defptr}{TOKEN} ) );
- return "too few parameters: define <name> Aqicn <STATION-UID>"
- if ( not defined( $hash->{UID} )
- and defined( $modules{Aqicn}{defptr}{TOKEN} ) );
- my $name = $a[0];
- $hash->{VERSION} = $version;
- $hash->{NOTIFYDEV} = "global";
- if ( defined( $hash->{TOKEN} ) ) {
- return
- "there is already a Aqicn Head Device, did you want to define a Aqicn station use: define <name> Aqicn <STATION-UID>"
- if ( $modules{Aqicn}{defptr}{TOKEN} );
- $hash->{HOST} = 'api.waqi.info';
- CommandAttr( undef, $name . ' room AQICN' )
- if ( AttrVal( $name, 'room', 'none' ) eq 'none' );
- readingsSingleUpdate( $hash, "state", "ready for search", 1 );
- Log3 $name, 3,
- "Aqicn ($name) - defined Aqicn Head Device with API-Key $hash->{TOKEN}";
- $modules{Aqicn}{defptr}{TOKEN} = $hash;
- }
- elsif ( defined( $hash->{UID} ) ) {
- CommandAttr( undef, $name . ' room AQICN' )
- if ( AttrVal( $name, 'room', 'none' ) eq 'none' );
- $hash->{INTERVAL} = 3600;
- $hash->{HEADDEVICE} = $modules{Aqicn}{defptr}{TOKEN}->{NAME};
- readingsSingleUpdate( $hash, "state", "initialized", 1 );
- Log3 $name, 3,
- "Aqicn ($name) - defined Aqicn Station Device with Station UID $hash->{UID}";
- $modules{Aqicn}{defptr}{UID} = $hash;
- }
- return undef;
- }
- sub Undef($$) {
- my ( $hash, $arg ) = @_;
- my $name = $hash->{NAME};
- if ( defined( $modules{Aqicn}{defptr}{TOKEN} ) and $hash->{TOKEN} ) {
- return
- "there is a Aqicn Station Device present, please delete all Station Device first"
- unless ( not defined( $modules{Aqicn}{defptr}{UID} ) );
- delete $modules{Aqicn}{defptr}{TOKEN};
- }
- elsif ( defined( $modules{Aqicn}{defptr}{UID} ) and $hash->{UID} ) {
- delete $modules{Aqicn}{defptr}{UID};
- }
- RemoveInternalTimer($hash);
- Log3 $name, 3, "Aqicn ($name) - Device $name deleted";
- return undef;
- }
- sub Attr(@) {
- my ( $cmd, $name, $attrName, $attrVal ) = @_;
- my $hash = $defs{$name};
- if ( $attrName eq "disable" ) {
- if ( $cmd eq "set" and $attrVal eq "1" ) {
- RemoveInternalTimer($hash);
- readingsSingleUpdate( $hash, "state", "disabled", 1 );
- Log3 $name, 3, "Aqicn ($name) - disabled";
- }
- elsif ( $cmd eq "del" ) {
- Log3 $name, 3, "Aqicn ($name) - enabled";
- }
- }
- if ( $attrName eq "disabledForIntervals" ) {
- if ( $cmd eq "set" ) {
- return
- "check disabledForIntervals Syntax HH:MM-HH:MM or 'HH:MM-HH:MM HH:MM-HH:MM ...'"
- unless ( $attrVal =~ /^((\d{2}:\d{2})-(\d{2}:\d{2})\s?)+$/ );
- Log3 $name, 3, "Aqicn ($name) - disabledForIntervals";
- readingsSingleUpdate( $hash, "state", "disabled", 1 );
- }
- elsif ( $cmd eq "del" ) {
- Log3 $name, 3, "Aqicn ($name) - enabled";
- readingsSingleUpdate( $hash, "state", "active", 1 );
- }
- }
- if ( $attrName eq "interval" ) {
- if ( $cmd eq "set" ) {
- if ( $attrVal < 30 ) {
- Log3 $name, 3,
- "Aqicn ($name) - interval too small, please use something >= 30 (sec), default is 300 (sec)";
- return
- "interval too small, please use something >= 30 (sec), default is 300 (sec)";
- }
- else {
- RemoveInternalTimer($hash);
- $hash->{INTERVAL} = $attrVal;
- Log3 $name, 3, "Aqicn ($name) - set interval to $attrVal";
- Timer_GetData($hash);
- }
- }
- elsif ( $cmd eq "del" ) {
- RemoveInternalTimer($hash);
- $hash->{INTERVAL} = 300;
- Log3 $name, 3, "Aqicn ($name) - set interval to default";
- Timer_GetData($hash);
- }
- }
- return undef;
- }
- sub Notify($$) {
- my ( $hash, $dev ) = @_;
- my $name = $hash->{NAME};
- return if ( IsDisabled($name) );
- my $devname = $dev->{NAME};
- my $devtype = $dev->{TYPE};
- my $events = deviceEvents( $dev, 1 );
- return if ( !$events );
- Timer_GetData($hash)
- if (
- (
- grep /^INITIALIZED$/,
- @{$events}
- or grep /^REREADCFG$/,
- @{$events}
- or grep /^MODIFIED.$name$/,
- @{$events}
- or ( grep /^DEFINED.$name$/, @{$events} and $init_done )
- )
- and defined( $hash->{UID} )
- );
- return;
- }
- sub Get($$@) {
- my ( $hash, $name, @aa ) = @_;
- my ( $cmd, @args ) = @aa;
- if ( $cmd eq 'update' ) {
- GetData($hash);
- return undef;
- }
- elsif ( $cmd eq 'stationSearchByCity' ) {
- return "usage: $cmd" if ( @args == 0 );
- my $city = join( " ", @args );
- my $ret;
- $ret = GetData( $hash, $city );
- return $ret;
- }
- else {
- my $list = '';
- $list .= 'update:noArg' if ( defined( $hash->{UID} ) );
- $list .= 'stationSearchByCity' if ( defined( $hash->{TOKEN} ) );
- return "Unknown argument $cmd, choose one of $list";
- }
- }
- sub Timer_GetData($) {
- my $hash = shift;
- my $name = $hash->{NAME};
- if ( not IsDisabled($name) ) {
- GetData($hash);
- }
- else {
- readingsSingleUpdate( $hash, 'state', 'disabled', 1 );
- }
- InternalTimer( gettimeofday() + $hash->{INTERVAL},
- 'Aqicn::Timer_GetData', $hash );
- Log3 $name, 4, "Aqicn ($name) - Call InternalTimer Timer_GetData";
- }
- sub GetData($;$) {
- my ( $hash, $cityName ) = @_;
- my $name = $hash->{NAME};
- my $host = $modules{Aqicn}{defptr}{TOKEN}->{HOST};
- my $token = $modules{Aqicn}{defptr}{TOKEN}->{TOKEN};
- my $uri;
- if ( $hash->{UID} ) {
- my $uid = $hash->{UID};
- $uri = $host . '/feed/@' . $hash->{UID} . '/?token=' . $token;
- readingsSingleUpdate( $hash, 'state', 'fetch data', 1 );
- }
- else {
- $uri =
- $host
- . '/search/?token='
- . $token
- . '&keyword='
- . urlEncode($cityName);
- }
- my $param = {
- url => "https://" . $uri,
- timeout => 5,
- method => 'GET',
- hash => $hash,
- doTrigger => 1,
- callback => \&ErrorHandling,
- };
- $param->{cl} = $hash->{CL}
- if ( $hash->{TOKEN} and ref( $hash->{CL} ) eq 'HASH' );
- HttpUtils_NonblockingGet($param);
- Log3 $name, 4, "Aqicn ($name) - Send with URI: https://$uri";
- }
- sub ErrorHandling($$$) {
- my ( $param, $err, $data ) = @_;
- my $hash = $param->{hash};
- my $name = $hash->{NAME};
- Log3 $name, 4, "Aqicn ($name) - Recieve JSON data: $data";
- #Log3 $name, 3, "Aqicn ($name) - Recieve HTTP Code: $param->{code}";
- #Log3 $name, 3, "Aqicn ($name) - Recieve Error: $err";
- ### Begin Error Handling
- if ( defined($err) ) {
- if ( $err ne "" ) {
- if ( $param->{cl} && $param->{cl}{canAsyncOutput} ) {
- asyncOutput( $param->{cl}, "Request Error: $err\r\n" );
- }
- readingsBeginUpdate($hash);
- readingsBulkUpdate( $hash, 'state', $err, 1 );
- readingsBulkUpdate( $hash, 'lastRequestError', $err, 1 );
- readingsEndUpdate( $hash, 1 );
- Log3 $name, 3, "Aqicn ($name) - RequestERROR: $err";
- return;
- }
- }
- if ( $data eq "" and exists( $param->{code} ) && $param->{code} ne 200 ) {
- #if( $param->{cl} && $param->{cl}{canAsyncOutput} ) {
- # asyncOutput( $param->{cl}, "Request Error: $param->{code}\r\n" );
- #}
- readingsBeginUpdate($hash);
- readingsBulkUpdate( $hash, 'state', $param->{code}, 1 );
- readingsBulkUpdate( $hash, 'lastRequestError', $param->{code}, 1 );
- Log3 $name, 3, "Aqicn ($name) - RequestERROR: " . $param->{code};
- readingsEndUpdate( $hash, 1 );
- Log3 $name, 5,
- "Aqicn ($name) - RequestERROR: received http code "
- . $param->{code}
- . " without any data after requesting";
- return;
- }
- if ( ( $data =~ /Error/i ) and exists( $param->{code} ) ) {
- #if( $param->{cl} && $param->{cl}{canAsyncOutput} ) {
- # asyncOutput( $param->{cl}, "Request Error: $param->{code}\r\n" );
- #}
- readingsBeginUpdate($hash);
- readingsBulkUpdate( $hash, 'state', $param->{code}, 1 );
- readingsBulkUpdate( $hash, "lastRequestError", $param->{code}, 1 );
- readingsEndUpdate( $hash, 1 );
- Log3 $name, 3,
- "Aqicn ($name) - statusRequestERROR: http error " . $param->{code};
- return;
- ### End Error Handling
- }
- Log3 $name, 4, "Aqicn ($name) - Recieve JSON data: $data";
- ResponseProcessing( $hash, $data, $param );
- }
- sub ResponseProcessing($$$) {
- my ( $hash, $json, $param ) = @_;
- my $name = $hash->{NAME};
- my $decode_json;
- my $readings;
- $decode_json = eval { decode_json($json) };
- if ($@) {
- Log3 $name, 4, "Aqicn ($name) - error while request: $@";
- readingsBeginUpdate($hash);
- readingsBulkUpdate( $hash, 'JSON_Error', $@ );
- readingsBulkUpdate( $hash, 'httpCode', $param->{code} );
- readingsBulkUpdate( $hash, 'state', 'JSON error' );
- readingsEndUpdate( $hash, 1 );
- return;
- }
- #### Verarbeitung der Readings zum passenden
- if ( $hash->{TOKEN} ) {
- ReadingsProcessing_SearchStationResponse( $decode_json, $param );
- readingsSingleUpdate( $hash, 'state', 'search finished', 1 );
- return;
- }
- elsif ( $hash->{UID} ) {
- $readings = ReadingsProcessing_AqiResponse($decode_json);
- }
- WriteReadings( $hash, $readings );
- }
- sub WriteReadings($$) {
- my ( $hash, $readings ) = @_;
- my $name = $hash->{NAME};
- Log3 $name, 4, "Aqicn ($name) - Write Readings";
- readingsBeginUpdate($hash);
- while ( my ( $r, $v ) = each %{$readings} ) {
- readingsBulkUpdate( $hash, $r, $v );
- }
- if ( defined( $readings->{'PM2.5-AQI'} ) ) {
- readingsBulkUpdateIfChanged(
- $hash,
- 'htmlStyle',
- '<div style="background-color: '
- . $AQIS{ AirPollutionLevel( $readings->{'PM2.5-AQI'} ) }
- {'bgcolor'}
- . ';"><font color="'
- . $AQIS{ AirPollutionLevel( $readings->{'PM2.5-AQI'} ) }
- {'font color'} . '">'
- . (
- (
- (
- AttrVal( 'global', 'language', 'none' ) eq 'DE'
- or AttrVal( $name, 'language', 'none' ) eq 'de'
- )
- and AttrVal( $name, 'language', 'none' ) ne 'en'
- )
- ? "$AQIS{AirPollutionLevel($readings->{'PM2.5-AQI'})}{'i18nde'}: $readings->{'PM2.5-AQI'} "
- : " $AQIS{AirPollutionLevel($readings->{'PM2.5-AQI'})}{'i18nen'}: $readings->{'PM2.5-AQI'}"
- )
- . '</div>'
- );
- readingsBulkUpdateIfChanged(
- $hash, 'state',
- (
- (
- (
- AttrVal( 'global', 'language', 'none' ) eq 'DE'
- or AttrVal( $name, 'language', 'none' ) eq 'de'
- )
- and AttrVal( $name, 'language', 'none' ) ne 'en'
- )
- ? "$AQIS{AirPollutionLevel($readings->{'PM2.5-AQI'})}{'i18nde'}: $readings->{'PM2.5-AQI'}"
- : "$AQIS{AirPollutionLevel($readings->{'PM2.5-AQI'})}{'i18nen'}: $readings->{'PM2.5-AQI'}"
- )
- );
- readingsBulkUpdateIfChanged(
- $hash, 'APL',
- (
- (
- (
- AttrVal( 'global', 'language', 'none' ) eq 'DE'
- or AttrVal( $name, 'language', 'none' ) eq 'de'
- )
- and AttrVal( $name, 'language', 'none' ) ne 'en'
- )
- ? "$AQIS{AirPollutionLevel($readings->{'PM2.5-AQI'})}{'i18nde'}"
- : "$AQIS{AirPollutionLevel($readings->{'PM2.5-AQI'})}{'i18nen'}"
- )
- );
- readingsBulkUpdateIfChanged(
- $hash,
- 'healthImplications',
- HealthImplications(
- $hash, AirPollutionLevel( $readings->{'PM2.5-AQI'} )
- )
- );
- }
- else {
- readingsBulkUpdateIfChanged(
- $hash,
- 'htmlStyle',
- '<div style="background-color: '
- . $AQIS{ AirPollutionLevel( $readings->{'AQI'} ) }{'bgcolor'}
- . ';"><font color="'
- . $AQIS{ AirPollutionLevel( $readings->{'PM2.5-AQI'} ) }
- {'font color'} . '">'
- . (
- (
- (
- AttrVal( 'global', 'language', 'none' ) eq 'DE'
- or AttrVal( $name, 'language', 'none' ) eq 'de'
- )
- and AttrVal( $name, 'language', 'none' ) ne 'en'
- )
- ? "$AQIS{AirPollutionLevel($readings->{'AQI'})}{'i18nde'}: $readings->{'AQI'} "
- : " $AQIS{AirPollutionLevel($readings->{'AQI'})}{'i18nen'}: $readings->{'AQI'}"
- )
- . '</div>'
- );
- readingsBulkUpdateIfChanged(
- $hash, 'state',
- (
- (
- (
- AttrVal( 'global', 'language', 'none' ) eq 'DE'
- or AttrVal( $name, 'language', 'none' ) eq 'de'
- )
- and AttrVal( $name, 'language', 'none' ) ne 'en'
- )
- ? "$AQIS{AirPollutionLevel($readings->{'AQI'})}{'i18nde'}: $readings->{'AQI'}"
- : "$AQIS{AirPollutionLevel($readings->{'AQI'})}{'i18nen'}: $readings->{'AQI'}"
- )
- );
- readingsBulkUpdateIfChanged(
- $hash, 'APL',
- (
- (
- (
- AttrVal( 'global', 'language', 'none' ) eq 'DE'
- or AttrVal( $name, 'language', 'none' ) eq 'de'
- )
- and AttrVal( $name, 'language', 'none' ) ne 'en'
- )
- ? "$AQIS{AirPollutionLevel($readings->{'AQI'})}{'i18nde'}"
- : "$AQIS{AirPollutionLevel($readings->{'AQI'})}{'i18nen'}"
- )
- );
- readingsBulkUpdateIfChanged(
- $hash,
- 'healthImplications',
- HealthImplications(
- $hash, AirPollutionLevel( $readings->{'AQI'} )
- )
- );
- }
- readingsEndUpdate( $hash, 1 );
- }
- #####
- #####
- ## my little Helper
- sub ReadingsProcessing_SearchStationResponse($$) {
- my ( $decode_json, $param ) = @_;
- if ( $param->{cl} and $param->{cl}->{TYPE} eq 'FHEMWEB' ) {
- my $ret = '<html><table><tr><td>';
- $ret .= '<table class="block wide">';
- $ret .= '<tr class="even">';
- $ret .= "<td><b>City</b></td>";
- $ret .= "<td><b>Last Update Time</b></td>";
- $ret .= "<td><b>Latitude</b></td>";
- $ret .= "<td><b>Longitude</b></td>";
- $ret .= "<td></td>";
- $ret .= '</tr>';
- if ( ref( $decode_json->{data} ) eq "ARRAY"
- and scalar( @{ $decode_json->{data} } ) > 0 )
- {
- my $linecount = 1;
- foreach my $dataset ( @{ $decode_json->{data} } ) {
- if ( $linecount % 2 == 0 ) {
- $ret .= '<tr class="even">';
- }
- else {
- $ret .= '<tr class="odd">';
- }
- $dataset->{station}{name} =~ s/'//g;
- $ret .=
- "<td>" . encode_utf8( $dataset->{station}{name} ) . "</td>";
- $ret .= "<td>$dataset->{'time'}{stime}</td>";
- $ret .= "<td>$dataset->{station}{geo}[0]</td>";
- $ret .= "<td>$dataset->{station}{geo}[1]</td>";
- ###### create Links
- my $aHref;
- # create Google Map Link
- $aHref =
- "<a target=\"_blank\" href=\"https://www.google.de/maps/search/"
- . $dataset->{station}{geo}[0] . "+"
- . $dataset->{station}{geo}[1]
- . "\">Station on Google Maps</a>";
- $ret .= "<td>" . $aHref . "</td>";
- # create define Link
- $aHref =
- "<a href=\""
- . $::FW_httpheader->{host}
- . "/fhem?cmd=define+"
- . makeDeviceName( $dataset->{station}{name} )
- . "+Aqicn+"
- . $dataset->{uid}
- . $::FW_CSRF
- . "\">Create Station Device</a>";
- $ret .= "<td>" . $aHref . "</td>";
- $ret .= '</tr>';
- $linecount++;
- }
- }
- $ret .= '</table></td></tr>';
- $ret .= '</table></html>';
- asyncOutput( $param->{cl}, $ret )
- if ( $param->{cl} and $param->{cl}{canAsyncOutput} );
- return;
- }
- elsif ( $param->{cl} and $param->{cl}->{TYPE} eq 'telnet' ) {
- my $ret = '';
- foreach my $dataset ( @{ $decode_json->{data} } ) {
- $ret .=
- encode_utf8( $dataset->{station}{name} )
- . "| $dataset->{'time'}{stime} | $dataset->{station}{geo}[0] | $dataset->{station}{geo}[1] | define "
- . makeDeviceName( $dataset->{station}{name} )
- . " Aqicn $dataset->{uid}\r\n";
- }
- asyncOutput( $param->{cl}, $ret )
- if ( $param->{cl} && $param->{cl}{canAsyncOutput} );
- return;
- }
- }
- sub ReadingsProcessing_AqiResponse($) {
- my ($decode_json) = @_;
- my %readings;
- if ( ref( $decode_json->{data} ) eq "HASH" ) {
- $readings{'CO-AQI'} = $decode_json->{data}{iaqi}{co}{v};
- $readings{'NO2-AQI'} = $decode_json->{data}{iaqi}{no2}{v};
- $readings{'PM10-AQI'} = $decode_json->{data}{iaqi}{pm10}{v};
- $readings{'PM2.5-AQI'} = $decode_json->{data}{iaqi}{pm25}{v};
- $readings{'AQI'} = $decode_json->{data}{aqi};
- $readings{'O3-AQI'} = $decode_json->{data}{iaqi}{o3}{v};
- $readings{'SO2-AQI'} = $decode_json->{data}{iaqi}{so2}{v};
- $readings{'temperature'} = $decode_json->{data}{iaqi}{t}{v};
- $readings{'pressure'} = $decode_json->{data}{iaqi}{p}{v};
- $readings{'humidity'} = $decode_json->{data}{iaqi}{h}{v};
- $readings{'status'} = $decode_json->{status};
- $readings{'pubDate'} = $decode_json->{data}{time}{s};
- $readings{'pubUnixTime'} = $decode_json->{data}{time}{v};
- $readings{'pubTimezone'} = $decode_json->{data}{time}{tz};
- $readings{'windSpeed'} = $decode_json->{data}{iaqi}{w}{v};
- $readings{'windDirection'} = $decode_json->{data}{iaqi}{wd}{v};
- $readings{'dewpoint'} = $decode_json->{data}{iaqi}{d}{v};
- $readings{'dominatPoll'} = $decode_json->{data}{dominentpol};
- }
- else {
- $readings{'status'} = 'no hash reference found';
- }
- return \%readings;
- }
- sub AirPollutionLevel($) {
- my $aqi = shift;
- return 1 unless ( defined($aqi) );
- my $apl;
- if ( $aqi < 51 ) { $apl = 1 }
- elsif ( $aqi < 101 ) { $apl = 2 }
- elsif ( $aqi < 151 ) { $apl = 3 }
- elsif ( $aqi < 201 ) { $apl = 4 }
- elsif ( $aqi < 301 ) { $apl = 5 }
- else { $apl = 6 }
- return $apl;
- }
- sub HealthImplications($$) {
- my ( $hash, $apl ) = @_;
- my $name = $hash->{NAME};
- my %HIen = (
- 1 =>
- 'Air quality is acceptable; however, for some pollutants there may be a moderate health concern for a very small number of people who are unusually sensitive to air pollution.',
- 2 =>
- 'Air quality is acceptable; however, for some pollutants there may be a moderate health concern for a very small number of people who are unusually sensitive to air pollution.',
- 3 =>
- 'Members of sensitive groups may experience health effects. The general public is not likely to be affected.',
- 4 =>
- 'Everyone may begin to experience health effects; members of sensitive groups may experience more serious health effects',
- 5 =>
- 'Health warnings of emergency conditions. The entire population is more likely to be affected.',
- 6 => 'Health alert: everyone may experience more serious health effects'
- );
- my %HIde = (
- 1 =>
- 'Die Qualität der Luft gilt als zufriedenstellend und die Luftverschmutzung stellt ein geringes oder kein Risiko dar',
- 2 =>
- 'Die Luftqualität ist insgesamt akzeptabel. Bei manchen Schadstoffe besteht jedoch eventuell eine geringe Gesundheitsgefahr für einen sehr kleinen Personenkreis, der sehr empfindlich auf Luftverschmutzung ist.',
- 3 =>
- 'Bei Mitgliedern von empfindlichen Personengruppen können gesundheitliche Auswirkungen auftreten. Die allgemeine Öffentlichkeit ist wahrscheinlich nicht betroffen.',
- 4 =>
- 'Erste gesundheitliche Auswirkungen können sich bei allen Personen einstellen. Bei empfindlichen Personengruppen können ernstere gesundheitliche Auswirkungen auftreten.',
- 5 =>
- 'Gesundheitswarnung aufgrund einer Notfallsituation. Die gesamte Bevölkerung ist voraussichtlich betroffen.',
- 6 =>
- 'Gesundheitsalarm: Jeder muss mit dem Auftreten ernsterer Gesundheitsschäden rechnen'
- );
- return (
- (
- AttrVal( 'global', 'language', 'none' ) eq 'DE'
- or AttrVal( $name, 'language', 'none' ) eq 'de'
- )
- and AttrVal( $name, 'language', 'none' ) ne 'en'
- ? $HIde{$apl}
- : $HIen{$apl}
- );
- }
- 1;
- =pod
- =item device
- =item summary Air Quality Index proving a transparent Air Quality information
- =item summary_DE Air Quality Index Nachweis einer transparenten Luftqualitätsinformation
- =begin html
- <a name="Aqicn"></a>
- <h3>Air Quality Index</h3>
- <ul>
- This modul fetch Air Quality data from http://aqicn.org.
- <br><br>
- <a name="Aqicndefine"></a>
- <b>Define</b>
- <ul><br>
- <code>define <name> Aqicn token=<TOKEN-KEY></code>
- <br><br>
- Example:
- <ul><br>
- <code>define aqicnMaster Aqicn token=12345678</code><br>
- </ul>
- <br>
- This statement creates the Aqicn Master Device.<br>
- After the device has been created, you can search Aqicn Station by city name and create automatically the station device.
- </ul>
- <br><br>
- <a name="Aqicnreadings"></a>
- <b>Readings</b>
- <ul>
- <li>APL - Air Pollution Level</li>
- <li>AQI - Air Quality Index (AQI) of the dominant pollutant in city. Values are converted from µg/m³ to AQI level using US EPA standards. For more detailed information: https://en.wikipedia.org/wiki/Air_quality_index and https://www.airnow.gov/index.cfm?action=aqi_brochure.index. </li>
- <li>CO-AQI - AQI of CO (carbon monoxide). An AQI of 100 for carbon monoxide corresponds to a level of 9 parts per million (averaged over 8 hours).</li>
- <li>NO2-AQI - AQI of NO2 (nitrogen dioxide). See also https://www.airnow.gov/index.cfm?action=pubs.aqiguidenox</li>
- <li>PM10-AQI - AQI of PM10 (respirable particulate matter). For particles up to 10 micrometers in diameter: An AQI of 100 corresponds to 150 micrograms per cubic meter (averaged over 24 hours).</li>
- <li>PM2.5-AQI - AQI of PM2.5 (fine particulate matter). For particles up to 2.5 micrometers in diameter: An AQI of 100 corresponds to 35 micrograms per cubic meter (averaged over 24 hours).</li>
- <li>O3-AQI - AQI of O3 (ozone). An AQI of 100 for ozone corresponds to an ozone level of 0.075 parts per million (averaged over 8 hours). See also https://www.airnow.gov/index.cfm?action=pubs.aqiguideozone</li>
- <li>SO2-AQI - AQI of SO2 (sulfur dioxide). An AQI of 100 for sulfur dioxide corresponds to a level of 75 parts per billion (averaged over one hour).</li>
- <li>temperature - Temperature in degrees Celsius</li>
- <li>pressure - Atmospheric pressure in hectopascals (hPa)</li>
- <li>humidity - Relative humidity in percent</li>
- <li>state- Current AQI and air pollution level</li>
- <li>status - condition of the data</li>
- <li>pubDate- Local time of publishing the data</li>
- <li>pubUnixTime - Unix time stamp of local time but converted wrongly, if local time is e.g. 1300 GMT+1, the time stamp shows 1300 UTC.</li>
- <li>pubTimezone - Time zone of the city (UTC)</li>
- <li>windspeed - Wind speed in kilometer per hour</li>
- <li>windDirection - Wind direction</li>
- <li>dominatPoll - Dominant pollutant in city</li>
- <li>dewpoint - Dew in degrees Celsius</li>
- <li>healthImplications - Information about Health Implications</li>
- <li>htmlStyle - can be used to format the STATE and FHEMWEB (Example: stateFormate htmlStyle</li>
- </ul>
- <br>
- <a name="Aqicnget"></a>
- <b>get</b>
- <ul>
- <li>stationSearchByCity - search station by city name and open the result in seperate popup window</li>
- <li>update - fetch new data every x times</li>
- </ul>
- <br>
- <a name="Aqicnattribute"></a>
- <b>Attribute</b>
- <ul>
- <li>interval - interval in seconds for automatically fetch data (default 3600)</li>
- </ul>
- </ul>
- =end html
- =begin html_DE
- <a name="Aqicn"></a>
- <h3>Air Quality Index</h3>
- =end html_DE
- =cut
|