<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
"http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv=Content-Type content="text/html; charset=utf8">
<title>/usr/web/sources/contrib/fhs/dic.c - Plan 9 from Bell Labs</title>
<!-- THIS FILE IS AUTOMATICALLY GENERATED. -->
<!-- EDIT sources.tr INSTEAD. -->
</meta>
</head>
<body>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="line-height: 1.2em; margin-left: 1.00in; text-indent: 0.00in; margin-right: 1.00in; margin-top: 0; margin-bottom: 0; text-align: center;">
<span style="font-size: 10pt"><a href="/plan9/">Plan 9 from Bell Labs</a>&rsquo;s /usr/web/sources/contrib/fhs/dic.c</span></p>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center><font size=-1>
Copyright © 2009 Alcatel-Lucent.<br />
Distributed under the
<a href="/plan9/license.html">Lucent Public License version 1.02</a>.
<br />
<a href="/plan9/download.html">Download the Plan 9 distribution.</a>
</font>
</center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<table width="100%" cellspacing=0 border=0><tr><td align="center">
<table cellspacing=0 cellpadding=5 bgcolor="#eeeeff"><tr><td align="left">
<pre>
<!-- END HEADER -->
/*
* Dictionary Server Protocol client (RFC2229)
*/
#include &lt;u.h&gt;
#include &lt;libc.h&gt;
#include &lt;bio.h&gt;
#include &lt;ctype.h&gt;

#pragma varargck type   "D" char*
#pragma varargck argpos sendcmd 1
#pragma varargck argpos simplecmd 1

typedef struct Resp Resp;
struct Resp {           /* Response */
       int             type;
       char    *line;
       int             code;
       char    msg[1024];      /* everything but response code */
       int             n;
       char    *word;
       char    *db;
       char    *dbdescr;
};

enum {
       Rstatus = 1,    /* status response */
       Rtext,                  /* text response */
       EOT                             /* End of Text marker */
};

Biobuf bin, bout;
char *addr;
char *database = "*";
char *strat = ".";
int domatch;
int showdb;
int showstrat;
int showserver;
int showinfo;

char*
dbquote(char *s)        /* double quote a string */
{
       char *q, *r, *t;

       /*
        * max size occurs when escaping each char
        * + 2 quotes surrounding it + null char
        */
       q = malloc(2*strlen(s)+2+1);
       r = q;
       *r++ = '"';
       for(t = s; *t; t++)
               switch(*t){
               case '\\':
               case '"':
                       *r++ = '\\';
                       *r++ = *t;
                       break;
               default:
                       *r++ = *t;
               }
       *r++ = '"';
       *r = 0;
       return q;
}

int
Dfmt(Fmt *f)
{
       char *s, *q;
       int n;

       s = va_arg(f-&gt;args, char*);
       q = dbquote(s);
       n = fmtprint(f, "%s", q);
       free(q);
       return n;
}

/*
* unquote a string quoted using quote char qc
*/
char*
unquote(char qc, char *s)
{
       char *t;

       *s++ = 0;               /* remove quote char */
       for(t = s; *t; t++){
               if(t[0] == '\\' &amp;&amp; t[1] == '\\')
                       t++;
               else if(t[0] == '\\' &amp;&amp; t[1] == qc)
                       t++;
               else if(t[0] == qc)
                       break;
               *s++ = *t;
       }
       if(t[0] == 0 || t[1] == 0)
               t = nil;        /* end of input string */
       else
               t++;    /* after the quoted string */
       *s = 0;
       return t;
}

int
qtokenize(char *s, char **args, int maxargs)
{
       int i;

       if(*s == 0)
               return 0;
       for(i = 0; i &lt; maxargs; i++){
               args[i] = s;
               if(*s == '"' || *s == '\''){
                       args[i] = &amp;s[1];
                       s = unquote(*s, s);
                       if(s == nil)
                               return i+1;
                       *s++ = 0;       /* skip space */
               }else{
                       s = strchr(s, ' ');
                       if(s == nil)
                               return i+1;
                       *s++ = 0;       /* skip space */
               }
       }
       return i;
}

