2 * ypp-commodities main program: argument parsing etc.
5 * This is part of ypp-sc-tools, a set of third-party tools for assisting
6 * players of Yohoho Puzzle Pirates.
8 * Copyright (C) 2009 Ian Jackson <ijackson@chiark.greenend.org.uk>
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.
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.
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/>.
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.
30 void debug_flush(void) {
31 sysassert(!ferror(debug));
32 sysassert(!fflush(debug));
35 const char *get_vardir(void) { return "."; }
40 mf_readscreenshot= 0020,
43 mode_findwindow= 0001,
44 mode_screenshot= 0011,
50 static char *o_screenshot_fn;
51 static int o_single_page, o_quiet;
52 static const char *o_outputmode= "upload";
54 FILE *screenshot_file;
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);
64 DEFINE_VWRAPPERF(static, badusage, NORET);
66 static void open_screenshot_file(const char *mode) {
67 screenshot_file= fopen(o_screenshot_fn, mode);
69 fatal("could not open screenshots file `%s': %s",
70 o_screenshot_fn, strerror(errno));
73 static void run_analysis(void) {
76 sysassert( tf= tmpfile() );
77 progress("running recognition...");
80 if (o_single_page && !strcmp(o_outputmode,"upload"))
81 fatal("Recognition successful, but refusing to upload partial data\n"
82 " (--single-page specified). Specify an output mode?");
84 sysassert( fseek(tf,0,SEEK_SET) == 0);
86 progress_log("processing results (--%s)...", o_outputmode);
88 sysassert( (processor= fork()) != -1 );
91 sysassert( dup2(fileno(tf),0) ==0 );
92 execlp("./yppsc-commod-processor", "yppsc-commod-processor",
93 o_outputmode, (char*)0);
94 sysassert(!"execlp commod-processor failed");
97 waitpid_check_exitstatus(processor, "output processor/uploader");
99 progress_log("all complete.");
102 int main(int argc, char **argv) {
106 #define ARGVAL ((*++argv) ? *argv : \
107 (badusage("missing value for option %s",arg),(char*)0))
109 while ((arg=*++argv)) {
110 if (!strcmp(arg,"--find-window-only"))
111 o_mode= mode_findwindow;
112 else if (!strcmp(arg,"--screenshot-only"))
113 o_mode= mode_screenshot;
114 else if (!strcmp(arg,"--analyse-only") ||
115 !strcmp(arg,"--same"))
116 o_mode= mode_analyse;
117 else if (!strcmp(arg,"--everything"))
119 else if (!strcmp(arg,"--single-page"))
121 else if (!strcmp(arg,"--quiet"))
123 else if (!strcmp(arg,"--raw-tsv"))
125 else if (!strcmp(arg,"--upload") ||
126 !strcmp(arg,"--arbitrage") ||
127 !strcmp(arg,"--tsv") ||
128 !strcmp(arg,"--best-prices"))
130 else if (!strcmp(arg,"--screenshot-file"))
131 o_screenshot_fn= ARGVAL;
133 else if (!strcmp(arg,"-D" #f)) \
134 debug_flags |= dbg_##f;
137 else if (!strcmp(arg,"--window-id")) {
139 unsigned long windowid= strtoul(ARGVAL,&ep,0);
140 if (*ep) badusage("invalid window id");
141 set_yppclient_window(windowid);
143 badusage("unknown option `%s'",arg);
146 if (!o_screenshot_fn) {
147 r= asprintf(&o_screenshot_fn,"%s/#pages#.ppm",get_vardir());
151 if (o_mode & mf_findwindow) {
152 screenshot_startup();
153 find_yppclient_window();
155 if (o_mode & mf_screenshot) {
156 open_screenshot_file("w");
157 if (o_single_page) take_one_screenshot();
158 else take_screenshots();
160 if (o_mode & mf_readscreenshot) {
161 open_screenshot_file("r");
162 if (o_single_page) read_one_screenshot();
163 else read_screenshots();
165 if (o_mode & mf_analyse) {
177 DEFINE_VWRAPPERF(, progress, )
178 DEFINE_VWRAPPERF(, progress_log, )
179 DEFINE_VWRAPPERF(, progress_spinner, )
180 DEFINE_VWRAPPERF(, warning, )
181 DEFINE_VWRAPPERF(, fatal, NORET)
183 static int last_progress_len;
185 static void vprogress_core(int spinner, const char *fmt, va_list al) {
189 if (!isatty(2)) return;
191 if (last_progress_len)
194 r= vfprintf(stderr,fmt,al);
197 putc(spinner,stderr);
201 if (r < last_progress_len) {
202 fprintf(stderr,"%*s", last_progress_len - r, "");
203 if (!r) putc('\r', stderr);
204 else while (last_progress_len-- > r) putc('\b',stderr);
206 last_progress_len= r;
208 if (ferror(stderr) || fflush(stderr)) _exit(16);
211 void vprogress(const char *fmt, va_list al) { vprogress_core(0,fmt,al); }
212 void vprogress_spinner(const char *fmt, va_list al) {
213 static const char spinchars[]="/-\\";
216 vprogress_core(spinchars[spinner],fmt,al);
218 spinner %= (sizeof(spinchars)-1);
221 void vprogress_log(const char *fmt, va_list al) {
225 vfprintf(stderr,fmt,al);
230 void vwarning(const char *fmt, va_list al) {
232 fputs("Warning: ",stderr);
233 vfprintf(stderr,fmt,al);
238 void vfatal(const char *fmt, va_list al) {
240 fputs("\n\nFatal error: ",stderr);
241 vfprintf(stderr,fmt,al);
243 fputs("\n\n",stderr);
247 void sysassert_fail(const char *file, int line, const char *what) {
251 "\nfatal operational error:\n"
252 " unsuccessful execution of: %s\n"
254 what, file,line, strerror(e));
258 void *mmalloc(size_t sz) {
261 sysassert( r= malloc(sz) );
264 void *mrealloc(void *p, size_t sz) {
267 sysassert( r= realloc(p,sz) );
271 void waitpid_check_exitstatus(pid_t pid, const char *what) {
275 got= waitpid(pid, &st, 0);
276 if (pid==-1) { sysassert(errno==EINTR); continue; }
279 sysassert( got==pid );
283 fatal("%s failed with nonzero exit status %d",
284 what, WEXITSTATUS(st));
285 } else if (WIFSIGNALED(st)) {
286 fatal("%s died due to signal %s%s", what,
287 strsignal(WTERMSIG(st)), WCOREDUMP(st)?" (core dumped)":"");
289 fatal("%s gave strange wait status %d", what, st);