Source for file Weatherdotcom.php
Documentation is available at Weatherdotcom.php
/* vim: set expandtab tabstop=4 shiftwidth=4: */
// +----------------------------------------------------------------------+
// +----------------------------------------------------------------------+
// | Copyright (c) 1997-2004 The PHP Group |
// +----------------------------------------------------------------------+
// | This source file is subject to version 2.0 of the PHP license, |
// | that is bundled with this package in the file LICENSE, and is |
// | available through the world-wide-web at |
// | http://www.php.net/license/2_02.txt. |
// | If you did not receive a copy of the PHP license and are unable to |
// | obtain it through the world-wide-web, please send a note to |
// | license@php.net so we can mail you a copy immediately. |
// +----------------------------------------------------------------------+
// | Authors: Alexander Wirtz <alex@pc4p.net> |
// +----------------------------------------------------------------------+
// $Id: Weatherdotcom.php,v 1.43 2004/07/10 11:12:44 eru Exp $
* @package Services_Weather
require_once "Services/Weather/Common.php";
// {{{ class Services_Weather_Weatherdotcom
* PEAR::Services_Weather_Weatherdotcom
* This class acts as an interface to the xml service of weather.com. It
* searches for given locations and retrieves current weather data as well
* as forecast for up to 10 days.
* For using the weather.com xml-service please visit
* http://www.weather.com/services/xmloap.html
* and follow the link to sign up, it's free! You will receive an email
* where to download the SDK with the needed images and guidelines how to
* publish live data from weather.com. Unfortunately the guidelines are a
* bit harsh, that's why there's no actual data-representation in this
* class, just the raw data. Also weather.com demands active caching, so I'd
* strongly recommend enabling the caching implemented in this class. It
* obeys to the times as written down in the guidelines.
* For working examples, please take a look at
* docs/Services_Weather/examples/weather.com-basic.php
* docs/Services_Weather/examples/weather.com-extensive.php
* @author Alexander Wirtz <alex@pc4p.net>
* @link http://www.weather.com/services/xmloap.html
* @example examples/weather.com-basic.php weather.com-basic.php
* @example examples/weather.com-extensive.php weather.com-extensive.php
* @package Services_Weather
* @license http://www.php.net/license/2_02.txt
* Partner-ID at weather.com
* @var string $_partnerID
* License key at weather.com
* @var string $_licenseKey
* Object containing the promotional links-data
* @var object stdClass $_links
* XML_Unserializer, used for processing the xml
* @var object XML_Unserializer $_unserializer
* Requires XML_Serializer to be installed
* @see Science_Weather::Science_Weather
function Services_Weather_Weatherdotcom ($options, &$error)
$this->Services_Weather_Common ($options, $perror);
// Set options accordingly
if (isset ($options["partnerID"])) {
if (isset ($options["licenseKey"])) {
include_once "XML/Unserializer.php";
$unserializer = &new XML_Unserializer (array ("complexType" => "object", "keyAttribute" => "type"));
$this->_unserializer = $unserializer;
// Can't acquire an object here, has to be clean on every request
include_once "HTTP/Request.php";
* Sets the neccessary account-information for weather.com, you'll
* receive them after registering for the XML-stream
* @param string $partnerID
* @param string $licenseKey
$this->_partnerID = $partnerID;
$this->_licenseKey = $licenseKey;
// {{{ _checkLocationID()
* Checks the id for valid values and thus prevents silly requests to
* @return PEAR_Error|bool
* @throws PEAR_Error::SERVICES_WEATHER_ERROR_NO_LOCATION
* @throws PEAR_Error::SERVICES_WEATHER_ERROR_INVALID_LOCATION
function _checkLocationID ($id)
// {{{ _parseWeatherData()
* Parses the data returned by the provided URL and caches it
* @return PEAR_Error|bool
* @throws PEAR_Error::SERVICES_WEATHER_ERROR_WRONG_SERVER_DATA
function _parseWeatherData ($id, $url)
$request = &new HTTP_Request ($url, array ("timeout" => $this->_httpTimeout));
$status = $request->sendRequest ();
$data = $request->getResponseBody ();
$status = $this->_unserializer->unserialize ($data);
$root = $this->_unserializer->getRootName ();
$data = $this->_unserializer->getUnserializedData ();
// Something wrong here...
} elseif ($root == "error") {
// We got an error back from weather.com
// Valid data, lets get started
// Loop through the different sub-parts of the data fro processing
$this->{"_". $varname} = $val;
if ($this->_cacheEnabled) {
// ...and cache if possible
$this->_cache->extSave ($id, $val, "", $expire, $varname);
* Searches IDs for given location, returns array of possible locations
* @param string $location
* @param bool $useFirst If set, first ID of result-array is returned
* @return PEAR_Error|array|string
* @throws PEAR_Error::SERVICES_WEATHER_ERROR_WRONG_SERVER_DATA
* @throws PEAR_Error::SERVICES_WEATHER_ERROR_UNKNOWN_LOCATION
// Get search data from server and unserialize
$searchURL = "http://xoap.weather.com/search/search?where=". urlencode(trim($location));
$status = $this->_unserializer ->unserialize($searchURL, true , array ("overrideOptions" => true , "complexType" => "array", "keyAttribute" => "id"));
$search = $this->_unserializer ->getUnserializedData ();
if (!$useFirst && (sizeof($search) > 1 )) {
} elseif ($useFirst || (sizeof($search) == 1 )) {
$searchReturn = key($search);
// {{{ searchLocationByCountry()
* Returns only false, as weather.com offers no country listing via
* Returns the data for the promotional links belonging to the ID
* @return PEAR_Error|array
$status = $this->_checkLocationID ($id);
$linksURL = "http://xoap.weather.com/weather/local/". $id. "?prod=xoap&par=". $this->_partnerID. "&key=". $this->_licenseKey. "&link=xoap";
if ($this->_cacheEnabled && ($links = $this->_cache ->get ($id, "links"))) {
$linksReturn["cache"] = "HIT";
// Same as in the function above...
$status = $this->_parseWeatherData ($id, $linksURL);
$linksReturn["cache"] = "MISS";
$linksReturn["promo"] = array ();
for ($i = 0; $i < sizeof($this->_links ->link); $i++ ) {
$linksReturn["promo"][$i] = array ();
$linksReturn["promo"][$i]["title"] = $this->_links ->link[$i]->t;
// B0rked response (returned is &par=xoap, should be &prod=xoap), fix it
$linksReturn["promo"][$i]["link"] = str_replace("par=", "prod=", $this->_links ->link[$i]->l );
$linksReturn["promo"][$i]["link"] .= "&par=". $this->_partnerID;
* Returns the data for the location belonging to the ID
* @return PEAR_Error|array
$status = $this->_checkLocationID ($id);
$locationReturn = array ();
$locationURL = "http://xoap.weather.com/weather/local/". $id. "?prod=xoap&par=". $this->_partnerID. "&key=". $this->_licenseKey;
if ($this->_cacheEnabled && ($location = $this->_cache ->get ($id, "location"))) {
$this->_location = $location;
$locationReturn["cache"] = "HIT";
// Same as in the function above...
$status = $this->_parseWeatherData ($id, $locationURL);
$locationReturn["cache"] = "MISS";
$locationReturn["name"] = $this->_location ->dnam;
$locationReturn["time"] = date($this->_timeFormat , strtotime($this->_location ->tm ));
$locationReturn["latitude"] = $this->_location ->lat;
$locationReturn["longitude"] = $this->_location ->lon;
$locationReturn["sunrise"] = date($this->_timeFormat , strtotime($this->_location ->sunr ));
$locationReturn["sunset"] = date($this->_timeFormat , strtotime($this->_location ->suns ));
$locationReturn["timezone"] = $this->_location ->zone;
* Returns the weather-data for the supplied location
* @param string $unitsFormat
* @return PEAR_Error|array
$status = $this->_checkLocationID ($id);
$weatherReturn = array ();
$weatherURL = "http://xoap.weather.com/weather/local/". $id. "?cc=*&prod=xoap&par=". $this->_partnerID. "&key=". $this->_licenseKey. "&unit=s";
if ($this->_cacheEnabled && ($weather = $this->_cache ->get ($id, "weather"))) {
$this->_weather = $weather;
$weatherReturn["cache"] = "HIT";
$status = $this->_parseWeatherData ($id, $weatherURL);
$weatherReturn["cache"] = "MISS";
// Some explanation for the next two lines:
// weather.com isn't always supplying the timezone in the update string, but
// uses "Local Time" as reference, which is imho utterly stupid, because it's
// inconsistent. Well, what I do here is check for this string and if I can
// find it, I calculate the difference between the timezone at the location
// and this computers timezone. This amount of seconds is then subtracted from
// the time the update-string has delivered.
$update = str_replace("Local Time", "", $this->_weather ->lsup );
$adjustTZ = ($update == $this->_weather ->lsup ) ? 0 : $this->_location ->zone * 3600 - date("Z");
$weatherReturn["update"] = gmdate(trim($this->_dateFormat. " ". $this->_timeFormat ), strtotime($update) - $adjustTZ);
$weatherReturn["updateRaw"] = $this->_weather ->lsup;
$weatherReturn["station"] = $this->_weather ->obst;
$weatherReturn["temperature"] = $this->convertTemperature($this->_weather ->tmp , "f", $units["temp"]);
$weatherReturn["feltTemperature"] = $this->convertTemperature($this->_weather ->flik , "f", $units["temp"]);
$weatherReturn["condition"] = $this->_weather ->t;
$weatherReturn["conditionIcon"] = $this->_weather ->icon;
$weatherReturn["pressure"] = $this->convertPressure($this->_weather ->bar ->r , "in", $units["pres"]);
$weatherReturn["pressureTrend"] = $this->_weather ->bar ->d;
$weatherReturn["wind"] = $this->convertSpeed($this->_weather ->wind ->s , "mph", $units["wind"]);
$weatherReturn["windGust"] = $this->convertSpeed($this->_weather ->wind ->gust , "mph", $units["wind"]);
$weatherReturn["windDegrees"] = $this->_weather ->wind ->d;
$weatherReturn["windDirection"] = $this->_weather ->wind ->t;
$weatherReturn["humidity"] = $this->_weather ->hmid;
$weatherReturn["visibility"] = $this->convertDistance($this->_weather ->vis , "sm", $units["vis"]);
$weatherReturn["visibility"] = $this->_weather ->vis;
$weatherReturn["uvIndex"] = $this->_weather ->uv ->i;
$weatherReturn["uvText"] = $this->_weather ->uv ->t;
$weatherReturn["dewPoint"] = $this->convertTemperature($this->_weather ->dewp , "f", $units["temp"]);
* Get the forecast for the next days
* @param int $days Values between 1 and 10
* @param string $unitsFormat
* @return PEAR_Error|array
function getForecast($id = "", $days = 2 , $unitsFormat = "")
$status = $this->_checkLocationID ($id);
$forecastReturn = array ();
$forecastURL = "http://xoap.weather.com/weather/local/". $id. "?dayf=10&prod=xoap&par=". $this->_partnerID. "&key=". $this->_licenseKey. "&unit=s";
if ($this->_cacheEnabled && ($forecast = $this->_cache ->get ($id, "forecast"))) {
$this->_forecast = $forecast;
$forecastReturn["cache"] = "HIT";
$status = $this->_parseWeatherData ($id, $forecastURL, $days);
$forecastReturn["cache"] = "MISS";
// Some explanation for the next two lines: (same as above)
// weather.com isn't always supplying the timezone in the update string, but
// uses "Local Time" as reference, which is imho utterly stupid, because it's
// inconsistent. Well, what I do here is check for this string and if I can
// find it, I calculate the difference between the timezone at the location
// and this computers timezone. This amount of seconds is then subtracted from
// the time the update-string has delivered.
$update = str_replace("Local Time", "", $this->_forecast ->lsup );
$adjustTZ = ($update == $this->_forecast ->lsup ) ? 0 : $this->_location ->zone * 3600 - date("Z");
$forecastReturn["update"] = gmdate($this->_dateFormat. " ". $this->_timeFormat , strtotime($update) - $adjustTZ);
$forecastReturn["updateRaw"] = $this->_forecast ->lsup;
$forecastReturn["days"] = array ();
for ($i = 0; $i < $days; $i++ ) {
"temperatureHigh" => $this->convertTemperature($this->_forecast ->day [$i]->hi , "f", $units["temp"]),
"temperatureLow" => $this->convertTemperature($this->_forecast ->day [$i]->low , "f", $units["temp"]),
"sunrise" => date($this->_timeFormat , strtotime($this->_forecast ->day [$i]->sunr )),
"sunset" => date($this->_timeFormat , strtotime($this->_forecast ->day [$i]->suns )),
"condition" => $this->_forecast ->day [$i]->part [0 ]->t ,
"conditionIcon" => $this->_forecast ->day [$i]->part [0 ]->icon ,
"wind" => $this->convertSpeed($this->_forecast ->day [$i]->part [0 ]->wind ->s , "mph", $units["wind"]),
"windGust" => $this->convertSpeed($this->_forecast ->day [$i]->part [0 ]->wind ->gust , "mph", $units["wind"]),
"windDegrees" => $this->_forecast ->day [$i]->part [0 ]->wind ->d ,
"windDirection" => $this->_forecast ->day [$i]->part [0 ]->wind ->t ,
"precipitation" => $this->_forecast ->day [$i]->part [0 ]->ppcp ,
"humidity" => $this->_forecast ->day [$i]->part [0 ]->hmid
"condition" => $this->_forecast ->day [$i]->part [1 ]->t ,
"conditionIcon" => $this->_forecast ->day [$i]->part [1 ]->icon ,
"wind" => $this->convertSpeed($this->_forecast ->day [$i]->part [1 ]->wind ->s , "mph", $units["wind"]),
"windGust" => $this->convertSpeed($this->_forecast ->day [$i]->part [1 ]->wind ->gust , "mph", $units["wind"]),
"windDegrees" => $this->_forecast ->day [$i]->part [1 ]->wind ->d ,
"windDirection" => $this->_forecast ->day [$i]->part [1 ]->wind ->t ,
"precipitation" => $this->_forecast ->day [$i]->part [1 ]->ppcp ,
"humidity" => $this->_forecast ->day [$i]->part [1 ]->hmid
$forecastReturn["days"][] = $day;
Documentation generated on Mon, 11 Mar 2019 13:56:24 -0400 by phpDocumentor 1.4.4. PEAR Logo Copyright © PHP Group 2004.
|