char*
getline(void)
{
       char *s;
       int n;

       s = Brdline(&amp;bin, '\n');
       if(s != nil){
               n = Blinelen(&amp;bin);
               s[n-2] = 0;
       }
       return s;
}

void
sendcmd(char *fmt, ...)
{
       va_list args;

       va_start(args, fmt);
       Bvprint(&amp;bout, fmt, args);
       va_end(args);
       Bprint(&amp;bout, "\r\n");
       Bflush(&amp;bout);
}

Resp*
getresp(void)
{
       static Resp response;
       Resp *r;
       char *arg[4], *line;
       int n;

       r = &amp;response;
       memset(r, 0, sizeof(*r));

       r-&gt;line = line = getline();
       if(isdigit(line[0]) &amp;&amp; isdigit(line[1]) &amp;&amp; isdigit(line[2]))
               r-&gt;type = Rstatus;
       else{
               if(line[0] == '.' &amp;&amp; line[1] == 0)
                       r-&gt;type = EOT;
               else if(strncmp(line, "..", 3) == 0){
                       r-&gt;type = Rtext;
                       r-&gt;line[1] = 0;
               }else
                       r-&gt;type = Rtext;
               return r;
       }

       /*
        * Process status response
        */
       utfecpy(r-&gt;msg, r-&gt;msg+sizeof(r-&gt;msg),
               &amp;line[(line[3]==0) ? 3 : 4]);
       n = qtokenize(line, arg, nelem(arg));
       r-&gt;code = atoi(arg[0]);
       switch(r-&gt;code){
       case 110:
       case 111:
       case 150:
       case 152:
               if(n &lt; 2)
                       return r;
               r-&gt;n = atoi(arg[1]);
               break;
       case 151:
               if(n &lt; 4)
                       return r;
               r-&gt;word = arg[1];
               r-&gt;db = arg[2];
               r-&gt;dbdescr = arg[3];
               break;
       }
       return r;
}

void
printtext(void)
{
       Resp *r;

       for(;;){
               r = getresp();
               if(r-&gt;type == EOT)
                       break;
               print("%s\n", r-&gt;line);
       }
}

void
simplecmd(char *fmt, ...)
{
       Resp *r;
       int e;
       va_list args;

       va_start(args, fmt);
       Bvprint(&amp;bout, fmt, args);
       va_end(args);
       Bprint(&amp;bout, "\r\n");
       Bflush(&amp;bout);

       r = getresp();
       e = r-&gt;code/100;
       if(r-&gt;type == Rstatus &amp;&amp; e != 5 &amp;&amp; e != 4){
               printtext();
               getresp(); /* 250 */
       }else
               sysfatal("bad response: %s", r-&gt;msg);
}

void
match(char *db, char *strat, char *query)
{
       Resp *r;
       char *f[2];
       int n, e;

       doquote = needsrcquote;
       quotefmtinstall();

       sendcmd("MATCH %s %s %D", db, strat, query);
       r = getresp();
       if(r-&gt;type == Rstatus &amp;&amp; r-&gt;code == 552){
               print("no match found\n");
               return;
       }
       e = r-&gt;code/100;
       if(r-&gt;type != Rstatus || e == 5 || e == 4)
               sysfatal("bad response: %s", r-&gt;msg);
       for(;;){
               r = getresp();
               if(r-&gt;type == EOT)
                       break;
               n = qtokenize(r-&gt;line, f, nelem(f));
               if(n != 2)
                       sysfatal("bad match output: %s", r-&gt;line);
               print("dic -d %s %q\n", f[0], f[1]);
       }
}

void
define(char *db, char *word)
{
       Resp *r;

       sendcmd("DEFINE %s %D", db, word);
       r = getresp();
       if(r-&gt;type == Rstatus &amp;&amp; r-&gt;code == 552){ /* no match */
               match(database, strat, word);
               return;
       }
       if(r-&gt;type != Rstatus || r-&gt;code != 150)
               sysfatal("bad response: %s", r-&gt;msg);
       for(;;){
               r = getresp();
               if(r-&gt;type == Rstatus &amp;&amp; r-&gt;code != 151)
                       break;
               print("From %s [%s]:\n\n", r-&gt;dbdescr, r-&gt;db);
               printtext();
               print("\n");
       }
       if(r-&gt;code != 250)
               sysfatal("command failed: %s", r-&gt;msg);
}

