another windows pingback fix
[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 #ifdef PLATFORM_WINDOWS
27 #include <windows.h>
28 #include <glibmm.h>
29 #else
30 #include <sys/utsname.h>
31 #endif
32
33 #include <curl/curl.h>
34
35 #include <glibmm/miscutils.h>
36
37 #include "pbd/compose.h"
38 #include "pbd/pthread_utils.h"
39
40 #include "ardour/filesystem_paths.h"
41 #include "ardour/rc_configuration.h"
42
43 #include "pingback.h"
44
45 using std::string;
46 using namespace ARDOUR;
47
48 static size_t
49 curl_write_data (char *bufptr, size_t size, size_t nitems, void *ptr)
50 {
51         /* we know its a string */
52
53         string* sptr = (string*) ptr;
54
55         for (size_t i = 0; i < nitems; ++i) {
56                 for (size_t n = 0; n < size; ++n) {
57                         if (*bufptr == '\n') {
58                                 break;
59                         }
60
61                         (*sptr) += *bufptr++;
62                 }
63         }
64
65         return size * nitems;
66 }
67
68 struct ping_call {
69     std::string version;
70     std::string announce_path;
71
72     ping_call (const std::string& v, const std::string& a)
73             : version (v), announce_path (a) {}
74 };
75
76 static void*
77 _pingback (void *arg)
78 {
79         ping_call* cm = static_cast<ping_call*> (arg);
80         CURL* c;
81         string return_str;
82         //initialize curl
83
84         curl_global_init (CURL_GLOBAL_NOTHING);
85         c = curl_easy_init ();
86         
87         curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, curl_write_data); 
88         curl_easy_setopt (c, CURLOPT_WRITEDATA, &return_str); 
89         char errbuf[CURL_ERROR_SIZE];
90         curl_easy_setopt (c, CURLOPT_ERRORBUFFER, errbuf); 
91
92         string url;
93
94 #ifdef __APPLE__
95         url = Config->get_osx_pingback_url ();
96 #elif defined PLATFORM_WINDOWS
97         url = Config->get_windows_pingback_url ();
98 #else
99         url = Config->get_linux_pingback_url ();
100 #endif
101
102         if (url.compare (0, 4, "http") != 0) {
103                 delete cm;
104                 return 0;
105         }
106
107         char* v = curl_easy_escape (c, cm->version.c_str(), cm->version.length());
108         url += v;
109         url += '?';
110         free (v);
111
112 #ifndef PLATFORM_WINDOWS
113         struct utsname utb;
114
115         if (uname (&utb)) {
116                 delete cm;
117                 return 0;
118         }
119
120         //string uts = string_compose ("%1 %2 %3 %4", utb.sysname, utb.release, utb.version, utb.machine);
121         string s;
122         char* query;
123
124         query = curl_easy_escape (c, utb.sysname, strlen (utb.sysname));
125         s = string_compose ("s=%1", query);
126         url += s;
127         url += '&';
128         free (query);
129
130         query = curl_easy_escape (c, utb.release, strlen (utb.release));
131         s = string_compose ("r=%1", query);
132         url += s;
133         url += '&';
134         free (query);
135
136         query = curl_easy_escape (c, utb.machine, strlen (utb.machine));
137         s = string_compose ("m=%1", query);
138         url += s;
139         free (query);
140 #else
141         // this is hilarious: https://msdn.microsoft.com/en-us/library/windows/desktop/ms724429%28v=vs.85%29.aspx
142         url += "r=&";
143
144         HKEY key;
145         DWORD size = PATH_MAX;
146         char tmp[PATH_MAX+1];
147         if (   (ERROR_SUCCESS == RegOpenKeyExA (HKEY_LOCAL_MACHINE, "Hardware\\Description\\System\\CentralProcessor\\0", 0, KEY_READ, &key))
148             && (ERROR_SUCCESS == RegQueryValueExA (key, "Identifier", 0, NULL, reinterpret_cast<LPBYTE>(tmp), &size))
149                         // or "ProcessorNameString"
150                  )
151         {
152                 string s = Glib::locale_to_utf8 (tmp);
153                 char* query = curl_easy_escape (c, s.c_str(), strlen (s.c_str()));
154                 s = string_compose ("m=%1", query);
155                 url += s;
156                 url += '&';
157                 free (query);
158         } else {
159                 url += "m=&";
160         }
161
162         url += "r=&";
163 # if ( defined(__x86_64__) || defined(_M_X64) )
164         url += "s=Windows64";
165 # else
166         url += "s=Windows32";
167 # endif
168
169 #ifndef NDEBUG
170         std::cerr << "Pingback: " << url << std::endl;
171 #endif
172
173 #endif /* PLATFORM_WINDOWS */
174
175         curl_easy_setopt (c, CURLOPT_URL, url.c_str());
176
177         return_str = "";
178
179         if (curl_easy_perform (c) == 0) {
180                 long http_status; 
181
182                 curl_easy_getinfo (c, CURLINFO_RESPONSE_CODE, &http_status);
183
184                 if (http_status != 200) {
185                         std::cerr << "Bad HTTP status" << std::endl;
186                         return 0;
187                 }
188
189                 if ( return_str.length() > 140 ) { // like a tweet :)
190                         std::cerr << "Announcement string is too long (probably behind a proxy)." << std::endl;
191                 } else {
192                         std::cerr << "Announcement is: " << return_str << std::endl;
193                         
194                         //write announcements to local file, even if the
195                         //announcement is empty
196                                 
197                         std::ofstream annc_file (cm->announce_path.c_str());
198                         
199                         if (annc_file) {
200                                 annc_file << return_str;
201                         }
202                 }
203         } else {
204                 std::cerr << "curl failed: " << errbuf << std::endl;
205         }
206
207         curl_easy_cleanup (c);
208         delete cm;
209         return 0;
210 }
211
212 namespace ARDOUR {
213
214 void pingback (const string& version, const string& announce_path) 
215 {
216         ping_call* cm = new ping_call (version, announce_path);
217         pthread_t thread;
218
219         pthread_create_and_store ("pingback", &thread, _pingback, cm);
220 }
221
222 }