Seite 1 von 1
GOOGLE WETTER XML API DLL
Verfasst: Freitag 22. Juni 2012, 13:33
von ExpertProfi
Hallo Kollegen,
nach längerer Abstinenz melde ich mich heute zurück mit einem neuen Projekt, das vielleicht Euren Geschmack trifft...
Abfrage der Google-Stadtwetter-Info aus dem Internet, wie z.B.
http://www.google.com/ig/api?weather=Kiel&hl=DE
für sehr viele Städte und in diversen Sprachen
Um die Daten in PL verfügbar zu machen, habe ich eine DLL geschrieben,
die die Daten abruft und die XML-Info parst. Google hat die Schnittstelle zwar nicht
dokumentiert, aber sie funktioniert z.Zt. prima. (Natürlich weiss ich nicht ob der
Service für immer lizenzfrei, offen und unverändert weiter besteht.)
Für alle die wissen möchten was die DLL macht, füge ich unten die Source an.
Hier nun das Ergebnis... Viel Spass mit der 0-cost-Wetterstation ... Freue mich auf Rückmeldungen...
DOWNLOAD:
http://www.expertprofi.wg.am/Google_Wetter_API_DLL.zip
- Wetter.jpg (99.76 KiB) 14242 mal betrachtet
Re: GOOGLE WETTER XML API DLL
Verfasst: Freitag 22. Juni 2012, 13:34
von ExpertProfi
... und hier die DLL-Source ...
Code: Alles auswählen
library Wetter;
// Google Wetter XML API
// Beispiel: http://www.google.com/ig/api?weather=Berlin';
uses
SysUtils,
Windows,
Classes,
VCL.controls, VCL.Forms,
idHttp, Xml.xmldom, Xml.XMLIntf, Xml.Win.msxmldom, Xml.XMLDoc,
IdBaseComponent, IdComponent, IdTCPConnection, IdTCPClient;
{$R *.RES}
Const Server = 'http://www.google.com/ig/api?weather=';
Inputs = 3; // number of inputs
Outputs = 12; // number of outputs
{INPUTS}
LocationInPin = 0; // $PLZ oder ORT
LanguageInPin = 1; // DE, UK, ...
CLKPin = 2; // Clock
{USER}
DLLIndex = 100; // DLL No. set by ProfiLab
Type TDLLParams = array[0..100] of extended; // Type of ProfiLab DLL parameters
PDLLParams = ^TDLLParams; // Pointer to ProfiLab DLL parameters
TStringParams = array[0..100] of PAnsiChar; // Additional string parameters for CalculateEx
PStringParams = ^TStringParams; // Pointer to string parameters
var XMLDoc: TXMLDocument;
DataList: array[0..100] of TStringList;
Location: String;
Language: String;
function NumInputs: Byte;
begin
result:=Inputs; //Define number of component input pins
end;
function NumOutputs: Byte;
begin
result:=Outputs; //Define number of component output pins
end;
Function InputName(Channel: Byte): ShortString; // Return name for each component input pin
begin
// string inputs must be named with leading '$' character
case Channel of
LocationInPin: result:='$LOC'; // Location: PLZ oder Ort
LanguageInPin: result:='$LAN'; // Language de, uk, ...
ClkPin: result:='CLK'; // Update Clk
end;
end;
Function OutputName(Channel: Byte): ShortString; // Return name for each component output pin
begin
// string outputs must be named with leading '$' character!
case Channel of
0: result:='$LOC'; // Ort
1: result:='$ZIP'; // PLZ
2: result:='$DTE'; // Datum
3: result:='$CDN'; // aktuelle Wetterbdingung
4: result:='$ °F'; // Aktuelle Temperatur F
5: result:='$ °C'; // Aktuelle Temperatur C
6: result:='$HUM'; // Luftfeuchtigkeit %
7: result:='$WND'; // Wind
8: result:='$FC1'; // Vorhersage heute + 1 Tag
9: result:='$FC2'; // Vorhersage heute + 2 Tag
10: result:='$FC3'; // Vorhersage heute + 3 Tag
11: result:='$FC4'; // Vorhersage heute + 4 Tag
end;
end;
// ########################### XML PARSING #########################
function GetNodeData(aNode: IXMLNode): String;
begin
result:='';
if aNode=nil then exit;
result:=aNode.LocalName;
try
result:=aNode.Attributes['data'];
except
end;
end;
procedure GetNodeChildren(aNode: IXMLNode);
var i: Integer;
ChildNode: IXMLNode;
aString: String;
begin
if aNode=nil then exit;
for i:=0 to aNode.ChildNodes.Count-1 do
begin
ChildNode:=aNode.ChildNodes[i];
DataList[0].add(GetNodeData(ChildNode));
GetNodeChildren(ChildNode);
end;
end;
function NodeByLocalName(aNode: IXMLNode; aLocalName: String; var Appear: Integer): IXMLNode;
var ChildNode: IXMLNode;
i: Integer;
begin
result:=nil;
if aNode=nil then exit;
if aNode.LocalName=aLocalName then
begin
dec(Appear);
if appear<=0 then
begin
result:=aNode;
exit;
end;
end;
for i:=0 to aNode.ChildNodes.Count-1 do
begin
ChildNode:=aNode.ChildNodes[i];
result:=NodeByLocalName(ChildNode, aLocalName, Appear);
if result<>nil then exit;
end;
end;
function StreamToString(Stream : TStream) : String;
var ms : TMemoryStream;
begin
Result := '';
ms := TMemoryStream.Create;
try
ms.LoadFromStream(Stream);
SetString(Result,PAnsiChar(ms.memory),ms.Size);
finally
ms.free;
end;
end;
function ReadSite(URI: string): String; // Request mit Indy 10 componenten
var
idClient: TIdHTTP;
msData: TMemoryStream;
ResultString: String;
begin
ResultString:='';
idClient := TIdHTTP.Create(Application);
idClient.Request.AcceptLanguage:=Language;
msData := TMemoryStream.Create;
try
idClient.Get(URI, msData);
msData.Seek(0, soFromBeginning);
ResultString := StreamToString(msData);
finally
idClient.Free;
msData.Free;
end;
result := ResultString;
end;
procedure GetDataList; // XML PARSER
var InfoNode: IXMLNode;
ConditionsNode: IXMLNode;
ForecastNode: IXMLNode;
aNode: IXMLNode;
ForeCastString, UnitSystem: String;
Appear,i: Integer;
begin
xmldoc.XML.Add(ReadSite('http://www.google.com/ig/api?weather='+Location));
xmldoc.Active:=true;
//General Info
Appear:=1;
InfoNode:=NodeByLocalName(xmldoc.DocumentElement, 'forecast_information', Appear);
Appear:=1;
aNode:=NodeByLocalName(InfoNode, 'city', Appear);
DataList[0].Add(GetNodeData(aNode));
Appear:=1;
aNode:=NodeByLocalName(InfoNode, 'postal_code', Appear);
DataList[0].Add(GetNodeData(aNode));
Appear:=1;
aNode:=NodeByLocalName(InfoNode, 'forecast_date', Appear);
DataList[0].Add(GetNodeData(aNode));
Appear:=1;
aNode:=NodeByLocalName(InfoNode, 'unit_system', Appear);
UnitSystem:=GetNodeData(aNode);
// Current condition
Appear:=1;
ConditionsNode:=NodeByLocalName(xmldoc.DocumentElement, 'current_conditions', Appear);
Appear:=1;
aNode:=NodeByLocalName(ConditionsNode, 'condition', Appear);
DataList[0].Add(GetNodeData(aNode));
Appear:=1;
aNode:=NodeByLocalName(ConditionsNode, 'temp_f', Appear);
DataList[0].Add(GetNodeData(aNode)+' °F');
Appear:=1;
aNode:=NodeByLocalName(ConditionsNode, 'temp_c', Appear);
Appear:=1;
DataList[0].Add(GetNodeData(aNode)+' °C');
Appear:=1;
aNode:=NodeByLocalName(ConditionsNode, 'humidity', Appear);
DataList[0].Add(GetNodeData(aNode));
Appear:=1;
aNode:=NodeByLocalName(ConditionsNode, 'wind_condition', Appear);
DataList[0].Add(GetNodeData(aNode));
//4x Forcast
for i:=1 to 4 do
begin
Appear:=i;
ForeCastNode:=NodeByLocalName(xmldoc.DocumentElement, 'forecast_conditions', Appear);
Appear:=1;
aNode:=NodeByLocalName(ForecastNode, 'day_of_week', Appear);
ForeCastString:=GetNodeData(aNode)+';';
Appear:=1;
aNode:=NodeByLocalName(ForecastNode, 'condition', Appear);
ForeCastString:=ForeCastString+GetNodeData(aNode)+';';
Appear:=1;
aNode:=NodeByLocalName(ForecastNode, 'low', Appear);
ForeCastString:=ForeCastString+GetNodeData(aNode)+';';
Appear:=1;
aNode:=NodeByLocalName(ForecastNode, 'high', Appear);
ForeCastString:=ForeCastString+GetNodeData(aNode)+';'+UnitSystem;
DataList[0].Add(ForeCastString);
end;
xmldoc.Active:=false;
end;
// ##################
// New CalculateEx method now allows string parameters...
Procedure CalculateEx(PInput,POutput,PUser: PDLLParams; PStrings: PStringParams); //called regularly from ProfiLab
var aString: AnsiString;
aWord: Word;
aChar: AnsiChar;
aExtended: extended;
Hour, Min, Sec, MSec: Word;
i: integer;
DLLNo: Integer;
begin
DLLNo:=round(PUser^[DLLIndex]); // Die ProfiLab DLL Nummer
if Datalist[0]=nil then DataList[0]:=TStringlist.Create; // Templiste zum Daten holen
if Datalist[DllNo]=nil then DataList[DllNo]:=TStringlist.Create; // Listen für bis zu 100 DLL-Import (Wetter für 100 Orte)
if (PInput^[ClkPin]<2.5) and (PUser^[ClkPin]>=2.5) then // Abfrage Taktsignal
begin
Location:=strPas(PStrings[LocationInPin]); // Der Ort
while pos(' ',location)>0 do location:=StringReplace(Location,' ',' ',[rfReplaceAll]);
location:=StringReplace(Location,' ','+',[rfReplaceAll]);
Language:=strPas(PStrings[LanguageInPin]); // Die Sprache
Datalist[0].Clear; // Templiste leeren
XMLDoc:=TXMLDocument.Create(Application); // XML vorbereiten
GetDataList; // Daten holen und XML parsen => Ergebnis in Datalist[0]
DataList[DLLno].Assign(Datalist[0]); // Tempdaten der passenden DLL zuordnen
XMLDoc.Free; // Daten fertig
end;
PUser^[ClkPin]:=PInput^[ClkPin]; // Alten Clock merken
// Ausgangswerte (aus passender Datenliste) ausgeben
for i:=0 to 11 do
begin
if DataList[DllNo].count>i then strPCopy(PStrings[i], DataList[DLLNo][i])
else strPCopy(PStrings[i], '');
end;
end;
//export methods for ProfiLab
exports NumInputs,
NumOutputs,
CalculateEx,
InputName,
OutputName;
begin
end.
Re: GOOGLE WETTER XML API DLL
Verfasst: Freitag 22. Juni 2012, 19:54
von HCS
Ein wahrhaftiger EXPERTPROFI!! Respekt!!
Werde versuchen, eine intelligente Brenner - Solarverknüpfung daraus zu machen.
So könnte man vorausschauend den Boiler laden und morgens wenn Sonne zu erwarten ist, den Brenner sperren.
Hast du dann vor, wenn Google das Protokoll etwas ändert, deine DLL wieder anzupassen?
Gruß
Bernd
Re: GOOGLE WETTER XML API DLL
Verfasst: Montag 25. Juni 2012, 08:01
von ExpertProfi
Danke, die Idee stammt aus einem Artikel, auf den ich zufällig gestossen bin...
http://www.keepaneye.de/wetterdaten-aus ... verwenden/
Dank fertiger Komponenten für XML und INDY10 für Netzprotokolle war die Sache nicht
allzu schwierig. Ich habe versucht das Parsen so zu gestalten, das die XML-Tags
auch dann noch funktionieren müssten, wenn Google den Service erweitern sollte,
solange die derzeitigen Tags erhalten bleiben. Ich wollte nur darauf hinweisen,
dass die Komponente von einem Google-Dienst abhängig ist, d.h. man weiss nicht
was die Zukunft bringt.
P.S. Ich habe gerade noch festgestellt, dass im ersten Upload noch ein Bug war.
(Schutzverletzung bei fehlender Netzwerkverbindung)
Ich habe jetzt eine neue Version hochgeladen. Bitte obigen Download ggf. nochmasl durchführen.
Die Source der neuen Version ist im Download enthalten.
25. Juni 2012 - 10.15 Uhr MESZ
Re: GOOGLE WETTER XML API DLL
Verfasst: Dienstag 26. Juni 2012, 08:11
von abacom
Wir freuen uns über dieses gelungene Projekt von ExpertProfi, welches vor längerer Zeit auch schon einmal angeregt wurde.
Wir selbst hatten etwas Ähnliches auch schon einmal ins Auge gefasst, aber mangels einer verlässlichen Dokumentation
seitens Google wieder verwerfen müssen. Gute Arbeit und vielen Dank!
Re: GOOGLE WETTER XML API DLL
Verfasst: Dienstag 26. Juni 2012, 11:24
von KAKTUS
Tolles Projekt.
Kaktus sagt DANKE !
PS: mit welcher Software hast du die DLL erstellt?
Re: GOOGLE WETTER XML API DLL
Verfasst: Mittwoch 27. Juni 2012, 07:16
von ExpertProfi
Hallo Kaktus,
die DLL ist mit Delphi XE2 erstellt. In älteren Delphi-Versionen
wird man die verwendeten Komponenten
(INDY10 und XML) vermutlich vergeblich suchen.
Es gibt auch eine kostenlose 30-Tage Trial-Version...
http://www.embarcadero.com/de/products/delphi
Re: GOOGLE WETTER XML API DLL
Verfasst: Mittwoch 27. Juni 2012, 15:53
von KAKTUS
Danke
Re: GOOGLE WETTER XML API DLL
Verfasst: Freitag 29. Juni 2012, 12:56
von tom_g
Hi ExpertProfi,
gelungene DLL ! Das Forum lebt von solchen "Perlen" wie Deiner Wetter-DLL !
Ich habe die Daten mal geloggt "messen übers Web"
Blau: Feuchte, rot: Temp, grün: Windspeed
Anbei das Beispiel des Klimas an meinem Wohnort über vergangene 2 Tage.
Wieso jittern die Signale so ?
Ich habe die Strings von Temp, Feuchtigkeit und Windspeed nur mit "value" in einen Wert konvertiert, und nicht weiter untersucht, ob die Konversion in allen Fällen gut funktioniert.
Beste Grüsse von Thomas
Re: GOOGLE WETTER XML API DLL
Verfasst: Montag 2. Juli 2012, 08:07
von abacom
Das müsste man dann wohl Google fragen. Der Dienst ist vermutlich für die Google Wetter-Gadgets.
http://www.google.com/ig/directory?type ... eather.xml
Re: GOOGLE WETTER XML API DLL
Verfasst: Sonntag 2. September 2012, 09:18
von HCS
... leider geht die Google-Funktion scheinbar nicht mehr!
Sehr schade.
Bernd
Re: GOOGLE WETTER XML API DLL
Verfasst: Donnerstag 4. Oktober 2012, 17:58
von Bernie
Hallo zusammen,
ich bekomme folgende Meldung:
Hat jemand eine Idee ?
Viele Grüße
Bernd
Re: GOOGLE WETTER XML API DLL
Verfasst: Donnerstag 4. Oktober 2012, 19:46
von KAKTUS
Re: GOOGLE WETTER XML API DLL
Verfasst: Dienstag 9. Oktober 2012, 12:38
von Bernie
Danke Kaktus für die schnelle Antwort. Da hätte ich ja auch selbst drauf kommen können
Viele Grüße
Bernd