void
usage(void)
{
       fprint(2, "usage: dic [-DISim] [-a addr] [-d db] [-s strategy] word\n");
       exits("usage");
}

void
main(int argc, char *argv[])
{
       int fd;
       Resp *r;

       fmtinstall('D', Dfmt);

       ARGBEGIN {
       case 'D':
               showdb++;
               break;
       case 'S':
               showstrat++;
               break;
       case 'I':
               showserver++;
               break;
       case 'i':
               showinfo++;
               break;
       case 'a':
               addr = EARGF(usage());
               break;
       case 'd':
               database = EARGF(usage());
               break;
       case 's':
               strat = EARGF(usage());
               /* fall through */
       case 'm':
               domatch++;
               break;
       default:
               usage();
       } ARGEND

       if(!showdb &amp;&amp; !showstrat &amp;&amp; !showserver &amp;&amp; argc &lt; 1)
               usage();

       if(addr)
               fd = dial(netmkaddr(addr, "tcp", "2628"), nil, nil, nil);
       else{
               fd = dial("tcp!localhost!2628", nil, nil, nil);
               if(fd &lt; 0)
                       fd = dial("tcp!dict.org!2628", nil, nil, nil);
       }
       if(fd &lt; 0)
               sysfatal("dial: %r");

       Binit(&amp;bin, fd, OREAD);
       Binit(&amp;bout, fd, OWRITE);

       r = getresp();
       if(r-&gt;type != Rstatus || r-&gt;code != 220)
               sysfatal("server not usable: %s", r-&gt;msg);
       sendcmd("CLIENT plan9/dic");
       getresp();              /* ignore reply */

       if(showdb)
               simplecmd("SHOW DB");
       else if(showstrat)
               simplecmd("SHOW STRAT");
       else if(showserver)
               simplecmd("SHOW SERVER");
       else{
               if(domatch)
                       match(database, strat, argv[0]);
               else if(showinfo)
                       simplecmd("SHOW INFO %s", argv[0]);
               else
                       define(database, argv[0]);
       }
       sendcmd("QUIT");

       Bterm(&amp;bin);
       Bterm(&amp;bout);
       close(fd);

       exits(nil);
}
<!-- BEGIN TAIL -->
</pre>
</td></tr></table>
</td></tr></table>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="line-height: 1.2em; margin-left: 1.00in; text-indent: 0.00in; margin-right: 1.00in; margin-top: 0; margin-bottom: 0; text-align: center;">
<span style="font-size: 10pt"></span></p>
<p style="margin-top: 0; margin-bottom: 0.50in"></p>
<p style="margin-top: 0; margin-bottom: 0.33in"></p>
<center><table border="0"><tr>
<td valign="middle"><a href="http://www.alcatel-lucent.com/"><img border="0" src="/plan9/img/logo_ft.gif" alt="Bell Labs" />
</a></td>
<td valign="middle"><a href="http://www.opensource.org"><img border="0" alt="OSI certified" src="/plan9/img/osi-certified-60x50.gif" />
</a></td>
<td><img style="padding-right: 45px;" alt="Powered by Plan 9" src="/plan9/img/power36.gif" />
</td>
</tr></table></center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center>
<span style="font-size: 10pt">(<a href="/plan9/">Return to Plan 9 Home Page</a>)</span>
</center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center><font size=-1>
<span style="font-size: 10pt"><a href="http://www.lucent.com/copyright.html">Copyright</a></span>
<span style="font-size: 10pt">© 2009 Alcatel-Lucent.</span>
<span style="font-size: 10pt">All Rights Reserved.</span>
<br />
<span style="font-size: 10pt">Comments to</span>
<span style="font-size: 10pt"><a href="mailto:[email protected]">[email protected]</a>.</span>
</font></center>
</body>
</html>