/**********************************************************************
* makegdbm.c May 1999
* Horms horms@verge.net.au
*
* Create a gdbm file from a : delimited flat file read from stdin
*
* perdition
* Mail retrieval proxy server
* Copyright (C) 1999-2005 Horms
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
* 02111-1307 USA
*
**********************************************************************/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <stdlib.h>
#include <ctype.h>
#include "options.h"
#ifdef DMALLOC
#include <dmalloc.h>
#endif
extern int errno;
#define MAX_LINE_LENGTH 4096
#define FIELD_DELIMITER ':'
void makegdbm_new(const makegdbm_options_t options);
void makegdbm_undo(const makegdbm_options_t options);
int main(int argc, char **argv){
makegdbm_options_t options;
options=makegdbm_options(argc, argv);
if(options.undo){
makegdbm_undo(options);
}
else{
makegdbm_new(options);
}
return(0);
}
/**********************************************************************
* makegdbm_new
* Create a nrew gdbm file from a flat file read from stdin
* pre: options: options structure specifying what to do
* post: gdbm file created and populated
**********************************************************************/
void makegdbm_new(const makegdbm_options_t options){
GDBM_FILE dbf;
datum key;
datum content;
int i;
char line[MAX_LINE_LENGTH];
int status;
int blank;
int lineno;
if((dbf=gdbm_open(options.mapname, 0, GDBM_NEWDB|GDBM_FAST, 0644, 0))==NULL){
fprintf(
stderr,
"makegdbm_new: gdbm_open: %s: %s\n",
gdbm_strerror(gdbm_errno),
strerror(errno)
);
exit(-1);
}
/*Warning: This loop contains some of the worst code ever written*/
status=0;
lineno=0;
while(fgets(line, MAX_LINE_LENGTH, stdin)!=NULL){
blank=1;
key.dptr=line;
key.dsize=-1;
content.dsize=-1;
for(i=0;i<MAX_LINE_LENGTH;i++){
if(blank && *(line+i)!=' ' && *(line+i)!='\t'&& *(line+i)!='\n'){
blank=0;
}
if(key.dsize==-1 && *(line+i)==FIELD_DELIMITER){
key.dsize=i;
content.dptr=line+i+1;
continue;
}
if(content.dsize==-1 && *(line+i)=='\n'){
content.dsize=i-key.dsize-1;
lineno++;
break;
}
}
fflush(NULL);
if(!blank && !status && (key.dsize<1 || content.dsize<1)){
status=1;
}
if(status){
if(*(line+i)=='\n'){
fprintf(stderr, "makegdbm: invalid input on line: %d \n", lineno);
status=0;
}
continue;
}
if(blank){
continue;
}
if(gdbm_store(dbf, key, content, GDBM_INSERT)>0){
*(key.dptr+key.dsize)='\0';
fprintf(stderr, "makegdbm: dupicate key: %s\n", key.dptr);
}
}
gdbm_close(dbf);
}
/**********************************************************************
* makegdbm_undo
* print out a : delimited flat file to stdout of a gdbm file
* pre: options: options structure sepcifying what to do
* post: glat file output to stdout
**********************************************************************/
void makegdbm_undo(makegdbm_options_t options){
GDBM_FILE dbf;
datum this_key;
datum next_key;
datum content;
if((dbf=gdbm_open(options.mapname, 0, GDBM_READER, 0644, 0))==NULL){
fprintf(
stderr,
"makegdbm_undo: gdbm_open: %s: %s\n",
gdbm_strerror(gdbm_errno),
strerror(errno)
);
exit(-1);
}
this_key=gdbm_firstkey(dbf);
if(this_key.dptr==NULL){
fprintf(stderr, "makegdbm: makegdbm_undo: no first key\n");
gdbm_close(dbf);
exit(0);
}
while(1){
fwrite(this_key.dptr, 1, this_key.dsize, stdout);
fwrite(":", 1, 2, stdout);
content=gdbm_fetch(dbf,this_key);
if(content.dptr!=NULL){
fwrite(content.dptr, 1, content.dsize, stdout);
}
fwrite("\n", 1, 2, stdout);
next_key=gdbm_nextkey(dbf,this_key);
if(next_key.dptr==NULL){
break;
}
free(this_key.dptr);
this_key=next_key;
}
gdbm_close(dbf);
}
syntax highlighted by Code2HTML, v. 0.9.1