add ProductName to windows pingback
[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         HKEY key;
143         DWORD size = PATH_MAX;
144         char tmp[PATH_MAX+1];
145
146         if (   (ERROR_SUCCESS == RegOpenKeyExA (HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion", 0, KEY_READ, &key))
147             && (ERROR_SUCCESS == RegQueryValueExA (key, "ProductName", 0, NULL, reinterpret_cast<LPBYTE>(tmp), &size))
148                         // or "BuildLab" instead of "ProductName"
149                  )
150         {
151                 string s = Glib::locale_to_utf8 (tmp);
152                 char* query = curl_easy_escape (c, s.c_str(), strlen (s.c_str()));
153                 s = string_compose ("r=%1", query);
154                 url += s;
155                 url += '&';
156                 free (query);
157         } else {
158                 url += "r=&";
159         }
160
161         if (   (ERROR_SUCCESS == RegOpenKeyExA (HKEY_LOCAL_MACHINE, "Hardware\\Description\\System\\CentralProcessor\\0", 0, KEY_READ, &key))
162             && (ERROR_SUCCESS == RegQueryValueExA (key, "Identifier", 0, NULL, reinterpret_cast<LPBYTE>(tmp), &size))
163                         // or "ProcessorNameString" instead of "Identifier"
164                  )
165         {
166                 string s = Glib::locale_to_utf8 (tmp);
167                 char* query = curl_easy_escape (c, s.c_str(), strlen (s.c_str()));
168                 s = string_compose ("m=%1", query);
169                 url += s;
170                 url += '&';
171                 free (query);
172         } else {
173                 url += "m=&";
174         }
175
176         url += "r=&";
177 # if ( defined(__x86_64__) || defined(_M_X64) )
178         url += "s=Windows64";
179 # else
180         url += "s=Windows32";
181 # endif
182
183 #ifndef NDEBUG
184         std::cerr << "Pingback: " << url << std::endl;
185 #endif
186
187 #endif /* PLATFORM_WINDOWS */
188
189         curl_easy_setopt (c, CURLOPT_URL, url.c_str());
190
191         return_str = "";
192
193         if (curl_easy_perform (c) == 0) {
194                 long http_status; 
195
196                 curl_easy_getinfo (c, CURLINFO_RESPONSE_CODE, &http_status);
197
198                 if (http_status != 200) {
199                         std::cerr << "Bad HTTP status" << std::endl;
200                         return 0;
201                 }
202
203                 if ( return_str.length() > 140 ) { // like a tweet :)
204                         std::cerr << "Announcement string is too long (probably behind a proxy)." << std::endl;
205                 } else {
206                         std::cerr << "Announcement is: " << return_str << std::endl;
207                         
208                         //write announcements to local file, even if the
209                         //announcement is empty
210                                 
211                         std::ofstream annc_file (cm->announce_path.c_str());
212                         
213                         if (annc_file) {
214                                 annc_file << return_str;
215                         }
216                 }
217         } else {
218                 std::cerr << "curl failed: " << errbuf << std::endl;
219         }
220
221         curl_easy_cleanup (c);
222         delete cm;
223         return 0;
224 }
225
226 namespace ARDOUR {
227
228 void pingback (const string& version, const string& announce_path) 
229 {
230         ping_call* cm = new ping_call (version, announce_path);
231         pthread_t thread;
232
233         pthread_create_and_store ("pingback", &thread, _pingback, cm);
234 }
235
236 }