remove pingback url info msg
[ardour.git] / gtk2_ardour / pingback.cc
1 /*
2     Copyright (C) 2012 Paul Davis 
3     Inspired by code from Ben Loftis @ Harrison Consoles
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18
19 */
20
21 #include <string>
22 #include <iostream>
23 #include <fstream>
24 #include <cstring>
25
26 #include <sys/utsname.h>
27 #include <curl/curl.h>
28
29 #include <glibmm/miscutils.h>
30
31 #include "pbd/compose.h"
32 #include "pbd/pthread_utils.h"
33 #include "ardour/filesystem_paths.h"
34
35 #include "pingback.h"
36
37 using std::string;
38
39 static size_t
40 curl_write_data (char *bufptr, size_t size, size_t nitems, void *ptr)
41 {
42         /* we know its a string */
43
44         string* sptr = (string*) ptr;
45
46         for (size_t i = 0; i < nitems; ++i) {
47                 for (size_t n = 0; n < size; ++n) {
48                         if (*bufptr == '\n') {
49                                 break;
50                         }
51
52                         (*sptr) += *bufptr++;
53                 }
54         }
55
56         return size * nitems;
57 }
58
59 struct ping_call {
60     std::string version;
61     std::string announce_path;
62
63     ping_call (const std::string& v, const std::string& a)
64             : version (v), announce_path (a) {}
65 };
66
67 static void*
68 _pingback (void *arg)
69 {
70         ping_call* cm = static_cast<ping_call*> (arg);
71         CURL* c;
72         struct utsname utb;
73         string return_str;
74
75         if (uname (&utb)) {
76                 return 0;
77         }
78
79         //initialize curl
80
81         curl_global_init (CURL_GLOBAL_NOTHING);
82         c = curl_easy_init ();
83         
84         curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, curl_write_data); 
85         curl_easy_setopt (c, CURLOPT_WRITEDATA, &return_str); 
86         char errbuf[CURL_ERROR_SIZE];
87         curl_easy_setopt (c, CURLOPT_ERRORBUFFER, errbuf); 
88         /* we really would prefer to be able to authenticate the certificate
89            but this has issues that right now (march 2013), i don't understand.
90         */
91         curl_easy_setopt (c, CURLOPT_SSL_VERIFYPEER, 0);
92
93         //get announcements from our server
94         std::cerr << "Checking for Announcements from ardour.org  ...\n";
95
96         string url;
97
98 #ifdef __APPLE__
99         url = "https://community.ardour.org/pingback/osx/";
100 #else
101         url = "https://community.ardour.org/pingback/linux/";
102 #endif
103
104         char* v = curl_easy_escape (c, cm->version.c_str(), cm->version.length());
105         url += v;
106         url += '?';
107         free (v);
108
109         string uts = string_compose ("%1 %2 %3 %4", utb.sysname, utb.release, utb.version, utb.machine);
110         string s;
111         char* query;
112
113         query = curl_easy_escape (c, utb.sysname, strlen (utb.sysname));
114         s = string_compose ("s=%1", query);
115         url += s;
116         url += '&';
117         free (query);
118
119         query = curl_easy_escape (c, utb.release, strlen (utb.release));
120         s = string_compose ("r=%1", query);
121         url += s;
122         url += '&';
123         free (query);
124
125         query = curl_easy_escape (c, utb.machine, strlen (utb.machine));
126         s = string_compose ("m=%1", query);
127         url += s;
128         free (query);
129
130         curl_easy_setopt (c, CURLOPT_URL, url.c_str());
131
132         return_str = "";
133
134         if (curl_easy_perform (c) == 0) {
135                 int http_status; 
136
137                 curl_easy_getinfo (c, CURLINFO_RESPONSE_CODE, &http_status);
138
139                 if (http_status != 200) {
140                         std::cerr << "Bad HTTP status" << std::endl;
141                         return 0;
142                 }
143
144                 if ( return_str.length() > 140 ) { // like a tweet :)
145                         std::cerr << "Announcement string is too long (probably behind a proxy)." << std::endl;
146                 } else {
147                         std::cerr << "Announcement is: " << return_str << std::endl;
148                         
149                         //write announcements to local file, even if the
150                         //announcement is empty
151                                 
152                         std::ofstream annc_file (cm->announce_path.c_str());
153                         
154                         if (annc_file) {
155                                 annc_file << return_str;
156                         }
157                 }
158         } else {
159                 std::cerr << "curl failed: " << errbuf << std::endl;
160         }
161
162         curl_easy_cleanup (c);
163         delete cm;
164
165         return 0;
166 }
167
168 namespace ARDOUR {
169
170 void pingback (const string& version, const string& announce_path) 
171 {
172         ping_call* cm = new ping_call (version, announce_path);
173         pthread_t thread;
174
175         pthread_create_and_store ("pingback", &thread, _pingback, cm);
176 }
177
178 }