Oops, add missing files.
authorBrenden Matthews <brenden@rty.ca>
Mon, 20 Jul 2009 15:21:33 +0000 (09:21 -0600)
committerBrenden Matthews <brenden@rty.ca>
Mon, 20 Jul 2009 15:21:33 +0000 (09:21 -0600)
src/ccurl_thread.c [new file with mode: 0644]
src/ccurl_thread.h [new file with mode: 0644]

diff --git a/src/ccurl_thread.c b/src/ccurl_thread.c
new file mode 100644 (file)
index 0000000..d9d9717
--- /dev/null
@@ -0,0 +1,199 @@
+/*
+ * Conky, a system monitor, based on torsmo
+ *
+ * Please see COPYING for details
+ *
+ * Copyright (c) 2005-2009 Brenden Matthews, Philip Kovacs, et. al.
+ *     (see AUTHORS)
+ * All rights reserved.
+ *
+ * This program 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 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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.
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+
+#include "conky.h"
+#include "logging.h"
+#include "ccurl_thread.h"
+
+#ifdef DEBUG
+#include <assert.h>
+#endif /* DEBUG */
+
+#include <curl/curl.h>
+#include <curl/types.h>
+#include <curl/easy.h>
+
+typedef struct _ccurl_memory_t {
+       char *memory;
+       size_t size;
+} ccurl_memory_t;
+
+ccurl_location_t *ccurl_find_location(ccurl_location_t **locations_head, char *uri)
+{
+       ccurl_location_t *tail = *locations_head;
+       ccurl_location_t *new = 0;
+       while (tail) {
+               if (tail->uri &&
+                               strcmp(tail->uri, uri) == EQUAL) {
+                       return tail;
+               }
+               tail = tail->next;
+       }
+       if (!tail) { /* new location!!!!!!! */
+               new = malloc(sizeof(ccurl_location_t));
+               memset(new, 0, sizeof(ccurl_location_t));
+               new->uri = strndup(uri, text_buffer_size);
+               tail = *locations_head;
+               while (tail && tail->next) {
+                       tail = tail->next;
+               }
+               if (!tail) {
+                       /* omg the first one!!!!!!! */
+                       *locations_head = new;
+               } else {
+                       tail->next = new;
+               }
+       }
+       return new;
+}
+
+void ccurl_free_locations(ccurl_location_t **locations_head)
+{
+       ccurl_location_t *tail = *locations_head;
+       ccurl_location_t *last = 0;
+
+       while (tail) {
+               if (tail->uri) free(tail->uri);
+               if (tail->result) free(tail->result);
+               last = tail;
+               tail = tail->next;
+               free(last);
+       }
+       *locations_head = 0;
+}
+
+size_t ccurl_write_memory_callback(void *ptr, size_t size, size_t nmemb, void *data)
+{
+       size_t realsize = size * nmemb;
+       ccurl_memory_t *mem = (ccurl_memory_t*)data;
+
+       mem->memory = (char *) realloc(mem->memory, mem->size + realsize + 1);
+       if (mem->memory) {
+               memcpy(&(mem->memory[mem->size]), ptr, realsize);
+               mem->size += realsize;
+               mem->memory[mem->size] = 0;
+       }
+       return realsize;
+}
+
+
+
+void ccurl_fetch_data(ccurl_location_t *curloc)
+{
+       CURL *curl = NULL;
+       CURLcode res;
+
+       // curl temps
+       ccurl_memory_t chunk;
+
+       chunk.memory = NULL;
+       chunk.size = 0;
+
+       curl = curl_easy_init();
+       if (curl) {
+               curl_easy_setopt(curl, CURLOPT_URL, curloc->uri);
+               curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 1);
+               curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, ccurl_write_memory_callback);
+               curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *) &chunk);
+               curl_easy_setopt(curl, CURLOPT_USERAGENT, "conky-curl/1.0");
+
+               res = curl_easy_perform(curl);
+               if (res == CURLE_OK && chunk.size) {
+                       timed_thread_lock(curloc->p_timed_thread);
+                       (*curloc->process_function)(curloc->result, chunk.memory);
+                       timed_thread_unlock(curloc->p_timed_thread);
+                       free(chunk.memory);
+               } else {
+                       ERR("weather: no data from server");
+               }
+
+               curl_easy_cleanup(curl);
+       }
+
+       return;
+}
+
+void *ccurl_thread(void *) __attribute__((noreturn));
+
+void ccurl_init_thread(ccurl_location_t *curloc, int interval)
+{
+#ifdef DEBUG
+       assert(curloc->result);
+#endif /* DEBUG */
+       curloc->p_timed_thread =
+               timed_thread_create(&ccurl_thread,
+                               (void *)curloc, interval * 1000000);
+
+       if (!curloc->p_timed_thread) {
+               ERR("curl thread: error creating timed thread");
+       }
+       timed_thread_register(curloc->p_timed_thread,
+                       &curloc->p_timed_thread);
+       if (timed_thread_run(curloc->p_timed_thread)) {
+               ERR("curl thread: error running timed thread");
+       }
+}
+
+void *ccurl_thread(void *arg)
+{
+       ccurl_location_t *curloc = (ccurl_location_t*)arg;
+
+       while (1) {
+               ccurl_fetch_data(curloc);
+               if (timed_thread_test(curloc->p_timed_thread, 0)) {
+                       timed_thread_exit(curloc->p_timed_thread);
+               }
+       }
+       /* never reached */
+}
+
+static ccurl_location_t *ccurl_locations_head = 0;
+
+void ccurl_free_info(void)
+{
+       ccurl_free_locations(&ccurl_locations_head);
+}
+
+void ccurl_parse_data(void *result, const char *data)
+{
+       strncpy(result, data, max_user_text);
+}
+
+void ccurl_process_info(char *p, int p_max_size, char *uri, int interval)
+{
+       ccurl_location_t *curloc = ccurl_find_location(&ccurl_locations_head, uri);
+       if (!curloc->p_timed_thread) {
+               curloc->result = malloc(max_user_text);
+               memset(curloc->result, 0, max_user_text);
+               curloc->process_function = &ccurl_parse_data;
+               ccurl_init_thread(curloc, interval);
+               if (!curloc->p_timed_thread) {
+                       ERR("error setting up weather thread");
+               }
+       }
+
+       timed_thread_lock(curloc->p_timed_thread);
+       strncpy(p, curloc->result, p_max_size);
+       timed_thread_unlock(curloc->p_timed_thread);
+}
+
diff --git a/src/ccurl_thread.h b/src/ccurl_thread.h
new file mode 100644 (file)
index 0000000..18d7b27
--- /dev/null
@@ -0,0 +1,49 @@
+/*
+ * Conky, a system monitor, based on torsmo
+ *
+ * Please see COPYING for details
+ *
+ * Copyright (c) 2005-2009 Brenden Matthews, Philip Kovacs, et. al.
+ *     (see AUTHORS)
+ * All rights reserved.
+ *
+ * This program 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 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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.
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+
+#ifndef _CURL_THREAD_H_
+#define _CURL_THREAD_H_
+
+typedef struct _ccurl_location_t {
+       char *uri; /* uri of location */
+       void *result; /* a pointer to some arbitrary data, will be freed by ccurl_free_info() if non-null */
+       timed_thread *p_timed_thread; /* internal thread pointer */
+       void (*process_function)(void *, const char *); /* function to call when data is ready to be processed, the first argument will be the result pointer, and the second argument is an internal buffer that shouldn't be mangled */
+       struct _ccurl_location_t *next; /* internal list pointer */
+} ccurl_location_t;
+
+/* find an existing location for the uri specified */
+ccurl_location_t *ccurl_find_location(ccurl_location_t **locations_head, char *uri);
+/* free all locations (as well as location->uri and location->result if
+ * non-null) */
+void ccurl_free_locations(ccurl_location_t **locations_head);
+/* initiates a thread at the location specified using the interval in seconds */
+void ccurl_init_thread(ccurl_location_t *curloc, int interval);
+
+/* for $curl, free internal list pointer */
+void ccurl_free_info(void);
+/* runs instance of $curl */
+void ccurl_process_info(char *p, int p_max_size, char *uri, int interval);
+
+#endif /* _CURL_THREAD_H_ */
+