chiark / gitweb /
eb7536d6fac4bcc3014f2b7c35880602febc3820
[ypp-sc-tools.main.git] / pctb / convert.c
1 /*
2  * ypp-commodities main program: argument parsing etc.
3  */
4 /*
5  *  This is part of ypp-sc-tools, a set of third-party tools for assisting
6  *  players of Yohoho Puzzle Pirates.
7  * 
8  *  Copyright (C) 2009 Ian Jackson <ijackson@chiark.greenend.org.uk>
9  * 
10  *  This program is free software: you can redistribute it and/or modify
11  *  it under the terms of the GNU General Public License as published by
12  *  the Free Software Foundation, either version 3 of the License, or
13  *  (at your option) any later version.
14  * 
15  *  This program is distributed in the hope that it will be useful,
16  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
17  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  *  GNU General Public License for more details.
19  * 
20  *  You should have received a copy of the GNU General Public License
21  *  along with this program.  If not, see <http://www.gnu.org/licenses/>.
22  * 
23  *  Yohoho and Puzzle Pirates are probably trademarks of Three Rings and
24  *  are used without permission.  This program is not endorsed or
25  *  sponsored by Three Rings.
26  */
27
28 #include "convert.h"
29
30 void debug_flush(void) {
31   sysassert(!ferror(debug));
32   sysassert(!fflush(debug));
33 }
34
35 const char *get_vardir(void) { return "."; }
36
37 static enum {
38   mf_findwindow=      0001,
39   mf_screenshot=      0010,
40   mf_readscreenshot=  0020,
41   mf_analyse=         0100,
42   
43   mode_findwindow=    0001,
44   mode_screenshot=    0011,
45   mode_analyse=       0120,
46
47   mode_all=           0111,
48 } o_mode= mode_all;
49
50 static char *o_screenshot_fn;
51 static int o_single_page, o_quiet;
52 static const char *o_outputmode= "upload";
53
54 FILE *screenshot_file;
55
56
57 static void vbadusage(const char *fmt, va_list) FMT(1,0) NORET;
58 static void vbadusage(const char *fmt, va_list al) {
59   fputs("bad usage: ",stderr);
60   vfprintf(stderr,fmt,al);
61   fputc('\n',stderr);
62   exit(12);
63 }
64 DEFINE_VWRAPPERF(static, badusage, NORET);
65
66 static void open_screenshot_file(const char *mode) {
67   screenshot_file= fopen(o_screenshot_fn, mode);
68   if (!screenshot_file)
69     fatal("could not open screenshots file `%s': %s",
70           o_screenshot_fn, strerror(errno));
71 }
72
73 static void run_analysis(void) {
74   FILE *tf;
75
76   sysassert( tf= tmpfile() );
77   progress("running recognition...");
78   analyse(tf);
79
80   sysassert( fseek(tf,0,SEEK_SET) == 0);
81
82   progress_log("processing results (--%s)...", o_outputmode);
83   pid_t processor;
84   sysassert( (processor= fork()) != -1 );
85
86   if (!processor) {
87     sysassert( dup2(fileno(tf),0) ==0 );
88     execlp("./yppsc-commod-processor", "yppsc-commod-processor",
89            o_outputmode, (char*)0);
90     sysassert(!"execlp commod-processor failed");
91   }
92
93   waitpid_check_exitstatus(processor, "output processor/uploader");
94   fclose(tf);
95   progress_log("all complete.");
96 }  
97
98 int main(int argc, char **argv) {
99   const char *arg;
100   int r;
101
102 #define ARGVAL  ((*++argv) ? *argv : \
103                  (badusage("missing value for option %s",arg),(char*)0))
104
105   while ((arg=*++argv)) {
106     if (!strcmp(arg,"--find-window-only"))
107       o_mode= mode_findwindow;
108     else if (!strcmp(arg,"--screenshot-only"))
109       o_mode= mode_screenshot;
110     else if (!strcmp(arg,"--analyse-only") ||
111              !strcmp(arg,"--same"))
112       o_mode= mode_analyse;
113     else if (!strcmp(arg,"--everything"))
114       o_mode= mode_all;
115     else if (!strcmp(arg,"--single-page"))
116       o_single_page= 1;
117     else if (!strcmp(arg,"--quiet"))
118       o_quiet= 1;
119     else if (!strcmp(arg,"--raw-tsv"))
120       o_outputmode= 0;
121     else if (!strcmp(arg,"--upload") ||
122              !strcmp(arg,"--arbitrage") ||
123              !strcmp(arg,"--tsv") ||
124              !strcmp(arg,"--best-prices"))
125       o_outputmode= arg+2;
126     else if (!strcmp(arg,"--screenshot-file"))
127       o_screenshot_fn= ARGVAL;
128 #define DF(f)                                   \
129     else if (!strcmp(arg,"-D" #f))              \
130       debug_flags |= dbg_##f;
131     DEBUG_FLAG_LIST
132 #undef DF
133     else if (!strcmp(arg,"--window-id")) {
134       char *ep;
135       unsigned long windowid= strtoul(ARGVAL,&ep,0);
136       if (*ep) badusage("invalid window id");
137       set_yppclient_window(windowid);
138     } else
139       badusage("unknown option `%s'",arg);
140   }
141   
142   if (!o_screenshot_fn) {
143     r= asprintf(&o_screenshot_fn,"%s/#pages#.ppm",get_vardir());
144     sysassert(r>=0);
145   }
146
147   if (o_mode & mf_findwindow) {
148     screenshot_startup();
149     find_yppclient_window();
150   }
151   if (o_mode & mf_screenshot) {
152     open_screenshot_file("w");
153     if (o_single_page) take_one_screenshot();
154     else take_screenshots();
155   }
156   if (o_mode & mf_readscreenshot) {
157     open_screenshot_file("r");
158     if (o_single_page) read_one_screenshot();
159     else read_screenshots();
160   }
161   if (o_mode & mf_analyse) {
162     if (o_outputmode)
163       run_analysis();
164     else
165       analyse(stdout);
166   }
167   return 0;
168 }
169
170
171
172
173 DEFINE_VWRAPPERF(, progress, )
174 DEFINE_VWRAPPERF(, progress_log, )
175 DEFINE_VWRAPPERF(, progress_spinner, )
176 DEFINE_VWRAPPERF(, warning, )
177 DEFINE_VWRAPPERF(, fatal, NORET)
178
179 static int last_progress_len;
180      
181 static void vprogress_core(int spinner, const char *fmt, va_list al) {
182   int r;
183   
184   if (o_quiet) return;
185   if (!isatty(2)) return;
186   
187   if (last_progress_len)
188     putc('\r',stderr);
189
190   r= vfprintf(stderr,fmt,al);
191
192   if (spinner) {
193     putc(spinner,stderr);
194     r++;
195   }
196
197   if (r < last_progress_len) {
198     fprintf(stderr,"%*s", last_progress_len - r, "");
199     if (!r) putc('\r', stderr);
200     else while (last_progress_len-- > r) putc('\b',stderr);
201   }
202   last_progress_len= r;
203
204   if (ferror(stderr) || fflush(stderr)) _exit(16);
205 }
206    
207 void vprogress(const char *fmt, va_list al) { vprogress_core(0,fmt,al); }
208 void vprogress_spinner(const char *fmt, va_list al) {
209   static const char spinchars[]="/-\\";
210   static int spinner;
211
212   vprogress_core(spinchars[spinner],fmt,al);
213   spinner++;
214   spinner %= (sizeof(spinchars)-1);
215 }
216
217 void vprogress_log(const char *fmt, va_list al) {
218   if (o_quiet) return;
219   
220   progress("");
221   vfprintf(stderr,fmt,al);
222   putc('\n',stderr);
223   fflush(stderr);
224 }
225
226 void vwarning(const char *fmt, va_list al) {
227   progress("");
228   fputs("Warning: ",stderr);
229   vfprintf(stderr,fmt,al);
230   fputs("\n",stderr);
231   fflush(stderr);
232 }
233
234 void vfatal(const char *fmt, va_list al) {
235   progress("");
236   fputs("\n\nFatal error: ",stderr);
237   vfprintf(stderr,fmt,al);
238   fflush(stderr);
239   fputs("\n\n",stderr);
240   _exit(4);
241 }
242
243 void sysassert_fail(const char *file, int line, const char *what) {
244   int e= errno;
245   progress("");
246   fprintf(stderr,
247           "\nfatal operational error:\n"
248           " unsuccessful execution of: %s\n"
249           " %s:%d: %s\n\n",
250           what, file,line, strerror(e));
251   _exit(16);
252 }
253
254 void *mmalloc(size_t sz) {
255   void *r;
256   if (!sz) return 0;
257   sysassert( r= malloc(sz) );
258   return r;
259 }
260 void *mrealloc(void *p, size_t sz) {
261   assert(sz);
262   void *r;
263   sysassert( r= realloc(p,sz) );
264   return r;
265 }
266
267 void waitpid_check_exitstatus(pid_t pid, const char *what) { 
268   pid_t got;
269   int st;
270   for (;;) {
271     got= waitpid(pid, &st, 0);
272     if (pid==-1) { sysassert(errno==EINTR); continue; }
273     break;
274   }
275   sysassert( got==pid );
276
277   if (WIFEXITED(st)) {
278     if (WEXITSTATUS(st))
279       fatal("%s failed with nonzero exit status %d",
280             what, WEXITSTATUS(st));
281   } else if (WIFSIGNALED(st)) {
282     fatal("%s died due to signal %s%s", what,
283           strsignal(WTERMSIG(st)), WCOREDUMP(st)?" (core dumped)":"");
284   } else {
285     fatal("%s gave strange wait status %d", what, st);
286   }
287 }