fixed wrong default for polyglotDir mentioned in docs.
[xboard.git] / zippy.c
1 /*
2  * zippy.c -- Implements Zippy the Pinhead chess player on ICS in XBoard
3  *
4  * Copyright 1991 by Digital Equipment Corporation, Maynard,
5  * Massachusetts.
6  *
7  * Enhancements Copyright 1992-2001, 2002, 2003, 2004, 2005, 2006,
8  * 2007, 2008, 2009, 2010, 2011 Free Software Foundation, Inc.
9  *
10  * Enhancements Copyright 2005 Alessandro Scotti
11  *
12  * The following terms apply to Digital Equipment Corporation's copyright
13  * interest in XBoard:
14  * ------------------------------------------------------------------------
15  * All Rights Reserved
16  *
17  * Permission to use, copy, modify, and distribute this software and its
18  * documentation for any purpose and without fee is hereby granted,
19  * provided that the above copyright notice appear in all copies and that
20  * both that copyright notice and this permission notice appear in
21  * supporting documentation, and that the name of Digital not be
22  * used in advertising or publicity pertaining to distribution of the
23  * software without specific, written prior permission.
24  *
25  * DIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
26  * ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
27  * DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
28  * ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
29  * WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
30  * ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
31  * SOFTWARE.
32  * ------------------------------------------------------------------------
33  *
34  * The following terms apply to the enhanced version of XBoard
35  * distributed by the Free Software Foundation:
36  * ------------------------------------------------------------------------
37  *
38  * GNU XBoard is free software: you can redistribute it and/or modify
39  * it under the terms of the GNU General Public License as published by
40  * the Free Software Foundation, either version 3 of the License, or (at
41  * your option) any later version.
42  *
43  * GNU XBoard is distributed in the hope that it will be useful, but
44  * WITHOUT ANY WARRANTY; without even the implied warranty of
45  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
46  * General Public License for more details.
47  *
48  * You should have received a copy of the GNU General Public License
49  * along with this program. If not, see http://www.gnu.org/licenses/.
50  *
51  *------------------------------------------------------------------------
52  ** See the file ChangeLog for a revision history.  */
53
54 #include "config.h"
55
56 #include <stdio.h>
57 #include <errno.h>
58 #include <sys/types.h>
59 #include <sys/stat.h>
60 #include <ctype.h>
61
62 #if STDC_HEADERS
63 # include <stdlib.h>
64 # include <string.h>
65 #else /* not STDC_HEADERS */
66 extern char *getenv();
67 # if HAVE_STRING_H
68 #  include <string.h>
69 # else /* not HAVE_STRING_H */
70 #  include <strings.h>
71 # endif /* not HAVE_STRING_H */
72 #endif /* not STDC_HEADERS */
73
74 #if TIME_WITH_SYS_TIME
75 # include <sys/time.h>
76 # include <time.h>
77 #else
78 # if HAVE_SYS_TIME_H
79 #  include <sys/time.h>
80 # else
81 #  include <time.h>
82 # endif
83 #endif
84 #define HI "hlelo "
85
86 #if HAVE_UNISTD_H
87 # include <unistd.h>
88 #endif
89
90 #include "common.h"
91 #include "zippy.h"
92 #include "frontend.h"
93 #include "backend.h"
94 #include "backendz.h"
95
96 char *SendMoveToBookUser P((int nr, ChessProgramState *cps, int initial)); // [HGM] book
97 void HandleMachineMove P((char *message, ChessProgramState *cps));
98
99 static char zippyPartner[MSG_SIZ];
100 static char zippyLastOpp[MSG_SIZ];
101 static char zippyOffender[MSG_SIZ]; // [HGM] aborter
102 static int zippyConsecGames;
103 static time_t zippyLastGameEnd;
104
105 extern void mysrandom(unsigned int seed);
106 extern int myrandom(void);
107
108 void ZippyInit()
109 {
110     char *p;
111
112     /* Get name of Zippy lines file */
113     p = getenv("ZIPPYLINES");
114     if (p != NULL) {
115       appData.zippyLines = p;
116     }
117
118     /* Get word that Zippy thinks is insulting */
119     p = getenv("ZIPPYPINHEAD");
120     if (p != NULL) {
121       appData.zippyPinhead = p;
122     }
123
124     /* What password is used for remote control? */
125     p = getenv("ZIPPYPASSWORD");
126     if (p != NULL) {
127       appData.zippyPassword = p;
128     }
129
130     /* What password is used for remote commands to gnuchess? */
131     p = getenv("ZIPPYPASSWORD2");
132     if (p != NULL) {
133       appData.zippyPassword2 = p;
134     }
135
136     /* Joke feature for people who try an old password */
137     p = getenv("ZIPPYWRONGPASSWORD");
138     if (p != NULL) {
139       appData.zippyWrongPassword = p;
140     }
141
142     /* While testing, I want to accept challenges from only one person
143        (namely, my "anonymous" account), so I set an environment
144        variable ZIPPYACCEPTONLY. */
145     p = getenv("ZIPPYACCEPTONLY");
146     if ( p != NULL ) {
147       appData.zippyAcceptOnly = p;
148     }
149
150     /* Should Zippy use "i" command? */
151     /* Defaults to 1=true */
152     p = getenv("ZIPPYUSEI");
153     if (p != NULL) {
154       appData.zippyUseI = atoi(p);
155     }
156
157     /* How does Zippy handle bughouse partnering? */
158     /* 0=say we can't play, 1=manual partnering, 2=auto partnering */
159     p = getenv("ZIPPYBUGHOUSE");
160     if (p != NULL) {
161       appData.zippyBughouse = atoi(p);
162     }
163
164     /* Does Zippy abort games with Crafty? */
165     /* Defaults to 0=false */
166     p = getenv("ZIPPYNOPLAYCRAFTY");
167     if (p != NULL) {
168       appData.zippyNoplayCrafty = atoi(p);
169     }
170
171     /* What ICS command does Zippy send at game end?  Default: "gameend". */
172     p = getenv("ZIPPYGAMEEND");
173     if (p != NULL) {
174       appData.zippyGameEnd = p;
175     }
176
177     /* What ICS command does Zippy send at game start?  Default: none. */
178     p = getenv("ZIPPYGAMESTART");
179     if (p != NULL) {
180       appData.zippyGameStart = p;
181     }
182
183     /* Should Zippy accept adjourns? */
184     /* Defaults to 0=false */
185     p = getenv("ZIPPYADJOURN");
186     if (p != NULL) {
187       appData.zippyAdjourn = atoi(p);
188     }
189
190     /* Should Zippy accept aborts? */
191     /* Defaults to 0=false */
192     p = getenv("ZIPPYABORT");
193     if (p != NULL) {
194       appData.zippyAbort = atoi(p);
195     }
196
197     /* Should Zippy play chess variants (besides bughouse)? */
198     p = getenv("ZIPPYVARIANTS");
199     if (p != NULL) {
200       appData.zippyVariants = p;
201     }
202     safeStrCpy(first.variants, appData.zippyVariants, sizeof(first.variants)/sizeof(first.variants[0]));
203
204     srandom(time(NULL));
205 }
206
207 /*
208  * Routines to implement Zippy talking
209  */
210
211
212 char *swifties[] = {
213     "i acclaims:", "i admonishes:", "i advertises:", "i advises:",
214     "i advocates:", "i affirms:", "i alleges:", "i anathematizes:",
215     "i animadverts:", "i announces:", "i apostrophizes:",
216     "i appeals:", "i applauds:", "i approves:", "i argues:",
217     "i articulates:", "i asserts:", "i asseverates:", "i attests:",
218     "i avers:", "i avows:", "i baas:", "i babbles:", "i banters:",
219     "i barks:", "i bawls:", "i bays:", "i begs:", "i belches:",
220     "i bellows:", "i belts out:", "i berates:", "i beshrews:",
221     "i blabbers:", "i blabs:", "i blares:", "i blasphemes:",
222     "i blasts:", "i blathers:", "i bleats:", "i blithers:",
223     "i blubbers:", "i blurts out:", "i blusters:", "i boasts:",
224     "i brags:", "i brays:", "i broadcasts:", "i burbles:",
225     "i buzzes:", "i cachinnates:", "i cackles:", "i caterwauls:",
226     "i calumniates:", "i caws:", "i censures:", "i chants:",
227     "i chatters:", "i cheeps:", "i cheers:", "i chides:", "i chins:",
228     "i chirps:", "i chortles:", "i chuckles:", "i claims:",
229     "i clamors:", "i clucks:", "i commands:", "i commends:",
230     "i comments:", "i commiserates:", "i communicates:",
231     "i complains:", "i concludes:", "i confabulates:", "i confesses:",
232     "i coos:", "i coughs:", "i counsels:", "i cries:", "i croaks:",
233     "i crows:", "i curses:", "i daydreams:", "i debates:",
234     "i declaims:", "i declares:", "i delivers:", "i denounces:",
235     "i deposes:", "i directs:", "i discloses:", "i disparages:",
236     "i discourses:", "i divulges:", "i documents:", "i drawls:",
237     "i dreams:", "i drivels:", "i drones:", "i effuses:",
238     /*"i ejaculates:",*/ "i elucidates:", "i emotes:", "i endorses:",
239     "i enthuses:", "i entreats:", "i enunciates:", "i eulogizes:",
240     "i exclaims:", "i execrates:", "i exhorts:", "i expatiates:",
241     "i explains:", "i explicates:", "i explodes:", "i exposes:",
242     "i exposits:", "i expostulates: ",
243     "i expounds:", "i expresses:", "i extols:",
244     "i exults:", "i fantasizes:", "i fibs:", "i filibusters:",
245     "i flatters:", "i flutes:", "i fools:", "i free-associates:",
246     "i fulminates:", "i gabbles:", "i gabs:", "i gasps:",
247     "i giggles:", "i gossips:", "i gripes:", "i groans:", "i growls:",
248     "i grunts:", "i guesses:", "i guffaws:", "i gushes:", "i hails:",
249     "i hallucinates:", "i harangues:", "i harmonizes:", "i hectors:",
250     "i hints:", "i hisses:", "i hollers:", "i honks:", "i hoots:",
251     "i hosannas:", "i howls:", "i hums:", "i hypothecates:",
252     "i hypothesizes:", "i imagines:", "i implies:", "i implores:",
253     "i imprecates:", "i indicates:", "i infers:",
254     "i informs everyone:",  "i instructs:", "i interjects:",
255     "i interposes:", "i intimates:", "i intones:", "i introspects:",
256     "i inveighs:", "i jabbers:", "i japes:", "i jests:", "i jibes:",
257     "i jives:", "i jokes:", "i joshes:", "i keens:", "i laments:",
258     "i lauds:", "i laughs:", "i lectures:", "i lies:", "i lilts:",
259     "i lisps:", "i maintains:", "i maledicts:", "i maunders:",
260     "i meows:", "i mewls:", "i mimes:", "i minces:", "i moans:",
261     "i moos:", "i mourns:", "i mouths:", "i mumbles:", "i murmurs:",
262     "i muses:", "i mutters:", "i nags:", "i natters:", "i neighs:",
263     "i notes:", "i nuncupates:", "i objurgates:", "i observes:",
264     "i offers:", "i oinks:", "i opines:", "i orates:", "i orders:",
265     "i panegyrizes:", "i pantomimes:", "i pants:", "i peals:",
266     "i peeps:", "i perorates:", "i persuades:", "i petitions:",
267     "i phonates:", "i pipes up:", "i pitches:", "i pleads:",
268     "i points out:", "i pontificates:", "i postulates:", "i praises:",
269     "i prates:", "i prattles:", "i preaches:", "i prescribes:",
270     "i prevaricates:", "i proclaims:", "i projects:", "i pronounces:",
271     "i proposes:", "i proscribes:", "i quacks:", "i queries:",
272     "i questions:", "i quips:", "i quotes:", "i rages:", "i rambles:",
273     "i rants:", "i raps:", "i rasps:", "i rattles:", "i raves:",
274     "i reacts:", "i recites:", "i recommends:", "i records:",
275     "i reiterates:", "i rejoins:", "i releases:", "i remarks:",
276     "i reminisces:", "i remonstrates:", "i repeats:", "i replies:",
277     "i reports:", "i reprimands:", "i reproaches:", "i reproves:",
278     "i resounds:", "i responds:", "i retorts:", "i reveals:",
279     "i reviles:", "i roars:", "i rumbles:", "i sanctions:",
280     "i satirizes:", "i sauces:", "i scolds:", "i screams:",
281     "i screeches:", "i semaphores:", "i sends:", "i sermonizes:",
282     "i shrieks:", "i sibilates:", "i sighs:", "i signals:",
283     "i signifies:", "i signs:", "i sings:", "i slurs:", "i snaps:",
284     "i snarls:", "i sneezes:", "i snickers:", "i sniggers:",
285     "i snivels:", "i snores:", "i snorts:", "i sobs:",
286     "i soliloquizes:", "i sounds off:", "i sounds out:", "i speaks:",
287     "i spews:", "i spits out:", "i splutters:", "i spoofs:",
288     "i spouts:", "i sputters:", "i squalls:", "i squawks:",
289     "i squeaks:", "i squeals:", "i stammers:", "i states:",
290     "i stresses:", "i stutters:", "i submits:", "i suggests:",
291     "i summarizes:", "i sums up:", "i swears:", "i talks:",
292     "i tattles:", "i teases:", "i telegraphs:", "i testifies:",
293     "i threatens:", "i thunders:", "i titters:", "i tongue-lashes:",
294     "i toots:", "i transcribes:", "i transmits:", "i trills:",
295     "i trumpets:", "i twaddles:", "i tweets:", "i twitters:",
296     "i types:", "i upbraids:", "i urges:", "i utters:", "i ventures:",
297     "i vibrates:", "i vilifies:", "i vituperates:", "i vocalizes:",
298     "i vociferates:", "i voices:", "i waffles:", "i wails:",
299     "i warbles:", "i warns:", "i weeps:", "i wheezes:", "i whimpers:",
300     "i whines:", "i whinnies:", "i whistles:", "i wisecracks:",
301     "i witnesses:", "i woofs:", "i writes:", "i yammers:", "i yawps:",
302     "i yells:", "i yelps:", "i yodels:", "i yowls:", "i zings:",
303 };
304
305 #define MAX_SPEECH 250
306
307 void Speak(how, whom)
308      char *how, *whom;
309 {
310     static FILE *zipfile = NULL;
311     static struct stat zipstat;
312     char zipbuf[MAX_SPEECH + 1];
313     static time_t lastShout = 0;
314     time_t now;
315     char  *p;
316     int c, speechlen;
317     Boolean done;
318
319     if (strcmp(how, "shout") == 0) {
320         now = time((time_t *) NULL);
321         if (now - lastShout < 1*60) return;
322         lastShout = now;
323         if (appData.zippyUseI) {
324             how = swifties[(unsigned) random() %
325                            (sizeof(swifties)/sizeof(char *))];
326         }
327     }
328
329     if (zipfile == NULL) {
330         zipfile = fopen(appData.zippyLines, "r");
331         if (zipfile == NULL) {
332             DisplayFatalError("Can't open Zippy lines file", errno, 1);
333             return;
334         }
335         fstat(fileno(zipfile), &zipstat);
336     }
337
338     for (;;) {
339         fseek(zipfile, (unsigned) random() % zipstat.st_size, 0);
340         do {
341           c = getc(zipfile);
342         } while (c != NULLCHAR && c != '^' && c != EOF);
343         if (c == EOF) continue;
344         while ((c = getc(zipfile)) == '\n') ;
345         if (c == EOF) continue;
346         break;
347     }
348     done = FALSE;
349
350     /* Don't use ics_prefix; we need to let FICS expand the alias i -> it,
351        but use the real command "i" on ICC */
352     safeStrCpy(zipbuf, how, sizeof(zipbuf)/sizeof(zipbuf[0]));
353     strcat(zipbuf, " ");
354     if (whom != NULL) {
355         strcat(zipbuf, whom);
356         strcat(zipbuf, " ");
357     }
358     speechlen = strlen(zipbuf);
359     p = zipbuf + speechlen;
360
361     while (++speechlen < MAX_SPEECH) {
362         if (c == NULLCHAR || c == '^') {
363             *p++ = '\n';
364             *p = '\0';
365             SendToICS(zipbuf);
366             return;
367         } else if (c == '\n') {
368             *p++ = ' ';
369             do {
370                 c = getc(zipfile);
371             } while (c == ' ');
372         } else if (c == EOF) {
373             break;
374         } else {
375             *p++ = c;
376             c = getc(zipfile);
377         }
378     }
379     /* Tried to say something too long, or junk at the end of the
380        file.  Try something else. */
381     Speak(how, whom);  /* tail recursion */
382 }
383
384 int ZippyCalled(str)
385      char *str;
386 {
387     return ics_handle[0] != NULLCHAR && StrCaseStr(str, ics_handle) != NULL;
388 }
389
390 static char opp_name[128][32];
391 static int num_opps=0;
392
393 extern ColorClass curColor;
394
395 static void SetCurColor( ColorClass color )
396 {
397     curColor = color;
398 }
399
400 static void ColorizeEx( ColorClass color, int cont )
401 {
402     if( appData.colorize ) {
403         Colorize( color, cont );
404         SetCurColor( color );
405     }
406 }
407
408 int ZippyControl(buf, i)
409      char *buf;
410      int *i;
411 {
412     char *player, *p;
413     char reply[MSG_SIZ];
414
415     /* Possibly reject Crafty as opponent */
416     if (appData.zippyPlay && appData.zippyNoplayCrafty && forwardMostMove < 4
417         && looking_at(buf, i, "* kibitzes: Hello from Crafty"))
418     {
419         player = StripHighlightAndTitle(star_match[0]);
420         if ((gameMode == IcsPlayingWhite &&
421              StrCaseCmp(player, gameInfo.black) == 0) ||
422             (gameMode == IcsPlayingBlack &&
423              StrCaseCmp(player, gameInfo.white) == 0)) {
424
425           snprintf(reply, MSG_SIZ, "%ssay This computer does not play Crafty clones\n%sabort\n%s+noplay %s\n",
426                   ics_prefix, ics_prefix, ics_prefix, player);
427           SendToICS(reply);
428         }
429         return TRUE;
430     }
431
432     /* If this is a computer, save the name.  Then later, once the */
433     /* game is really started, we will send the "computer" notice to */
434     /* the engine.  */
435     if (appData.zippyPlay &&
436         looking_at(buf, i, "* is in the computer list")) {
437         int i;
438         for (i=0;i<num_opps;i++)
439           if (!strcmp(opp_name[i],star_match[0])) break;
440         if (i >= num_opps) safeStrCpy(opp_name[num_opps++],star_match[0], sizeof(opp_name[num_opps])/sizeof(opp_name[num_opps][0]));
441     }
442     if (appData.zippyPlay && looking_at(buf, i, "* * is a computer *")) {
443         int i;
444         for (i=0;i<num_opps;i++)
445           if (!strcmp(opp_name[i],star_match[1])) break;
446         if (i >= num_opps) safeStrCpy(opp_name[num_opps++],star_match[1], sizeof(opp_name[num_opps])/sizeof(opp_name[num_opps][0]));
447     }
448
449     /* Tells and says */
450     if (appData.zippyPlay &&
451         (looking_at(buf, i, "* offers to be your bughouse partner") ||
452          looking_at(buf, i, "* tells you: [automatic message] I chose you"))) {
453         player = StripHighlightAndTitle(star_match[0]);
454         if (appData.zippyBughouse > 1 && first.initDone) {
455             snprintf(reply, MSG_SIZ,"%spartner %s\n", ics_prefix, player);
456             SendToICS(reply);
457             if (strcmp(zippyPartner, player) != 0) {
458               safeStrCpy(zippyPartner, player, sizeof(zippyPartner)/sizeof(zippyPartner[0]));
459               SendToProgram(reply + strlen(ics_prefix), &first);
460             }
461         } else if (appData.zippyBughouse > 0) {
462             snprintf(reply, MSG_SIZ, "%sdecline %s\n", ics_prefix, player);
463             SendToICS(reply);
464         } else {
465           snprintf(reply, MSG_SIZ, "%stell %s This computer cannot play bughouse\n",
466                     ics_prefix, player);
467             SendToICS(reply);
468         }
469         return TRUE;
470     }
471
472     if (appData.zippyPlay && appData.zippyBughouse && first.initDone &&
473         looking_at(buf, i, "* agrees to be your partner")) {
474         player = StripHighlightAndTitle(star_match[0]);
475         snprintf(reply, MSG_SIZ, "partner %s\n", player);
476         if (strcmp(zippyPartner, player) != 0) {
477           safeStrCpy(zippyPartner, player, sizeof(zippyPartner)/sizeof(zippyPartner[0]));
478           SendToProgram(reply, &first);
479         }
480         return TRUE;
481     }
482
483     if (appData.zippyPlay && appData.zippyBughouse && first.initDone &&
484         (looking_at(buf, i, "are no longer *'s partner") ||
485          looking_at(buf, i,
486                     "* tells you: [automatic message] I'm no longer your"))) {
487         player = StripHighlightAndTitle(star_match[0]);
488         if (strcmp(zippyPartner, player) == 0) {
489             zippyPartner[0] = NULLCHAR;
490             SendToProgram("partner\n", &first);
491         }
492         return TRUE;
493     }
494
495     if (appData.zippyPlay && appData.zippyBughouse && first.initDone &&
496         (looking_at(buf, i, "no longer have a bughouse partner") ||
497          looking_at(buf, i, "partner has disconnected") ||
498          looking_at(buf, i, "partner has just chosen a new partner"))) {
499       zippyPartner[0] = NULLCHAR;
500       SendToProgram("partner\n", &first);
501       return TRUE;
502     }
503
504     if (appData.zippyPlay && appData.zippyBughouse && first.initDone &&
505         looking_at(buf, i, "* (your partner) tells you: *")) {
506         /* This pattern works on FICS but not ICC */
507         player = StripHighlightAndTitle(star_match[0]);
508         if (strcmp(zippyPartner, player) != 0) {
509           safeStrCpy(zippyPartner, player, sizeof(zippyPartner)/sizeof(zippyPartner[0]));
510           snprintf(reply, MSG_SIZ, "partner %s\n", player);
511           SendToProgram(reply, &first);
512         }
513         snprintf(reply, MSG_SIZ, "ptell %s\n", star_match[1]);
514         SendToProgram(reply, &first);
515         return TRUE;
516     }
517
518     if (looking_at(buf, i, "* tells you: *") ||
519         looking_at(buf, i, "* says: *"))
520     {
521         player = StripHighlightAndTitle(star_match[0]);
522         if (appData.zippyPassword[0] != NULLCHAR &&
523             strncmp(star_match[1], appData.zippyPassword,
524                     strlen(appData.zippyPassword)) == 0) {
525             p = star_match[1] + strlen(appData.zippyPassword);
526             while (*p == ' ') p++;
527             SendToICS(p);
528             SendToICS("\n");
529         } else if (appData.zippyPassword2[0] != NULLCHAR && first.initDone &&
530             strncmp(star_match[1], appData.zippyPassword2,
531                     strlen(appData.zippyPassword2)) == 0) {
532             p = star_match[1] + strlen(appData.zippyPassword2);
533             while (*p == ' ') p++;
534             SendToProgram(p, &first);
535             SendToProgram("\n", &first);
536         } else if (appData.zippyWrongPassword[0] != NULLCHAR &&
537             strncmp(star_match[1], appData.zippyWrongPassword,
538                     strlen(appData.zippyWrongPassword)) == 0) {
539             p = star_match[1] + strlen(appData.zippyWrongPassword);
540             while (*p == ' ') p++;
541             snprintf(reply, MSG_SIZ, "wrong %s\n", player);
542             SendToICS(reply);
543         } else if (appData.zippyBughouse && first.initDone &&
544                    strcmp(player, zippyPartner) == 0) {
545             SendToProgram("ptell ", &first);
546             SendToProgram(star_match[1], &first);
547             SendToProgram("\n", &first);
548         } else if (strncmp(star_match[1], HI, 6) == 0) {
549             extern char* programVersion;
550             snprintf(reply, MSG_SIZ, "%stell %s %s\n",
551                     ics_prefix, player, programVersion);
552             SendToICS(reply);
553         } else if (strncmp(star_match[1], "W0W!! ", 6) == 0) {
554             extern char* programVersion;
555             snprintf(reply, MSG_SIZ, "%stell %s %s\n", ics_prefix,
556                     player, programVersion);
557             SendToICS(reply);
558         } else if (appData.zippyTalk && (((unsigned) random() % 10) < 9)) {
559             if (strcmp(player, ics_handle) != 0) {
560                 Speak("tell", player);
561             }
562         }
563
564         ColorizeEx( ColorTell, FALSE );
565
566         return TRUE;
567     }
568
569     if( appData.colorize && looking_at(buf, i, "* (*) seeking") ) {
570         ColorizeEx(ColorSeek, FALSE);
571         return FALSE;
572     }
573
574     if (looking_at(buf, i, "* spoofs you:")) {
575         player = StripHighlightAndTitle(star_match[0]);
576         snprintf(reply, MSG_SIZ, "spoofedby %s\n", player);
577         SendToICS(reply);
578     }
579
580     return FALSE;
581 }
582
583 int ZippyConverse(buf, i)
584      char *buf;
585      int *i;
586 {
587     static char lastgreet[MSG_SIZ];
588     char reply[MSG_SIZ];
589     int oldi;
590
591     /* Shouts and emotes */
592     if (looking_at(buf, i, "--> * *") ||
593         looking_at(buf, i, "* shouts: *"))
594     {
595       if (appData.zippyTalk) {
596         char *player = StripHighlightAndTitle(star_match[0]);
597         if (strcmp(player, ics_handle) == 0) {
598             return TRUE;
599         } else if (appData.zippyPinhead[0] != NULLCHAR &&
600                    StrCaseStr(star_match[1], appData.zippyPinhead) != NULL) {
601           snprintf(reply, MSG_SIZ, "insult %s\n", player);
602             SendToICS(reply);
603         } else if (ZippyCalled(star_match[1])) {
604             Speak("shout", NULL);
605         }
606       }
607
608       ColorizeEx(ColorShout, FALSE);
609
610       return TRUE;
611     }
612
613     if (looking_at(buf, i, "* kibitzes: *")) {
614       if (appData.zippyTalk && ((unsigned) random() % 10) < 9) {
615         char *player = StripHighlightAndTitle(star_match[0]);
616         if (strcmp(player, ics_handle) != 0) {
617             Speak("kibitz", NULL);
618         }
619       }
620
621       ColorizeEx(ColorKibitz, FALSE);
622
623       return TRUE;
624     }
625
626     if (looking_at(buf, i, "* whispers: *")) {
627       if (appData.zippyTalk && ((unsigned) random() % 10) < 9) {
628         char *player = StripHighlightAndTitle(star_match[0]);
629         if (strcmp(player, ics_handle) != 0) {
630             Speak("whisper", NULL);
631         }
632       }
633
634       ColorizeEx(ColorKibitz, FALSE);
635
636       return TRUE;
637     }
638
639     /* Messages */
640     if ((looking_at(buf, i, ". * (*:*): *") && isdigit(star_match[1][0])) ||
641          looking_at(buf, i, ". * at *:*: *")) {
642       if (appData.zippyTalk) {
643         FILE *f;
644         char *player = StripHighlightAndTitle(star_match[0]);
645
646         if (strcmp(player, ics_handle) != 0) {
647             if (((unsigned) random() % 10) < 9)
648               Speak("message", player);
649             f = fopen("zippy.messagelog", "a");
650             fprintf(f, "%s (%s:%s): %s\n", player,
651                     star_match[1], star_match[2], star_match[3]);
652             fclose(f);
653         }
654       }
655       return TRUE;
656     }
657
658     /* Channel tells */
659     oldi = *i;
660     if (looking_at(buf, i, "*(*: *")) {
661         char *player;
662         char *channel;
663         if (star_match[0][0] == NULLCHAR  ||
664             strchr(star_match[0], ' ') ||
665             strchr(star_match[1], ' ')) {
666             /* Oops, did not want to match this; probably a message */
667             *i = oldi;
668             return FALSE;
669         }
670         if (appData.zippyTalk) {
671           player = StripHighlightAndTitle(star_match[0]);
672           channel = strrchr(star_match[1], '(');
673           if (channel == NULL) {
674             channel = star_match[1];
675           } else {
676             channel++;
677           }
678           channel[strlen(channel)-1] = NULLCHAR;
679
680           /* Tell to the channel only if someone mentions our name */
681           if (ZippyCalled(star_match[2])) {
682             Speak("tell", channel);
683           }
684
685           ColorizeEx( atoi(channel) == 1 ? ColorChannel1 : ColorChannel, FALSE );
686         }
687         return TRUE;
688     }
689
690     if (!appData.zippyTalk) return FALSE;
691
692     if ((looking_at(buf, i, "You have * message") &&
693          atoi(star_match[0]) != 0) ||
694         looking_at(buf, i, "* has left a message for you") ||
695         looking_at(buf, i, "* just sent you a message")) {
696       snprintf(reply, MSG_SIZ, "%smessages\n%sclearmessages *\n",
697                 ics_prefix, ics_prefix);
698         SendToICS(reply);
699         return TRUE;
700     }
701
702     if (looking_at(buf, i, "Notification: * has arrived")) {
703         if (((unsigned) random() % 3) == 0) {
704             char *player = StripHighlightAndTitle(star_match[0]);
705             safeStrCpy(lastgreet, player, sizeof(lastgreet)/sizeof(lastgreet[0]));
706             snprintf(reply, MSG_SIZ, "greet %s\n", player);
707             SendToICS(reply);
708             Speak("tell", player);
709         }
710     }
711
712     if (looking_at(buf, i, "Notification: * has departed")) {
713         if (((unsigned) random() % 3) == 0) {
714             char *player = StripHighlightAndTitle(star_match[0]);
715             snprintf(reply, MSG_SIZ, "farewell %s\n", player);
716             SendToICS(reply);
717         }
718     }
719
720     if (looking_at(buf, i, "Not sent -- * is censoring you")) {
721         char *player = StripHighlightAndTitle(star_match[0]);
722         if (strcmp(player, lastgreet) == 0) {
723           snprintf(reply, MSG_SIZ, "%s-notify %s\n", ics_prefix, player);
724             SendToICS(reply);
725         }
726     }
727
728     if (looking_at(buf, i, "command is currently turned off")) {
729         appData.zippyUseI = 0;
730     }
731
732     return FALSE;
733 }
734
735 void ZippyGameStart(white, black)
736      char *white, *black;
737 {
738     if (!first.initDone) {
739       /* Game is starting prematurely.  We can't deal with this */
740       SendToICS(ics_prefix);
741       SendToICS("abort\n");
742       SendToICS(ics_prefix);
743       SendToICS("say Sorry, the chess program is not initialized yet.\n");
744       return;
745     }
746
747     if (appData.zippyGameStart[0] != NULLCHAR) {
748       SendToICS(appData.zippyGameStart);
749       SendToICS("\n");
750     }
751 }
752
753 void ZippyGameEnd(result, resultDetails)
754      ChessMove result;
755      char *resultDetails;
756 {
757     if (appData.zippyAcceptOnly[0] == NULLCHAR &&
758         appData.zippyGameEnd[0] != NULLCHAR) {
759       SendToICS(appData.zippyGameEnd);
760       SendToICS("\n");
761     }
762     zippyLastGameEnd = time(0);
763     if(forwardMostMove < appData.zippyShortGame)
764       safeStrCpy(zippyOffender, zippyLastOpp, sizeof(zippyOffender)/sizeof(zippyOffender[0]));
765     else
766       zippyOffender[0] = 0; // [HGM] aborter
767 }
768
769 /*
770  * Routines to implement Zippy playing chess
771  */
772
773 void ZippyHandleChallenge(srated, swild, sbase, sincrement, opponent)
774      char *srated, *swild, *sbase, *sincrement, *opponent;
775 {
776     char buf[MSG_SIZ];
777     int base, increment, i=0;
778     char rated;
779     VariantClass variant;
780     char *varname;
781
782     rated = srated[0];
783     variant = StringToVariant(swild);
784     varname = VariantName(variant);
785     base = atoi(sbase);
786     increment = atoi(sincrement);
787
788     /* [DM] If icsAnalyzeEngine active we don't accept automatic games */
789     if (appData.icsActive && appData.icsEngineAnalyze) return;
790
791     /* If desired, you can insert more code here to decline matches
792        based on rated, variant, base, and increment, but it is
793        easier to use the ICS formula feature instead. */
794
795     if (variant == VariantLoadable) {
796       snprintf(buf, MSG_SIZ,
797          "%stell %s This computer can't play wild type %s\n%sdecline %s\n",
798                 ics_prefix, opponent, swild, ics_prefix, opponent);
799         SendToICS(buf);
800         return;
801     }
802     if (StrStr(appData.zippyVariants, varname) == NULL ||
803               ((i=first.protocolVersion) != 1 && StrStr(first.variants, varname) == NULL) /* [HGM] zippyvar */
804                                                           ) {
805       snprintf(buf, MSG_SIZ,
806          "%stell %s This computer can't play %s [%s], only %s\n%sdecline %s\n",
807                 ics_prefix, opponent, swild, varname,
808                 i ? first.variants : appData.zippyVariants,                               /* [HGM] zippyvar */
809                 ics_prefix, opponent);
810         SendToICS(buf);
811         return;
812     }
813
814     /* Are we blocking match requests from all but one person? */
815     if (appData.zippyAcceptOnly[0] != NULLCHAR &&
816         StrCaseCmp(opponent, appData.zippyAcceptOnly)) {
817         /* Yes, and this isn't him.  Ignore challenge. */
818         return;
819     }
820
821     /* Too many consecutive games with same opponent?  If so, make him
822        wait until someone else has played or a timeout has elapsed. */
823     if (appData.zippyMaxGames &&
824         strcmp(opponent, zippyLastOpp) == 0 &&
825         zippyConsecGames >= appData.zippyMaxGames &&
826         difftime(time(0), zippyLastGameEnd) < appData.zippyReplayTimeout) {
827       snprintf(buf, MSG_SIZ,  "%stell %s Sorry, you have just played %d consecutive games against %s.  To give others a chance, please wait %d seconds or until someone else has played.\n%sdecline %s\n",
828               ics_prefix, opponent, zippyConsecGames, ics_handle,
829               appData.zippyReplayTimeout, ics_prefix, opponent);
830       SendToICS(buf);
831       return;
832     }
833
834     /* [HGM] aborter: opponent is cheater that aborts games he doesn't like on first move. Make him wait */
835     if (strcmp(opponent, zippyOffender) == 0 &&
836         difftime(time(0), zippyLastGameEnd) < appData.zippyReplayTimeout) {
837       snprintf(buf, MSG_SIZ,  "%stell %s Sorry, your previous game against %s was rather short. "
838                    " It will wait %d seconds to see if a tougher opponent comes along.\n%sdecline %s\n",
839               ics_prefix, opponent, ics_handle,
840               appData.zippyReplayTimeout, ics_prefix, opponent);
841       SendToICS(buf);
842       return;
843     }
844
845     /* Engine not yet initialized or still thinking about last game? */
846     if (!first.initDone || first.lastPing != first.lastPong) {
847       snprintf(buf, MSG_SIZ,  "%stell %s I'm not quite ready for a new game yet; try again soon.\n%sdecline %s\n",
848               ics_prefix, opponent, ics_prefix, opponent);
849       SendToICS(buf);
850       return;
851     }
852
853     snprintf(buf, MSG_SIZ, "%saccept %s\n", ics_prefix, opponent);
854     SendToICS(buf);
855     if (appData.zippyTalk) {
856       Speak("tell", opponent);
857     }
858 }
859
860
861 /* Accept matches */
862 int ZippyMatch(buf, i)
863      char *buf;
864      int *i;
865 {
866     if (looking_at(buf, i, "* * match * * requested with * (*)")) {
867
868         ZippyHandleChallenge(star_match[0], star_match[1],
869                              star_match[2], star_match[3],
870                              StripHighlightAndTitle(star_match[4]));
871         return TRUE;
872     }
873
874     /* Old FICS 0-increment form */
875     if (looking_at(buf, i, "* * match * requested with * (*)")) {
876
877         ZippyHandleChallenge(star_match[0], star_match[1],
878                              star_match[2], "0",
879                              StripHighlightAndTitle(star_match[3]));
880         return TRUE;
881     }
882
883     if (looking_at(buf, i,
884                    "* has made an alternate proposal of * * match * *.")) {
885
886         ZippyHandleChallenge(star_match[1], star_match[2],
887                              star_match[3], star_match[4],
888                              StripHighlightAndTitle(star_match[0]));
889         return TRUE;
890     }
891
892     /* FICS wild/nonstandard forms */
893     if (looking_at(buf, i, "Challenge: * (*) *(*) * * * * Loaded from *")) {
894         /* note: star_match[2] can include "[white] " or "[black] "
895            before our own name. */
896         if(star_match[8] == NULL || star_match[8][0] == 0) // [HGM] chessd: open-source ICS has file on next line
897              ZippyHandleChallenge(star_match[4], star_match[5],
898                              star_match[6], star_match[7],                           StripHighlightAndTitle(star_match[0]));
899         else ZippyHandleChallenge(star_match[4], star_match[8],
900                              star_match[6], star_match[7],
901                              StripHighlightAndTitle(star_match[0]));
902         return TRUE;
903     }
904
905     if (looking_at(buf, i,
906                    "Challenge: * (*) *(*) * * * * : * * Loaded from *")) {
907         /* note: star_match[2] can include "[white] " or "[black] "
908            before our own name. */
909         ZippyHandleChallenge(star_match[4], star_match[10],
910                              star_match[8], star_match[9],
911                              StripHighlightAndTitle(star_match[0]));
912         return TRUE;
913     }
914
915     /* Regular forms */
916     if (looking_at(buf, i, "Challenge: * (*) *(*) * * * * : * *") |
917         looking_at(buf, i, "Challenge: * (*) *(*) * * * * * *")) {
918         /* note: star_match[2] can include "[white] " or "[black] "
919            before our own name. */
920         ZippyHandleChallenge(star_match[4], star_match[5],
921                              star_match[8], star_match[9],
922                              StripHighlightAndTitle(star_match[0]));
923         return TRUE;
924     }
925
926     if (looking_at(buf, i, "Challenge: * (*) *(*) * * * *")) {
927         /* note: star_match[2] can include "[white] " or "[black] "
928            before our own name. */
929         ZippyHandleChallenge(star_match[4], star_match[5],
930                              star_match[6], star_match[7],
931                              StripHighlightAndTitle(star_match[0]));
932         return TRUE;
933     }
934
935
936         if (looking_at(buf, i, "Your opponent offers you a draw") ||
937             looking_at(buf, i, "* offers you a draw")) {
938             if (first.sendDrawOffers && first.initDone) {
939                 SendToProgram("draw\n", &first);
940             }
941             return TRUE;
942         }
943
944     if (looking_at(buf, i, "requests that the game be aborted") ||
945         looking_at(buf, i, "would like to abort")) {
946         if (appData.zippyAbort ||
947             (gameMode == IcsPlayingWhite && whiteTimeRemaining < 0) ||
948             (gameMode == IcsPlayingBlack && blackTimeRemaining < 0)) {
949             SendToICS(ics_prefix);
950             SendToICS("abort\n");
951         } else {
952             SendToICS(ics_prefix);
953             if (appData.zippyTalk)
954               SendToICS("say Whoa no!  I am having FUN!!\n");
955             else
956               SendToICS("say Sorry, this computer doesn't accept aborts.\n");
957         }
958         return TRUE;
959     }
960
961     if (looking_at(buf, i, "requests adjournment") ||
962         looking_at(buf, i, "would like to adjourn")) {
963       if (appData.zippyAdjourn) {
964         SendToICS(ics_prefix);
965         SendToICS("adjourn\n");
966       } else {
967         SendToICS(ics_prefix);
968         if (appData.zippyTalk)
969           SendToICS("say Whoa no!  I am having FUN playing NOW!!\n");
970         else
971           SendToICS("say Sorry, this computer doesn't accept adjourns.\n");
972       }
973       return TRUE;
974     }
975
976     return FALSE;
977 }
978
979 /* Initialize chess program with data from the first board
980  * of a new or resumed game.
981  */
982 void ZippyFirstBoard(moveNum, basetime, increment)
983      int moveNum, basetime, increment;
984 {
985     char buf[MSG_SIZ];
986     int w, b;
987     char *opp = (gameMode==IcsPlayingWhite ? gameInfo.black : gameInfo.white);
988     Boolean sentPos = FALSE;
989     char *bookHit = NULL; // [HGM] book
990
991     if (!first.initDone) {
992       /* Game is starting prematurely.  We can't deal with this */
993       SendToICS(ics_prefix);
994       SendToICS("abort\n");
995       SendToICS(ics_prefix);
996       SendToICS("say Sorry, the chess program is not initialized yet.\n");
997       return;
998     }
999
1000     /* Send the variant command if needed */
1001     if (gameInfo.variant != VariantNormal) {
1002       snprintf(buf, MSG_SIZ,  "variant %s\n", VariantName(gameInfo.variant));
1003       SendToProgram(buf, &first);
1004     }
1005
1006     if ((startedFromSetupPosition && moveNum == 0) ||
1007         (!appData.getMoveList && moveNum > 0)) {
1008       SendToProgram("force\n", &first);
1009       SendBoard(&first, moveNum);
1010       sentPos = TRUE;
1011     }
1012
1013     snprintf(buf, MSG_SIZ,  "level 0 %d %d\n", basetime, increment);
1014     SendToProgram(buf, &first);
1015
1016     /* Count consecutive games from one opponent */
1017     if (strcmp(opp, zippyLastOpp) == 0) {
1018       zippyConsecGames++;
1019     } else {
1020       zippyConsecGames = 1;
1021       safeStrCpy(zippyLastOpp, opp, sizeof(zippyLastOpp)/sizeof(zippyLastOpp[0]));
1022     }
1023
1024     /* Send the "computer" command if the opponent is in the list
1025        we've been gathering. */
1026     for (w=0; w<num_opps; w++) {
1027         if (!strcmp(opp_name[w], opp)) {
1028             SendToProgram(first.computerString, &first);
1029             break;
1030         }
1031     }
1032
1033     /* Ratings might be < 0 which means "we haven't seen a ratings
1034        message from ICS." Send 0 in that case */
1035     w = (gameInfo.whiteRating >= 0) ? gameInfo.whiteRating : 0;
1036     b = (gameInfo.blackRating >= 0) ? gameInfo.blackRating : 0;
1037
1038     firstMove = FALSE;
1039     if (gameMode == IcsPlayingWhite) {
1040         if (first.sendName) {
1041           snprintf(buf, MSG_SIZ,  "name %s\n", gameInfo.black);
1042           SendToProgram(buf, &first);
1043         }
1044         safeStrCpy(ics_handle, gameInfo.white, MSG_SIZ);
1045         snprintf(buf, MSG_SIZ,  "rating %d %d\n", w, b);
1046         SendToProgram(buf, &first);
1047         if (sentPos) {
1048             /* Position sent above, engine is in force mode */
1049             if (WhiteOnMove(moveNum)) {
1050               /* Engine is on move now */
1051               if (first.sendTime) {
1052                 if (first.useColors) {
1053                   SendToProgram("black\n", &first); /*gnu kludge*/
1054                   SendTimeRemaining(&first, TRUE);
1055                   SendToProgram("white\n", &first);
1056                 } else {
1057                   SendTimeRemaining(&first, TRUE);
1058                 }
1059               }
1060               bookHit = SendMoveToBookUser(forwardMostMove-1, &first, TRUE); // [HGM] book: send go or retrieve book move
1061             } else {
1062                 /* Engine's opponent is on move now */
1063                 if (first.usePlayother) {
1064                   if (first.sendTime) {
1065                     SendTimeRemaining(&first, TRUE);
1066                   }
1067                   SendToProgram("playother\n", &first);
1068                 } else {
1069                   /* Need to send a "go" after opponent moves */
1070                   firstMove = TRUE;
1071                 }
1072             }
1073         } else {
1074             /* Position not sent above, move list might be sent later */
1075             if (moveNum == 0) {
1076                 /* No move list coming; at start of game */
1077               if (first.sendTime) {
1078                 if (first.useColors) {
1079                   SendToProgram("black\n", &first); /*gnu kludge*/
1080                   SendTimeRemaining(&first, TRUE);
1081                   SendToProgram("white\n", &first);
1082                 } else {
1083                   SendTimeRemaining(&first, TRUE);
1084                 }
1085               }
1086 //            SendToProgram("go\n", &first);
1087               bookHit = SendMoveToBookUser(forwardMostMove-1, &first, TRUE); // [HGM] book: send go or retrieve book move
1088             }
1089         }
1090     } else if (gameMode == IcsPlayingBlack) {
1091         if (first.sendName) {
1092           snprintf(buf, MSG_SIZ,  "name %s\n", gameInfo.white);
1093           SendToProgram(buf, &first);
1094         }
1095         safeStrCpy(ics_handle, gameInfo.black, MSG_SIZ);
1096         snprintf(buf, MSG_SIZ,  "rating %d %d\n", b, w);
1097         SendToProgram(buf, &first);
1098         if (sentPos) {
1099             /* Position sent above, engine is in force mode */
1100             if (!WhiteOnMove(moveNum)) {
1101                 /* Engine is on move now */
1102               if (first.sendTime) {
1103                 if (first.useColors) {
1104                   SendToProgram("white\n", &first); /*gnu kludge*/
1105                   SendTimeRemaining(&first, FALSE);
1106                   SendToProgram("black\n", &first);
1107                 } else {
1108                   SendTimeRemaining(&first, FALSE);
1109                 }
1110               }
1111 //            SendToProgram("go\n", &first);
1112               bookHit = SendMoveToBookUser(forwardMostMove-1, &first, TRUE); // [HGM] book: send go or retrieve book move
1113             } else {
1114                 /* Engine's opponent is on move now */
1115                 if (first.usePlayother) {
1116                   if (first.sendTime) {
1117                     SendTimeRemaining(&first, FALSE);
1118                   }
1119                   SendToProgram("playother\n", &first);
1120                 } else {
1121                   /* Need to send a "go" after opponent moves */
1122                   firstMove = TRUE;
1123                 }
1124             }
1125         } else {
1126             /* Position not sent above, move list might be sent later */
1127             /* Nothing needs to be done here */
1128         }
1129     }
1130
1131     if(bookHit) { // [HGM] book: simulate book reply
1132         static char bookMove[MSG_SIZ]; // a bit generous?
1133
1134         programStats.depth = programStats.nodes = programStats.time =
1135         programStats.score = programStats.got_only_move = 0;
1136         sprintf(programStats.movelist, "%s (xbook)", bookHit);
1137
1138         safeStrCpy(bookMove, "move ", sizeof(bookMove)/sizeof(bookMove[0]));
1139         strcat(bookMove, bookHit);
1140         HandleMachineMove(bookMove, &first);
1141     }
1142 }
1143
1144
1145 void
1146 ZippyHoldings(white_holding, black_holding, new_piece)
1147      char *white_holding, *black_holding, *new_piece;
1148 {
1149     char buf[MSG_SIZ];
1150     if (gameMode != IcsPlayingBlack && gameMode != IcsPlayingWhite) return;
1151     snprintf(buf, MSG_SIZ,  "holding [%s] [%s] %s\n",
1152             white_holding, black_holding, new_piece);
1153     SendToProgram(buf, &first);
1154 }