diff --git a/ChangeLog b/ChangeLog index d574678e..8827a7c5 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,6 @@ +2009-07-18 + * www.weather.com can now be used as well as a source of weather data + 2009-07-11 * Added support for $desktop, $desktop_number and $desktop_name (sf.net #2040528) diff --git a/configure.ac.in b/configure.ac.in index a56bd6d3..9c9facdc 100644 --- a/configure.ac.in +++ b/configure.ac.in @@ -356,9 +356,10 @@ AC_ARG_ENABLE([weather], # AM_CONDITIONAL(BUILD_WEATHER, test x$want_weather = xyes) if test x$want_weather = xyes; then + PKG_CHECK_MODULES([libxml2], libxml-2.0) PKG_CHECK_MODULES([libcurl], libcurl) - conky_CFLAGS="$conky_CFLAGS $libcurl_CFLAGS" - conky_LIBS="$conky_LIBS $libcurl_LIBS" + conky_CFLAGS="$conky_CFLAGS $libxml2_CFLAGS $libcurl_CFLAGS" + conky_LIBS="$conky_LIBS $libxml2_LIBS $libcurl_LIBS" AC_DEFINE(WEATHER, 1, [Define if you want weather support]) fi @@ -706,6 +707,7 @@ dnl AC_DEFINE(DEFAULTNETDEV, "eth0", [Default networkdevice]) AC_DEFINE(CONFIG_FILE, "$HOME/.conkyrc", [Configfile of the user]) +AC_DEFINE(XOAP_FILE, "$HOME/.xoaprc", [User xoap keys file]) AC_DEFINE(MAX_SPECIALS_DEFAULT, 512, [Default maximum number of special things, e.g. fonts, offsets, aligns, etc.]) AC_DEFINE(MAX_USER_TEXT_DEFAULT, 16384, [Default maximum size of config TEXT buffer, i.e. below TEXT line.]) AC_DEFINE(DEFAULT_TEXT_BUFFER_SIZE, 256, [Default size used for temporary, static text buffers]) diff --git a/doc/variables.xml b/doc/variables.xml index 214a959c..0c69c857 100644 --- a/doc/variables.xml +++ b/doc/variables.xml @@ -3290,23 +3290,40 @@ - + Download, parse and display METAR data. - For the 'URI', right now only - http://weather.noaa.gov/pub/data/observations/metar/stations/ - is supported. Other sources might be supported in the future. + For the 'URI', there are two possibilities: + + + http://weather.noaa.gov/pub/data/observations/metar/stations/ + + + http://xoap.weather.com/weather/local/ + + + The first one is free to use but the second requires you to + register and obtain your partner ID and license key. + These two must be written, separated by a space, into a file + called .xoaprc which needs to be placed into your home directory. - 'icao' must be a valid icao for the required location + 'locID' must be a valid location identifier for the required + uri. For the NOAA site this must be a valid ICAO (see for instance https://pilotweb.nas.faa.gov/qryhtml/icao/). + For the weather.com site this must be a valid location ID + (see for instance http://aspnetresources.com/tools/locid.aspx). 'data_type' must be one of the following: last_update - + temperature @@ -3319,7 +3336,7 @@ - pressurer + pressure @@ -3341,7 +3358,9 @@ weather + etc.). This is not used if you are querying the + weather.com site since this data is aggregated + into the cloud_cover one 'delay_in_minutes' (optional, default 30) cannot be diff --git a/src/conky.c b/src/conky.c index 9809943d..84edfb3e 100644 --- a/src/conky.c +++ b/src/conky.c @@ -318,6 +318,11 @@ static int cpu_avg_samples, net_avg_samples, diskio_avg_samples; char *overwrite_file = NULL; FILE *overwrite_fpointer = NULL; char *append_file = NULL; FILE *append_fpointer = NULL; +/* xoap suffix for weather from weather.com */ +#ifdef WEATHER +static char *xoap = NULL; +#endif /* WEATHER */ + #ifdef X11 static int show_graph_scale; @@ -2820,35 +2825,53 @@ static struct text_object *construct_text_object(const char *s, END OBJ_THREAD(weather, 0) if (arg) { int argc, interval; - char *icao = (char *) malloc(5 * sizeof(char)); + char *locID = (char *) malloc(9 * sizeof(char)); char *uri = (char *) malloc(128 * sizeof(char)); char *data_type = (char *) malloc(32 * sizeof(char)); char *tmp_p; - argc = sscanf(arg, "%119s %4s %31s %d", uri, icao, data_type, &interval); + argc = sscanf(arg, "%119s %8s %31s %d", uri, locID, data_type, &interval); - //icao MUST BE upper-case - tmp_p = icao; + //locID MUST BE upper-case + tmp_p = locID; while (*tmp_p) { - *tmp_p = toupper(*tmp_p); - tmp_p++; + *tmp_p = toupper(*tmp_p); + tmp_p++; } + //Construct complete uri + if (strstr(uri, "xoap.weather.com")) { + if(xoap != NULL) { + strcat(uri, locID); + strcat(uri, xoap); + } else { + free(uri); + uri = NULL; + } + } else if (strstr(uri, "weather.noaa.gov")) { + strcat(uri, locID); + strcat(uri, ".TXT"); + } else if (!strstr(uri, "localhost") && !strstr(uri, "127.0.0.1")) { + CRIT_ERR(obj, free_at_crash, \ + "could not recognize the weather uri"); + } - strcat(uri, icao); - strcat(uri, ".TXT"); obj->data.weather.uri = uri; - obj->data.weather.data_type = data_type; - // The data retrieval interval is limited to half an hour + //Limit the data retrieval interval to half hour min if (interval < 30) { interval = 30; } - obj->data.weather.interval = interval * 60; // convert to seconds - free(icao); + + //Convert to seconds + obj->data.weather.interval = interval * 60; + free(locID); + + DBGP("weather: fetching %s from %s every %d seconds", \ + data_type, uri, obj->data.weather.interval); } else { - CRIT_ERR(obj, free_at_crash, "weather needs arguments: [interval in minutes]"); + CRIT_ERR(obj, free_at_crash, "weather needs arguments: [interval in minutes]"); } #endif #ifdef HAVE_LUA @@ -4647,7 +4670,11 @@ static void generate_text_internal(char *p, int p_max_size, #endif #ifdef WEATHER OBJ(weather) { - process_weather_info(p, p_max_size, obj->data.weather.uri, obj->data.weather.data_type, obj->data.weather.interval); + if( obj->data.weather.uri != NULL ) { + process_weather_info(p, p_max_size, obj->data.weather.uri, obj->data.weather.data_type, obj->data.weather.interval); + } else { + strncpy(p, "invalid xoap keys file", p_max_size); + } } #endif #ifdef HAVE_LUA @@ -8885,6 +8912,42 @@ static void load_config_file_x11(const char *f) } #endif /* X11 */ +#ifdef WEATHER +/* + * TODO: make the xoap keys file readable from the config file + * make the keys directly readable from the config file + * make the xoap keys file giveable as a command line option + */ +static void load_xoap_keys(void) +{ + FILE *fp; + char *par = (char *) malloc(11 * sizeof(char)); + char *key = (char *) malloc(17 * sizeof(char)); + + xoap = (char *) malloc(64 * sizeof(char)); + to_real_path(xoap, XOAP_FILE); + fp = fopen(xoap, "r"); + if (fp != NULL) { + if( fscanf(fp, "%10s %16s", par, key) == 2 ) { + strcpy(xoap, "?cc=*&link=xoap&prod=xoap&par="); + strcat(xoap, par); + strcat(xoap, "&key="); + strcat(xoap, key); + strcat(xoap, "&unit=m"); + } else { + free(xoap); + xoap = NULL; + } + fclose(fp); + } else { + free(xoap); + xoap = NULL; + } + free(par); + free(key); +} +#endif /* WEATHER */ + static void print_help(const char *prog_name) { printf("Usage: %s [OPTION]...\n" PACKAGE_NAME" is a system monitor that renders text on desktop or to own transparent\n" @@ -9268,6 +9331,12 @@ int main(int argc, char **argv) #endif /* ! CONF_OUTPUT */ } } + +#ifdef WEATHER + /* Load xoap keys, if existing */ + load_xoap_keys(); +#endif /* WEATHER */ + #ifdef HAVE_SYS_INOTIFY_H inotify_fd = inotify_init(); #endif /* HAVE_SYS_INOTIFY_H */ diff --git a/src/weather.c b/src/weather.c index 361bdfb8..6d290275 100644 --- a/src/weather.c +++ b/src/weather.c @@ -21,6 +21,11 @@ * */ +/* + * TODO: Add weather forecast info from weather.com + * + */ + #include "conky.h" #include "logging.h" #include "weather.h" @@ -33,6 +38,7 @@ #include #include #include +#include /* Possible sky conditions */ #define NUM_CC_CODES 6 @@ -120,6 +126,75 @@ int rel_humidity(int dew_point, int air) { #endif /* MATH */ } +//TODO: Lets get rid of the recursion +static void parse_cc(PWEATHER *res, xmlNodePtr cc) +{ + + xmlNodePtr cur = NULL; + + for (cur = cc; cur; cur = cur->next) { + if (cur->type == XML_ELEMENT_NODE) { + if (!xmlStrcmp(cur->name, (const xmlChar *) "lsup")) { + strncpy(res->lastupd, (char *)cur->children->content, 31); + } else if (!xmlStrcmp(cur->name, (const xmlChar *) "tmp")) { + res->temp = atoi((char *)cur->children->content); + } else if (!xmlStrcmp(cur->name, (const xmlChar *) "t")) { + if(res->xoap_t[0] == '\0') { + strncpy(res->xoap_t, (char *)cur->children->content, 31); + } + } else if (!xmlStrcmp(cur->name, (const xmlChar *) "r")) { + res->bar = atoi((char *)cur->children->content); + } else if (!xmlStrcmp(cur->name, (const xmlChar *) "s")) { + res->wind_s = atoi((char *)cur->children->content); + } else if (!xmlStrcmp(cur->name, (const xmlChar *) "d")) { + if (isdigit((char)cur->children->content[0])) { + res->wind_d = atoi((char *)cur->children->content); + } + } else if (!xmlStrcmp(cur->name, (const xmlChar *) "hmid")) { + res->hmid = atoi((char *)cur->children->content); + } + } + parse_cc(res, cur->children); + } + return; +} + +static void parse_weather_xml(PWEATHER *res, const char *data) +{ + xmlDocPtr doc; + xmlNodePtr cur; + + if (!(doc = xmlReadMemory(data, strlen(data), "", NULL, 0))) { + ERR("weather: can't read xml data"); + return; + } + + cur = xmlDocGetRootElement(doc); + + while(cur) { + if (cur->type == XML_ELEMENT_NODE) { + if (!xmlStrcmp(cur->name, (const xmlChar *) "weather")) { + cur = cur->children; + while (cur != NULL) { + if (cur->type == XML_ELEMENT_NODE) { + if (!xmlStrcmp(cur->name, (const xmlChar *) "cc")) { + parse_cc(res, cur->children); + xmlFreeDoc(doc); + return; + } + } + cur = cur->next; + } + } + } + cur = cur->next; + } + + ERR("weather: incorrect xml data"); + xmlFreeDoc(doc); + return ; +} + /* * Horrible hack to avoid using regexes * @@ -337,6 +412,7 @@ static inline void parse_token(PWEATHER *res, char *token) { //First 3 digits are wind direction strncpy(s_tmp, token, 3); + s_tmp[3]='\0'; res->wind_d=atoi(s_tmp); //4th and 5th digit are wind speed in knots (convert to km/hr) @@ -393,6 +469,7 @@ static inline void parse_token(PWEATHER *res, char *token) { //First 3 digits are wind direction strncpy(s_tmp, token, 3); + s_tmp[3]='\0'; res->wind_d=atoi(s_tmp); //4th and 5th digit are wind speed in m/s (convert to km/hr) @@ -410,38 +487,44 @@ static inline void parse_token(PWEATHER *res, char *token) { static void parse_weather(PWEATHER *res, const char *data) { - char s_tmp[256]; - const char delim[] = " "; - + //Reset results memset(res, 0, sizeof(PWEATHER)); - //Divide time stamp and metar data - if (sscanf(data, "%[^'\n']\n%[^'\n']", res->lastupd, s_tmp) == 2) { + //Check if it is an xml file + if ( strncmp(data, "lastupd, s_tmp) == 2) { - if ((strtok_r(s_tmp, delim, &p_save)) != NULL) { + //Process all tokens + char *p_tok = NULL; + char *p_save = NULL; - //Jump first token, must be icao - p_tok = strtok_r(NULL, delim, &p_save); + if ((strtok_r(s_tmp, delim, &p_save)) != NULL) { - do { + //Jump first token, must be icao + p_tok = strtok_r(NULL, delim, &p_save); - parse_token(res, p_tok); - p_tok = strtok_r(NULL, delim, &p_save); + do { - } while (p_tok != NULL); - } - return; - } - else { - return; + parse_token(res, p_tok); + p_tok = strtok_r(NULL, delim, &p_save); + + } while (p_tok != NULL); + } + return; + } + else { + return; + } } } - void fetch_weather_info(location *curloc) { CURL *curl = NULL; @@ -468,7 +551,7 @@ void fetch_weather_info(location *curloc) timed_thread_unlock(curloc->p_timed_thread); free(chunk.memory); } else { - ERR("No data from server"); + ERR("weather: no data from server"); } curl_easy_cleanup(curl); @@ -485,12 +568,12 @@ void init_thread(location *curloc, int interval) timed_thread_create(&weather_thread, (void *)curloc, interval * 1000000); if (!curloc->p_timed_thread) { - ERR("Error creating weather timed thread"); + ERR("weather: error creating timed thread"); } timed_thread_register(curloc->p_timed_thread, &curloc->p_timed_thread); if (timed_thread_run(curloc->p_timed_thread)) { - ERR("Error running weather timed thread"); + ERR("weather: error running timed thread"); } } @@ -507,14 +590,15 @@ void process_weather_info(char *p, int p_max_size, char *uri, char *data_type, i location *curloc = find_location(uri); if (!curloc->p_timed_thread) init_thread(curloc, interval); - timed_thread_lock(curloc->p_timed_thread); if (strcmp(data_type, "last_update") == EQUAL) { strncpy(p, curloc->data.lastupd, p_max_size); } else if (strcmp(data_type, "temperature") == EQUAL) { temp_print(p, p_max_size, curloc->data.temp, TEMP_CELSIUS); } else if (strcmp(data_type, "cloud_cover") == EQUAL) { - if (curloc->data.cc == 0) { + if (curloc->data.xoap_t[0] != '\0') { + strncpy(p, curloc->data.xoap_t, p_max_size); + } else if (curloc->data.cc == 0) { strncpy(p, "", p_max_size); } else if (curloc->data.cc < 3) { strncpy(p, "clear", p_max_size); @@ -575,6 +659,7 @@ void process_weather_info(char *p, int p_max_size, char *uri, char *data_type, i } else if (strcmp(data_type, "weather") == EQUAL) { strncpy(p, wc[curloc->data.wc], p_max_size); } + timed_thread_unlock(curloc->p_timed_thread); } diff --git a/src/weather.h b/src/weather.h index ff2d6873..3c9b9d4a 100644 --- a/src/weather.h +++ b/src/weather.h @@ -30,7 +30,8 @@ /* WEATHER data */ typedef struct PWEATHER_ { - char lastupd[17]; + char lastupd[32]; + char xoap_t[32]; int temp; int dew; int cc; @@ -39,6 +40,16 @@ typedef struct PWEATHER_ { int wind_d; int hmid; int wc; + /* + * TODO: + * Is it worth investigating about using icons from weather.com? + * We could use them for data from noaa as well. + * They can display nicely with cimlib_add_image (with appropriate + * #ifdefs on imlib2 and x11), and an additional input argoment for position. + + char icon[3]; + + */ } PWEATHER; /* Prototypes */