1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-23 15:07:03 +01:00

* gpgkeys_hkp.c, Makefile.am: Convert over to using iobufs.

This commit is contained in:
David Shaw 2002-08-27 19:11:36 +00:00
parent fd79d4ee61
commit 813fa3f98b
3 changed files with 263 additions and 375 deletions

View File

@ -1,5 +1,7 @@
2002-08-27 David Shaw <dshaw@jabberwocky.com> 2002-08-27 David Shaw <dshaw@jabberwocky.com>
* gpgkeys_hkp.c, Makefile.am: Convert over to using iobufs.
* gpgkeys_hkp.c (http_get, http_post): Use CRLF for line endings. * gpgkeys_hkp.c (http_get, http_post): Use CRLF for line endings.
* gpgkeys_hkp.c: Include util.h on RISC OS as per Stefan. Include * gpgkeys_hkp.c: Include util.h on RISC OS as per Stefan. Include

View File

@ -31,4 +31,4 @@ libexec_SCRIPTS = @GPGKEYS_MAILTO@
noinst_SCRIPTS = gpgkeys_test noinst_SCRIPTS = gpgkeys_test
gpgkeys_ldap_LDADD = @LDAPLIBS@ @NETLIBS@ gpgkeys_ldap_LDADD = @LDAPLIBS@ @NETLIBS@
gpgkeys_hkp_LDADD = @NETLIBS@ gpgkeys_hkp_LDADD = ../util/libutil.a @NETLIBS@

View File

