chiark / gitweb /
291647e79f4c3930fec1d9ae7c9ccb45718f4695
[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_screenshots_fn;
51 static int o_single_page, o_quiet;
52 static const char *o_outputmode= "upload";
53
54 FILE *screenshots_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_screenshots_file(const char *mode) {
67   screenshots_file= fopen(o_screenshots_fn, mode);
68   if (!screenshots_file)
69     fatal("could not open screenshots file `%s': %s",
70           o_screenshots_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,"--single-page"))
114       o_single_page= 1;
115     else if (!strcmp(arg,"--quiet"))
116       o_quiet= 1;
117     else if (!strcmp(arg,"--upload") ||
118              !strcmp(arg,"--arbitrage") ||
119              !strcmp(arg,"--tsv") ||
120              !strcmp(arg,"--best-prices"))
121       o_outputmode= arg+2;
122     else if (!strcmp(arg,"--screenshots-file"))
123       o_screenshots_fn= ARGVAL;
124 #define DF(f)                                   \
125     else if (!strcmp(arg,"-D" #f))              \
126       debug_flags |= dbg_##f;
127     DEBUG_FLAG_LIST
128 #undef DF
129     else if (!strcmp(arg,"--window-id")) {
130       char *ep;
131       unsigned long windowid= strtoul(ARGVAL,&ep,0);
132       if (*ep) badusage("invalid window id");
133       set_yppclient_window(windowid);
134     } else
135       badusage("unknown option `%s'",arg);
136   }
137   
138   if (!o_screenshots_fn) {
139     r= asprintf(&o_screenshots_fn,"%s/#pages#.ppm",get_vardir());
140     sysassert(r>=0);
141   }
142
143   if (o_mode & mf_findwindow) {
144     screenshot_startup();
145     find_yppclient_window();
146   }
147   if (o_mode & mf_screenshot) {
148     open_screenshots_file("w");
149     if (o_single_page) take_one_screenshot();
150     else take_screenshots();
151   }
152   if (o_mode & mf_readscreenshot) {
153     open_screenshots_file("r");
154     if (o_single_page) read_one_screenshot();
155     else read_screenshots();
156   }
157   if (o_mode & mf_analyse) {
158     run_analysis();
159   }
160   return 0;
161 }
162
163
164
165
166 DEFINE_VWRAPPERF(, progress, )
167 DEFINE_VWRAPPERF(, progress_log, )
168 DEFINE_VWRAPPERF(, progress_spinner, )
169 DEFINE_VWRAPPERF(, warning, )
170 DEFINE_VWRAPPERF(, fatal, NORET)
171
172 static int last_progress_len;
173      
174 static void vprogress_core(int spinner, const char *fmt, va_list al) {
175   int r;
176   
177   if (o_quiet) return;
178   if (!isatty(2)) return;
179   
180   if (last_progress_len)
181     putc('\r',stderr);
182
183   r= vfprintf(stderr,fmt,al);
184
185   if (spinner) {
186     putc(spinner,stderr);
187     r++;
188   }
189
190   if (r < last_progress_len) {
191     fprintf(stderr,"%*s", last_progress_len - r, "");
192     if (!r) putc('\r', stderr);
193     else while (last_progress_len-- > r) putc('\b',stderr);
194   }
195   last_progress_len= r;
196
197   if (ferror(stderr) || fflush(stderr)) _exit(16);
198 }
199    
200 void vprogress(const char *fmt, va_list al) { vprogress_core(0,fmt,al); }
201 void vprogress_spinner(const char *fmt, va_list al) {
202   static const char spinchars[]="/-\\";
203   static int spinner;
204
205   vprogress_core(spinchars[spinner],fmt,al);
206   spinner++;
207   spinner %= (sizeof(spinchars)-1);
208 }
209
210 void vprogress_log(const char *fmt, va_list al) {
211   if (o_quiet) return;
212   
213   progress("");
214   vfprintf(stderr,fmt,al);
215   putc('\n',stderr);
216   fflush(stderr);
217 }
218
219 void vwarning(const char *fmt, va_list al) {
220   progress("");
221   fputs("Warning: ",stderr);
222   vfprintf(stderr,fmt,al);
223   fputs("\n",stderr);
224   fflush(stderr);
225 }
226
227 void vfatal(const char *fmt, va_list al) {
228   progress("");
229   fputs("\n\nFatal error: ",stderr);
230   vfprintf(stderr,fmt,al);
231   fflush(stderr);
232   fputs("\n\n",stderr);
233   _exit(4);
234 }
235
236 void sysassert_fail(const char *file, int line, const char *what) {
237   int e= errno;
238   progress("");
239   fprintf(stderr,
240           "\nfatal operational error:\n"
241           " unsuccessful execution of: %s\n"
242           " %s:%d: %s\n\n",
243           what, file,line, strerror(e));
244   _exit(16);
245 }
246
247 void *mmalloc(size_t sz) {
248   void *r;
249   if (!sz) return 0;
250   sysassert( r= malloc(sz) );
251   return r;
252 }
253 void *mrealloc(void *p, size_t sz) {
254   assert(sz);
255   void *r;
256   sysassert( r= realloc(p,sz) );
257   return r;
258 }
259
260 void waitpid_check_exitstatus(pid_t pid, const char *what) { 
261   pid_t got;
262   int st;
263   for (;;) {
264     got= waitpid(pid, &st, 0);
265     if (pid==-1) { sysassert(errno==EINTR); continue; }
266     break;
267   }
268   sysassert( got==pid );
269
270   if (WIFEXITED(st)) {
271     if (WEXITSTATUS(st))
272       fatal("%s failed with nonzero exit status %d",
273             what, WEXITSTATUS(st));
274   } else if (WIFSIGNALED(st)) {
275     fatal("%s died due to signal %s%s", what,
276           strsignal(WTERMSIG(st)), WCOREDUMP(st)?" (core dumped)":"");
277   } else {
278     fatal("%s gave strange wait status %d", what, st);
279   }
280 }