@ -20,21 +20,16 @@
#include <config.h> #include <config.h>
#include <stdio.h> #include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <unistd.h>
#include <string.h> #include <string.h>
#include <errno.h>
#include <ctype.h> #include <ctype.h>
#include <time.h>
#include <stdlib.h> #include <stdlib.h>
#include "keyserver.h" #include <errno.h>
#include <unistd.h>
#ifdef __riscos__ #include "iobuf.h"
#include "http.h"
#include "memory.h"
#include "util.h" #include "util.h"
#endif #include "keyserver.h"
#define GET 0 #define GET 0
#define SEND 1 #define SEND 1
@ -42,7 +37,7 @@
#define MAX_LINE 80 #define MAX_LINE 80
int verbose=0,include_disabled=0,include_revoked=0; int verbose=0,include_disabled=0,include_revoked=0;
char *basekeyspacedn=NULL; unsigned int http_flags=0;
char host[80]={'\0'}; char host[80]={'\0'};
char portstr[10]={'\0'}; char portstr[10]={'\0'};
FILE *input=NULL,*output=NULL,*console=NULL,*server=NULL; FILE *input=NULL,*output=NULL,*console=NULL,*server=NULL;
@ -53,145 +48,46 @@ struct keylist
struct keylist *next; struct keylist *next;
}; };
#ifndef HAVE_HSTRERROR static int
const char *hstrerror(int err) urlencode_filter( void *opaque, int control,
IOBUF a, byte *buf, size_t *ret_len)
{ {
if(err<0) size_t size = *ret_len;
return "Resolver internal error"; int rc=0;
switch(err) if( control == IOBUFCTRL_FLUSH ) {
{ const byte *p;
case 0: for(p=buf; size; p++, size-- ) {
return "Resolver Error 0 (no error)"; if( isalnum(*p) || *p == '-' )
iobuf_put( a, *p );
case HOST_NOT_FOUND: else if( *p == ' ' )
return "Unknown host"; /* 1 HOST_NOT_FOUND */ iobuf_put( a, '+' );
else {
case TRY_AGAIN: char numbuf[5];
return "Host name lookup failure"; /* 2 TRY_AGAIN */ sprintf(numbuf, "%%%02X", *p );
iobuf_writestr(a, numbuf );
case NO_RECOVERY:
return "Unknown server error"; /* 3 NO_RECOVERY */
case NO_ADDRESS:
return "No address associated with name"; /* 4 NO_ADDRESS */
default:
return "Unknown resolver error";
} }
}
#endif /* !HAVE_HSTRERROR */
int http_connect(const char *http_host,unsigned short port)
{
int sock=-1;
struct hostent *ent;
struct sockaddr_in addr;
sock=socket(AF_INET,SOCK_STREAM,0);
if(sock==-1)
{
fprintf(console,"gpgkeys: internal socket error: %s\n",strerror(errno));
goto fail;
} }
ent=gethostbyname(http_host);
if(ent==NULL)
{
fprintf(console,"gpgkeys: DNS error: %s\n",hstrerror(h_errno));
goto fail;
} }
else if( control == IOBUFCTRL_DESC )
addr.sin_family=AF_INET; *(char**)buf = "urlencode_filter";
addr.sin_addr.s_addr=*(int *)ent->h_addr_list[0]; return rc;
addr.sin_port=htons(port?port:11371);
if(connect(sock,(struct sockaddr *)&addr,sizeof(addr))==-1)
{
fprintf(console,"gpgkeys: unable to contact keyserver: %s\n",
strerror(errno));
goto fail;
}
server=fdopen(sock,"r+");
if(server==NULL)
{
fprintf(console,"gpgkeys: unable to fdopen socket: %s\n",
strerror(errno));
goto fail;
}
if(verbose>3)
fprintf(console,"gpgkeys: HKP connect to %s:%d\n",
http_host,port?port:11371);
return 0;
fail:
if(sock>-1)
close(sock);
return -1;
}
void http_disconnect(void)
{
if(verbose>3)
fprintf(console,"gpgkeys: HKP disconnect from %s\n",host);
fclose(server);
}
int http_get(const char *op,const char *search)
{
fprintf(server,"GET /pks/lookup?op=%s&search=%s HTTP/1.0\r\n\r\n",op,search);
if(verbose>2)
fprintf(console,"gpgkeys: HTTP GET /pks/lookup?op=%s&search=%s HTTP/1.0\n",
op,search);
return 0;
}
int http_post(const char *data)
{
char line[MAX_LINE];
int result;
fprintf(server,
"POST /pks/add HTTP/1.0\r\n"
"Content-type: application/x-www-form-urlencoded\n"
"Content-Length: %d\r\n\r\n%s",strlen(data),data);
if(verbose>2)
fprintf(console,
"gpgkeys: HTTP POST /pks/add HTTP/1.0\n"
"gpgkeys: Content-type: application/x-www-form-urlencoded\n"
"gpgkeys: Content-Length: %d\n\n",strlen(data));
/* Now wait for a response */
while(fgets(line,MAX_LINE,server)!=NULL)
if(sscanf(line,"HTTP/%*f %d OK",&result)==1)
return result;
return -1;
} }
/* Returns 0 on success, -1 on failure, and 1 on eof */ /* Returns 0 on success, -1 on failure, and 1 on eof */
int send_key(void) int send_key(void)
{ {
int err,gotit=0,keylen,maxlen,ret=-1; int rc,gotit=0,ret=-1;
char keyid[17],line[MAX_LINE],*key; char keyid[17];
char *request;
struct http_context hd;
unsigned int status;
IOBUF temp = iobuf_temp();
char line[MAX_LINE];
key=strdup("keytext="); request=m_alloc(strlen(host)+100);
if(key==NULL)
{
fprintf(console,"gpgkeys: unable to allocate for key\n");
goto fail;
}
maxlen=keylen=strlen(key); iobuf_push_filter(temp,urlencode_filter,NULL);
/* Read and throw away stdin until we see the BEGIN */ /* Read and throw away stdin until we see the BEGIN */
@ -213,7 +109,7 @@ int send_key(void)
/* Now slurp up everything until we see the END */ /* Now slurp up everything until we see the END */
while(fgets(line,MAX_LINE,input)!=NULL) while(fgets(line,MAX_LINE,input))
if(sscanf(line,"KEY %16s END\n",keyid)==1) if(sscanf(line,"KEY %16s END\n",keyid)==1)
{ {
gotit=1; gotit=1;
@ -221,39 +117,7 @@ int send_key(void)
} }
else else
{ {
char *c=line; iobuf_writestr(temp,line);
while(*c!='\0')
{
if(maxlen-keylen<4)
{
maxlen+=1024;
key=realloc(key,maxlen);
if(key==NULL)
{
fprintf(console,"gpgkeys: unable to reallocate for key\n");
goto fail;
}
}
if(isalnum(*c) || *c=='-')
{
key[keylen++]=*c;
key[keylen]='\0';
}
else if(*c==' ')
{
key[keylen++]='+';
key[keylen]='\0';
}
else
{
sprintf(&key[keylen],"%%%02X",*c);
keylen+=3;
}
c++;
}
} }
if(!gotit) if(!gotit)
@ -262,29 +126,62 @@ int send_key(void)
goto fail; goto fail;
} }
err=http_post(key); iobuf_flush_temp(temp);
if(err!=200)
sprintf(request,"x-hkp://%s%s%s/pks/add",
host,portstr[0]?":":"",portstr[0]?portstr:"");
rc=http_open(&hd,HTTP_REQ_POST,request,http_flags);
if(rc)
{ {
fprintf(console,"gpgkeys: remote server returned error %d\n",err); fprintf(console,"gpgkeys: unable to connect to `%s'\n",host);
goto fail;
}
sprintf(request,"Content-Length: %u\n",
(unsigned)iobuf_get_temp_length(temp)+9);
iobuf_writestr(hd.fp_write,request);
http_start_data(&hd);
iobuf_writestr(hd.fp_write,"keytext=");
iobuf_write(hd.fp_write,
iobuf_get_temp_buffer(temp),iobuf_get_temp_length(temp));
iobuf_put(hd.fp_write,'\n');
rc=http_wait_response(&hd,&status);
if(rc)
{
fprintf(console,"gpgkeys: error sending to `%s': %s\n",
host,g10_errstr(rc));
goto fail;
}
if((status/100)!=2)
{
fprintf(console,"gpgkeys: remote server returned error %d\n",status);
fprintf(output,"KEY %s FAILED\n",keyid);
goto fail; goto fail;
} }
ret=0; ret=0;
fail: fail:
m_free(request);
free(key); iobuf_close(temp);
http_close(&hd);
if(ret!=0)
fprintf(output,"KEY %s FAILED\n",keyid);
return ret; return ret;
} }
int get_key(char *getkey) int get_key(char *getkey)
{ {
int err,gotit=0; int rc,gotit=0;
char search[29],line[MAX_LINE]; unsigned int maxlen=1024,buflen=0;
char search[29];
char *request;
struct http_context hd;
byte *line=NULL;
/* Build the search string. HKP only uses the short key IDs. */ /* Build the search string. HKP only uses the short key IDs. */
@ -325,15 +222,25 @@ int get_key(char *getkey)
fprintf(console,"gpgkeys: requesting key 0x%s from hkp://%s%s%s\n", fprintf(console,"gpgkeys: requesting key 0x%s from hkp://%s%s%s\n",
getkey,host,portstr[0]?":":"",portstr[0]?portstr:""); getkey,host,portstr[0]?":":"",portstr[0]?portstr:"");
err=http_get("get",search); request=m_alloc(strlen(host)+100);
if(err!=0)
{
fprintf(console,"gpgkeys: HKP fetch error: %s\n",strerror(errno));
fprintf(output,"KEY 0x%s FAILED\n",getkey);
return -1;
}
while(fgets(line,MAX_LINE,server)) sprintf(request,"x-hkp://%s%s%s/pks/lookup?op=get&search=%s",
host,
portstr[0]?":":"",
portstr[0]?portstr:"", search);
fprintf(console,"request is \"%s\"\n",request);
rc=http_open_document(&hd,request,http_flags);
if(rc!=0)
{
fprintf(console,"gpgkeys: HKP fetch error: %s\n",
rc==G10ERR_NETWORK?strerror(errno):g10_errstr(rc));
fprintf(output,"KEY 0x%s FAILED\n",getkey);
}
else
{
while(iobuf_read_line(hd.fp_read,&line,&buflen,&maxlen))
{ {
if(gotit) if(gotit)
{ {
@ -352,76 +259,11 @@ int get_key(char *getkey)
gotit=1; gotit=1;
} }
} }
return 0;
}
void print_quoted(FILE *stream,char *string,char delim)
{
while(*string)
{
if(*string==delim)
fprintf(stream,"\\x%02X",*string);
else
fputc(*string,stream);
string++;
}
}
void append_quoted(char *buffer,char *string,char delim)
{
while(*buffer)
buffer++;
while(*string)
{
if(*string==delim)
{
sprintf(buffer,"\\x%02X",*string);
buffer+=4;
}
else
*buffer=*string;
buffer++;
string++;
} }
*buffer='\0'; m_free(request);
}
unsigned int scan_isodatestr( const char *string )
{
int year, month, day;
struct tm tmbuf;
time_t stamp;
int i;
if( strlen(string) != 10 || string[4] != '-' || string[7] != '-' )
return 0; return 0;
for( i=0; i < 4; i++ )
if( !isdigit(string[i]) )
return 0;
if( !isdigit(string[5]) || !isdigit(string[6]) )
return 0;
if( !isdigit(string[8]) || !isdigit(string[9]) )
return 0;
year = atoi(string);
month = atoi(string+5);
day = atoi(string+8);
/* some basic checks */
if( year < 1970 || month < 1 || month > 12 || day < 1 || day > 31 )
return 0;
memset( &tmbuf, 0, sizeof tmbuf );
tmbuf.tm_mday = day;
tmbuf.tm_mon = month-1;
tmbuf.tm_year = year - 1900;
tmbuf.tm_isdst = -1;
stamp = mktime( &tmbuf );
if( stamp == (time_t)-1 )
return 0;
return stamp;
} }
/* Remove anything <between brackets> and de-urlencode in place. Note /* Remove anything <between brackets> and de-urlencode in place. Note
@ -495,20 +337,57 @@ dehtmlize(char *line)
} }
} }
static int
write_quoted(IOBUF a, const char *buf, char delim)
{
char quoted[5];
sprintf(quoted,"\\x%02X",delim);
while(*buf)
{
if(*buf==delim)
{
if(iobuf_writestr(a,quoted))
return -1;
}
else if(*buf=='\\')
{
if(iobuf_writestr(a,"\\x5c"))
return -1;
}
else
{
if(iobuf_writebyte(a,*buf))
return -1;
}
buf++;
}
return 0;
}
/* pub 2048/<a href="/pks/lookup?op=get&search=0x3CB3B415">3CB3B415</a> 1998/04/03 David M. Shaw &lt;<a href="/pks/lookup?op=get&search=0x3CB3B415">dshaw@jabberwocky.com</a>&gt; */ /* pub 2048/<a href="/pks/lookup?op=get&search=0x3CB3B415">3CB3B415</a> 1998/04/03 David M. Shaw &lt;<a href="/pks/lookup?op=get&search=0x3CB3B415">dshaw@jabberwocky.com</a>&gt; */
/* Luckily enough, both the HKP server and NAI HKP interface to their /* Luckily enough, both the HKP server and NAI HKP interface to their
LDAP server are close enough in output so the same function can LDAP server are close enough in output so the same function can
parse them both. */ parse them both. */
int parse_hkp_index(char *line,char **buffer) static int
parse_hkp_index(IOBUF buffer,char *line)
{ {
static int open=0,revoked=0; static int open=0,revoked=0;
static char *key=NULL,*uid=NULL,*type=NULL; static char *key=NULL,*type=NULL;
static unsigned int bits,createtime; #ifdef __riscos__
static char *uid=NULL;
#else
static unsigned char *uid=NULL;
#endif
static u32 bits,createtime;
int ret=0; int ret=0;
/* printf("Open %d, LINE: %s, uid: %s\n",open,line,uid); */ /* printf("Open %d, LINE: \"%s\", uid: %s\n",open,line,uid); */
dehtmlize(line); dehtmlize(line);
@ -519,11 +398,11 @@ int parse_hkp_index(char *line,char **buffer)
response. This only complains about problems within the key response. This only complains about problems within the key
section itself. Headers and footers should not matter. */ section itself. Headers and footers should not matter. */
if(open && line[0]!='\0' && if(open && line[0]!='\0' &&
strncasecmp(line,"pub ",4)!=0 && ascii_memcasecmp(line,"pub ",4)!=0 &&
strncasecmp(line," ",4)!=0) ascii_memcasecmp(line," ",4)!=0)
{ {
free(key); m_free(key);
free(uid); m_free(uid);
fprintf(console,"gpgkeys: this keyserver is not fully HKP compatible\n"); fprintf(console,"gpgkeys: this keyserver is not fully HKP compatible\n");
return -1; return -1;
} }
@ -535,43 +414,23 @@ int parse_hkp_index(char *line,char **buffer)
if(!(revoked && !include_revoked)) if(!(revoked && !include_revoked))
{ {
char intstr[11],*buf; char intstr[11];
buf=realloc(*buffer, if(key)
(*buffer?strlen(*buffer):0)+ write_quoted(buffer,key,':');
(strlen(key)*4)+ iobuf_writestr(buffer,":");
1+ write_quoted(buffer,uid,':');
(strlen(uid)*4) iobuf_writestr(buffer,":");
+1 iobuf_writestr(buffer,revoked?"1:":":");
+2
+10
+3
+3
+1
+10
+1
+1
+20);
if(buf)
*buffer=buf;
else
return -1;
append_quoted(*buffer,key,':');
append_quoted(*buffer,":",0);
append_quoted(*buffer,uid,':');
append_quoted(*buffer,":",0);
append_quoted(*buffer,revoked?"1:":":",0);
sprintf(intstr,"%u",createtime); sprintf(intstr,"%u",createtime);
append_quoted(*buffer,intstr,':'); write_quoted(buffer,intstr,':');
append_quoted(*buffer,":::",0); iobuf_writestr(buffer,":::");
if(type) if(type)
append_quoted(*buffer,type,':'); write_quoted(buffer,type,':');
append_quoted(*buffer,":",0); iobuf_writestr(buffer,":");
sprintf(intstr,"%u",bits); sprintf(intstr,"%u",bits);
append_quoted(*buffer,intstr,':'); write_quoted(buffer,intstr,':');
append_quoted(*buffer,"\n",0); iobuf_writestr(buffer,"\n");
ret=1; ret=1;
} }
@ -579,14 +438,14 @@ int parse_hkp_index(char *line,char **buffer)
if(strncmp(line," ",4)!=0) if(strncmp(line," ",4)!=0)
{ {
revoked=0; revoked=0;
free(key); m_free(key);
free(uid); m_free(uid);
uid=NULL; uid=NULL;
open=0; open=0;
} }
} }
if(strncasecmp(line,"pub ",4)==0) if(ascii_memcasecmp(line,"pub ",4)==0)
{ {
char *tok,*temp; char *tok,*temp;
@ -609,12 +468,9 @@ int parse_hkp_index(char *line,char **buffer)
tok=strsep(&line," "); tok=strsep(&line," ");
if(tok==NULL) if(tok==NULL)
{
key=strdup("00000000");
return ret; return ret;
}
key=strdup(tok); key=m_strdup(tok);
tok=strsep(&line," "); tok=strsep(&line," ");
if(tok==NULL) if(tok==NULL)
@ -637,7 +493,7 @@ int parse_hkp_index(char *line,char **buffer)
{ {
if(line==NULL) if(line==NULL)
{ {
uid=strdup("Key index corrupted"); uid=m_strdup("Key index corrupted");
return ret; return ret;
} }
@ -653,7 +509,7 @@ int parse_hkp_index(char *line,char **buffer)
return ret; return ret;
} }
uid=strdup(line); uid=m_strdup(line);
} }
return ret; return ret;
@ -661,20 +517,15 @@ int parse_hkp_index(char *line,char **buffer)
int search_key(char *searchkey) int search_key(char *searchkey)
{ {
int ret=-1,err,count=0; int max=0,len=0,ret=-1,rc;
char *search,*request,*buffer=NULL; struct http_context hd;
char line[1024]; char *search=NULL,*request=searchkey;
int max,len; byte *line=NULL;
fprintf(output,"SEARCH %s BEGIN\n",searchkey); fprintf(output,"SEARCH %s BEGIN\n",searchkey);
/* Build the search string. It's going to need url-encoding. */ /* Build the search string. It's going to need url-encoding. */
max=0;
len=0;
search=NULL;
request=searchkey;
while(*request!='\0') while(*request!='\0')
{ {
if(max-len<3) if(max-len<3)
@ -698,34 +549,71 @@ int search_key(char *searchkey)
search[len]='\0'; search[len]='\0';
request=m_alloc(strlen(host)+100+strlen(search));
sprintf(request,"x-hkp://%s%s%s/pks/lookup?op=index&search=%s",
host,portstr[0]?":":"",portstr[0]?portstr:"",search);
if(verbose>2) if(verbose>2)
fprintf(console,"gpgkeys: HKP search for: %s\n",search); fprintf(console,"gpgkeys: HKP search for: %s\n",search);
fprintf(console,("gpgkeys: searching for \"%s\" from HKP server %s\n"), fprintf(console,("gpgkeys: searching for \"%s\" from HKP server %s\n"),
searchkey,host); searchkey,host);
http_get("index",search); rc=http_open_document(&hd,request,http_flags);
if(rc)
free(search);
while(fgets(line,1024,server))
{ {
err=parse_hkp_index(line,&buffer); fprintf(console,"gpgkeys: can't search keyserver `%s': %s\n",
if(err==-1) host,rc==G10ERR_NETWORK?strerror(errno):g10_errstr(rc));
goto fail; }
else
{
unsigned int maxlen=1024,buflen=0;
int count=1;
int ret;
IOBUF buffer;
count+=err; buffer=iobuf_temp();
rc=1;
while(rc!=0)
{
/* This is a judgement call. Is it better to slurp up all
the results before prompting the user? On the one hand,
it probably makes the keyserver happier to not be blocked
on sending for a long time while the user picks a key.
On the other hand, it might be nice for the server to be
able to stop sending before a large search result page is
complete. */
rc=iobuf_read_line(hd.fp_read,&line,&buflen,&maxlen);
ret=parse_hkp_index(buffer,line);
if(ret==-1)
break;
if(rc!=0)
count+=ret;
} }
fprintf(output,"COUNT %d\n%s",count,buffer); http_close(&hd);
/* fprintf(output,"COUNT -1\n%s",buffer); */
count--;
if(ret>-1)
fprintf(output,"COUNT %d\n%s",count,iobuf_get_temp_buffer(buffer));
fprintf(output,"SEARCH %s END\n",searchkey); fprintf(output,"SEARCH %s END\n",searchkey);
ret=0; iobuf_close(buffer);
m_free(line);
fail: ret=0;
free(buffer); }
m_free(request);
free(search);
return ret; return ret;
} }
@ -866,11 +754,20 @@ int main(int argc,char *argv[])
else else
include_revoked=1; include_revoked=1;
} }
else if(strcasecmp(start,"honor-http-proxy")==0 || else if(strcasecmp(start,"honor-http-proxy")==0)
strcasecmp(start,"broken-http-proxy")==0)
{ {
fprintf(stderr,"gpgkeys: HKP does not currently support %s\n", if(no)
start); http_flags&=~HTTP_FLAG_TRY_PROXY;
else
http_flags|=HTTP_FLAG_TRY_PROXY;
}
else if(strcasecmp(start,"broken-http-proxy")==0)
{
if(no)
http_flags&=~HTTP_FLAG_NO_SHUTDOWN;
else
http_flags|=HTTP_FLAG_NO_SHUTDOWN;
} }
continue; continue;
@ -967,13 +864,9 @@ int main(int argc,char *argv[])
while(keyptr!=NULL) while(keyptr!=NULL)
{ {
http_connect(host,port);
if(get_key(keyptr->str)==-1) if(get_key(keyptr->str)==-1)
failed++; failed++;
http_disconnect();
keyptr=keyptr->next; keyptr=keyptr->next;
} }
break; break;
@ -984,11 +877,9 @@ int main(int argc,char *argv[])
do do
{ {
http_connect(host,port);
ret2=send_key(); ret2=send_key();
if(ret2==-1) if(ret2==-1)
failed++; failed++;
http_disconnect();
} }
while(ret2!=1); while(ret2!=1);
} }
@ -1026,16 +917,12 @@ int main(int argc,char *argv[])
/* Nail that last space */ /* Nail that last space */
searchkey[strlen(searchkey)-1]='\0'; searchkey[strlen(searchkey)-1]='\0';
http_connect(host,port);
if(search_key(searchkey)==-1) if(search_key(searchkey)==-1)
{ {
fprintf(output,"SEARCH %s FAILED\n",searchkey); fprintf(output,"SEARCH %s FAILED\n",searchkey);
failed++; failed++;
} }
http_disconnect();
free(searchkey); free(searchkey);
} }
@ -1046,7 +933,6 @@ int main(int argc,char *argv[])
ret=KEYSERVER_OK; ret=KEYSERVER_OK;
fail: fail:
while(keylist!=NULL) while(keylist!=NULL)
{ {
struct keylist *current=keylist; struct keylist *current=keylist;