djl@dplace.UUCP (Dave Lampe) (04/17/89)
This is the first release of my index card database program. It is only usable on Sys V since it requires terminfo, not termcap. Dave Lampe {ames | lll-tis | sun | pyramid}!pacbell!dplace!djl (415) 455-1571 (H) (408) 986-1820 (W) ---- Cut Here and unpack ---- #!/bin/sh # shar: Shell Archiver (v1.22) # # This is part 1 of a multipart archive # do not concatenate these parts, unpack them in order with /bin/sh # # Run the following text with /bin/sh to create: # README # cardfile.1 # Makefile # cardfile.c # add.c # change.c # common.c # compress.c # define.c # delete.c # dumpdb.c # extract.c # find.c # findrcds.c # getkey.c # keymatch.c # maint.c # menu.c # printdb.c # putrcd.c # rawio.c # rbuildak.c # screen.c # setupkeys.c # updak.c # ascii.h # cardfile.h # library.def # library.db # library.ak0 # library.ak1 # library.ak2 # library.ak3 # if test -r s2_seq_.tmp then echo "Must unpack archives in sequence!" next=`cat s2_seq_.tmp`; echo "Please unpack part $next next" exit 1; fi echo "x - extracting README (Text)" sed 's/^X//' << 'SHAR_EOF' > README && XThis program is a simple screen oriented database manager. It does have Xsome rudimentary file formatting and printing capabilities. I use it as a Xlibrary catalog. The metaphor is a stack of index cards with fields and Xsubfields on them. To explain subfields, think of a book with multiple Xauthors. Each author must be treated equally and there may be many authors. XIf you search for any book written by "John Smith" you want to find it Xwhether he is the only author or the fifteenth author. X XIt runs only on System V because it uses the terminfo capabilities. Some Xof the source may look a little strange because I wrote it originally to Xrun under CPM. I still have the CPM version if anyone wants it. X XThere are two things that need to be done yet. It needs to be converted Xto run under BSD and the indexing needs to be converted from a sequential Xsearch to something faster such as a B tree. Even with a sequential search, Xthe speed is acceptable on a database of about a 500K. X XIncluded in the package is a piece of the catalog file for my library so Xthat you can get the feeling for what it does. After you make it, Xexecute "cardfile library" and then play. X XIf anyone finds any bugs (What? Bugs in MY code? :-) ), or makes any Ximprovements, please let me know. X XDave Lampe X{ames | lll-tis | sun | pyramid}!pacbell!dplace!djl X(415) 455-1571 (H) X(408) 986-1820 (W) SHAR_EOF chmod 0644 README || echo "restore of README fails" echo "x - extracting cardfile.1 (Text)" sed 's/^X//' << 'SHAR_EOF' > cardfile.1 && X.\ "@(#)cardfile.1 2.2 Delta Date 4/16/89 ExtrDate 4/16/89 "; X.if t .po 1i X.TH CARDFILE 1 djl X.SH NAME Xcardfile \- simple index card database X.SH SYNOPSIS X\fBcardfile [-r] file\fP X.SH DESCRIPTION X.I Cardfile Xis a simple screen oriented index card database program. XIf X.I file Xdoes not exist, X.I cardfile Xwill prompt for a definition of the fields to appear in each record. XEach field has a name, a maximum length, Xa flag to determine if it is required or not, Xa flag to determine if it is an index field X(whether or not it can be searched for), Xand if multiple values may appear in the field, Xwhat character will separate the values. XFor example a book may have multiple authors separated by semicolons. XWhen all fields have been defined, Xreturn a blank screen and X.I cardfile Xwill create all necessary files and exit. XReenter X.I cardfile Xwith the same filename and you will be able to start adding the data. X.P XThe \fB-r\fP flag tells \fIcardfile\fP to open thje database readonly. XYou must use this flag to retrieve data from a database on which Xyou do not have write permission. XWhen \fIcardfile\fP starts, the main menu is the first screen presented. XYou may use the cursor keys or the tab keys to move. XWhen the cursor is next to the option desired the return key Xwill select it. XThe options on the main menu are: X.SS Find XThe \fIFIND\fP option is used to retrieve data from the file. XYou will be presented with a screen containing all the fields flagged Xas index fields. XEnter the value for which you wish to search. XIf the value is enclosed in quotes (\fB"\fP) it must match exactly. XOtherwise, case is ignored and asterisk (\fB*\fP) can be used at the Xend of a string to match anything, Xi.e. \fIFarmer*\fP will match "Farmer,P.J." or "farmer,philip", Xor "FARMER9999". XSearch values within a field may be connected by "\fB&\fP" for X\fBand\fP or "\fB|\fP" for \fBor\fP, Xi.e. if you enter "1986|1987" any record with a value of 1986 Xor 1987 will be selected X("and" only makes sense if multiple values are allowed in the field.) XIf values are entered in multiple fields, a record must satisfy Xboth criteria to be selected. X.P XWhen all values have been input, Xhit return and the database will be searched. XThe selected records will be displayed on the screen one at a time. XReturn will display the next record, XCtrl-B will move backwards through the list, Xand Esc will abort the display. X.SS Add XThe \fIADD\fP option is used to add a new record to the database. XYou will be presented with a screen with all the fields defined Xon the record. XThe maximum size of each field will appear after the field name. XWhen all the data has been entered for a record, Xreturn will save the data and blank the fields. XThe data is not actually written to the file until Xyou leave the \fIadd\fP screen. XTo leave the \fIADD\fP screen simply return a blank screen. X.SS Change XThe \fICHANGE\fP option starts out like \fIFIND\fP but Xwhen the selected records are displayed, XCtrl-C will display an \fIADD\fP screen with the data from Xthe selected record as the initial value of each field. XChange the data as desired and then hit return to write the data Xand display the next selected record. X.SS Delete XThe \fIDELETE\fP option starts out like \fIFIND\fP but Xwhen the selected records are displayed, XCtrl-D will delete the record from the database. XThe record is not physically deleted from the file until X\fICOMPRESS\fP is run, the record is only marked and ignored. X.SS Print XThe \fIPRINT\fP option is used to format and print the database Xor a subset of the database. XThe first screen asks for the output format, the extracted file Xif any (see \fIExtract\fP on the \fIMaint\fP menu), Xthe output file, and the output width. X.if n .P X.if t .bp XAny character in the output format will be printed as entered Xexcept for \fB%\fP sequences. XThe recognized sequences are: X.nf X.ta 0.5i,1.5i X %N The contents of field N X %N(form) The contents of field N in \fIprintf\fP(3) X "%form" format, i.e. "%1(%-20s)" will print X field 1 left justified in a 20 character field. X %n Print a new-line. X %t Output a tab character. X %f Output a form feed. X %t(NN) Advance to column NN. X %% Print a %. X.fi XIf the extract name field is left blank, the entire database will be dumped. XThe output file may be an ordinary file, Xor it may be specified as "| \fIcommand\fP" in which case X\fIcommand\fP will be started and the print piped to it. XIf the output width is missing, it defaults to 80. X.SS Exit XThe \fIEXIT\fP option will return to UNIX. X.SS Maint XThe \fIMAINT\fP option will generate a submenu of infrequently used actions. XThe actions available from the maintenance menu are: X.br X.po +0.5i X.ll -0.5i X.SS Dump XThe \fIDUMP\fP option of the maintenance menu Xis used to dump all records in the database and in the index files Xto the printer. XThe records are not formated, they are printed exactly as in the database. X.SS Compress XThe \fICOMPRESS\fP option of the maintenance menu Xwill reclaim the space taken by records marked as deleted Xand then will rebuild the index files. XAfter \fICOMPRESS\fP has been run a record can no longer be recovered. X.SS "Rebuild AK's" XThe \fIREBUILD AK's\fP option of the maintenance menu Xwill recreate the index files from the main database file. XThis is necessary if the main database has been changed by any means Xother than this program. X.SS Extract XThe \fIEXTRACT\fP option of the maintenance menu Xis used to write selected records from the Xdatabase to another file. XYou are first asked for the name of the output file and then Xpresented with a screen like \fIFIND\fP to select the records to Xbe extracted. X.SS Exit XThe \fIEXIT\fP option of the maintenance menu will return to the main menu. X.br X.po -0.5i X.ll +0.5i X.SH "CONTROL KEYS" XThe keys used to control the screens are defined in \fItermcap\fP(4). X.sp X.nf X.ta 0.5i,1.5i,3i X TERMCAP UNIX-PC ACTION X CAP-NAME KEY X.sp X kcuf1 \(-> move right or to next field if at X end of field X.if t .sp 0.5 X kcub1 \(<- move left or to previous field if X at start of field X.if t .sp 0.5 X kbs Back same as kcub1 X Space X.if t .sp 0.5 X ht Tab move to start of next field X.if t .sp 0.5 X kcud1 Down same as ht X Arrow X.if t .sp 0.5 X kcbt shift- move to start of this field or previous X Tab field if at start X.if t .sp 0.5 X kcuu1 Up same as kcbt X Arrow X.if t .sp 0.5 X kel Clear clear to end of field X Line X.if t .sp 0.5 X kf4 Funct same as kel X Key 4 X.if t .sp 0.5 X kdch1 Dlete delete character under cursor X Char X.if t .sp 0.5 X kf3 Funct same as kdch1 X Key 3 X.if t .sp 0.5 X kich1 Input insert a blank under the cursor X Mode X.if t .sp 0.5 X kf2 Funct same as kich1 X Key 2 X.if t .sp 0.5 X kf5 Funct go to the next page of a multi-page form X Key 5 X.if t .sp 0.5 X kf6 Funct go to the previous page of a multi-page form X Key 6 X.fi X.SH FILES X.ta 1.5i X\fIfile\fP.def The file of field and file definitions X.br X\fIfile\fP.db The database records X.br X\fIfile\fP.ak\fBN\fP The index files for searches X.SH NOTE XNo field may contain a colon "\fB:\fP" as it is used as a field Xseparator in the database. X.br XThe maximum size of a field is 255 characters Xand of the total record is 1024 characters. XThe maximum number of fields in a record is 11. XThese are arbitrary numbers and are easy to change. X.br XAt least one field must be defined as an index field. X.br XThe termcap name X.I kcbt Xmay not be defined in some versions of UNIX. XJust use the up arrow instead. X.br XThere is no concurrency control in \fIcardfile\fP. SHAR_EOF chmod 0444 cardfile.1 || echo "restore of cardfile.1 fails" echo "x - extracting Makefile (Text)" sed 's/^X//' << 'SHAR_EOF' > Makefile && X# @(#)Makefile 2.3 DeltaDate 4/16/89 ExtrDate 4/16/89 X X# These are for compiling on the Sun X#CC = /usr/5bin/cc X#LINT = /usr/5bin/lint X#DEFINES = -DBSD -DRE X X X# -DRE get full regular expression matches for doing searches XDEFINES = -DRE XLINT = lint X XBINDIR = /usr/local/bin/ XMANDIR = /usr/man/man1/ X X# X# Nothing past here should need changing X# X X.SUFFIXES: .1 .1~ X.c~.c: X $(GET) $(GFLAGS) $< X X.1~.1: X $(GET) $(GFLAGS) $< X X XSHAR = shar XSHARFLAGS = -v -c -l60 -o cardfile.shr X X#CFLAGS = -g -DDEBUG $(DEFINES) XCFLAGS = -g $(DEFINES) X XDOCS = cardfile.1 X XHDRS = ascii.h cardfile.h X XSRC = \ X cardfile.c \ X add.c \ X change.c \ X common.c \ X compress.c \ X define.c \ X delete.c \ X dumpdb.c \ X extract.c \ X find.c \ X findrcds.c \ X getkey.c \ X keymatch.c \ X maint.c \ X menu.c \ X printdb.c \ X putrcd.c \ X rawio.c \ X rbuildak.c \ X screen.c \ X setupkeys.c \ X updak.c \ X $(NULL) X XOBJ = \ X cardfile.o \ X add.o \ X change.o \ X common.o \ X compress.o \ X define.o \ X delete.o \ X dumpdb.o \ X extract.o \ X find.o \ X findrcds.o \ X getkey.o \ X keymatch.o \ X maint.o \ X menu.o \ X printdb.o \ X putrcd.o \ X rawio.o \ X rbuildak.o \ X screen.o \ X setupkeys.o \ X updak.o \ X $(NULL) X XTESTDB = \ X library.def \ X library.db \ X library.ak0 \ X library.ak1 \ X library.ak2 \ X library.ak3 \ X $(NULL) X X############################################################################### X# X# Make targets X# X############################################################################### X Xall: cardfile cardfile.1 X Xcardfile: $(OBJ) X $(CC) -o cardfile $(OBJ) -lPW -lcurses X Xinstall: all X cp cardfile $(BINDIR)cardfile X strip $(BINDIR)cardfile X cp cardfile.1 $(MANDIR)cardfile.1 X Xclean: X rm -f *.o cardfile cardfile.1 $(SRC) $(HDRS) Makefile X Xprint: prt_src prt_docs X Xprt_src: $(HDRS) $(SRC) Makefile X cpr -w96 -l88 $(HDRS) $(SRC) Makefile | \ X lp -o-v8 -o-t -o-h12 X Xprt_docs: $(DOCS) X nroff -man cardfile.1 | lp -o-qc X Xshar: $(HDRS) $(SRC) $(DOCS) Makefile $(TESTDB) X $(SHAR) $(SHARFLAGS) README $(DOCS) Makefile \ X $(SRC) $(HDRS) $(TESTDB) X Xlint: $(HDRS) $(SRC) X $(LINT) $(SRC) X X############################################################################### X# X# Object dependencies X# X############################################################################### X Xadd.o: cardfile.h ascii.h Xcardfile.o: cardfile.h ascii.h Xchange.o: cardfile.h ascii.h Xcommon.o: cardfile.h ascii.h Xcompress.o: cardfile.h Xdefine.o: cardfile.h ascii.h Xdelete.o: cardfile.h ascii.h Xdumpdb.o: cardfile.h ascii.h Xextract.o: cardfile.h ascii.h Xfind.o: cardfile.h ascii.h Xfindrcds.o: cardfile.h ascii.h Xgetkey.o: cardfile.h Xkeymatch.o: cardfile.h Xmaint.o: cardfile.h Xmenu.o: cardfile.h Xprintdb.o: cardfile.h ascii.h Xputrcd.o: cardfile.h Xrbuildak.o: cardfile.h Xscreen.o: cardfile.h ascii.h Xsetupkeys.o: cardfile.h ascii.h Xupdak.o: cardfile.h ascii.h SHAR_EOF chmod 0444 Makefile || echo "restore of Makefile fails" echo "x - extracting cardfile.c (Text)" sed 's/^X//' << 'SHAR_EOF' > cardfile.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)cardfile.c 2.2 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* CARDFILE.C */ X/* This is the main routine for cardfile X*/ X#include "ascii.h" X#include "stdio.h" X#include "cardfile.h" X#include <curses.h> X#include <term.h> X#include <signal.h> X X#define FIND 0 X#define ADD 1 X#define CHANGE 2 X#define DELETE 3 X#define PRINT 4 X#define MAINT 5 X#define EXIT 6 X XFILE *def_fp; Xint readonly; Xchar fname[FNSIZE]; Xextern char *getfield(); Xchar *functs[] = {"FIND ", X "ADD ", X "CHANGE ", X "DELETE ", X "PRINT ", X "MAINTENANCE", X "EXIT ", X 0 X }; X Xchar datadir[FNSIZE]; Xchar *dbname; Xstruct termio instty, outstty; Xvoid setupscr(); Xvoid exit(); Xunsigned sleep(); X Xmain(argc, argv) Xint argc; Xchar **argv; X{ X int num_flds, i; X char line_buf[133]; X struct Fdata fields[MAXFLDS+1], *fp; X int num_ak; X struct AKdata ak_data[MAXAK+1]; X int func; X char *cp; X extern char *optarg; X extern int optind, opterr; X X opterr = 0; X while ((i = getopt(argc, argv, "r")) != EOF) { X switch (i) { X case 'r': X ++readonly; X break; X case '?': X usage(argv[0]); X exit(1); X } X } X if (optind != argc-1) { X usage(argv[0]); X exit(1); X } X signal(SIGINT, getout); X signal(SIGQUIT, getout); X signal(SIGTERM, getout); X setupscr(); /* from here on use getout, not exit to reset screen */ X if ((cp = strrchr(argv[optind], '/')) != NULL) { X *cp = '\0'; X sprintf(datadir, "%s/", argv[optind]); X dbname = cp + 1; X } else { X *datadir = '\0'; X dbname = argv[optind]; X } X if (strlen(dbname) > 10) { X printf("Database name cannot be longer than 10 characters.\n"); X sleep(5); X getout(); X } X /* build definition file name */ X sprintf(fname, "%s%s.def", datadir, dbname); X if ((def_fp = fopen(fname,"r")) == NULL) { X db_define(dbname); X getout(); X } X if (fgets(line_buf, 132, def_fp) == NULL) { X printf("Unable to read DEF file\n"); X sleep(5); X getout(); X } X if (! readonly && access(fname, 06) != 0) { X printf("You can not write to the database, try\n%s -r %s%s\n", X argv[0], datadir, dbname); X sleep(5); X getout(); X } X strcpy(fname, getfield(line_buf, ":")); X num_flds = atoi(getfield(0, ":")); X for (i=0; i<num_flds; i++) { /* get definition of each field */ X fgets(line_buf, 132, def_fp); X fp = &fields[i]; X strcpy(fp->F_title, getfield(line_buf, ":")); X fp->F_key = *getfield(0, ":"); X strcpy(fp->F_seps, getfield(0, ":")); X fp->F_required = *getfield(0, ":"); X fp->F_length = atoi(getfield(0, ":")); X } X fp = &fields[i]; X fp->F_title[0] = '\0'; X fgets(line_buf, 10, def_fp); X num_ak = atoi(line_buf); X for (i=0; i<num_ak; i++) { /* get definition of each AK file */ X fgets(line_buf, 132, def_fp); X line_buf[strlen(line_buf)-1] = '\0'; X ak_data[i].A_fldnum = atoi(getfield(line_buf, ":")); X strcpy(ak_data[i].A_akname, getfield(0, ":")); X } X ak_data[i].A_fldnum = -1; X fclose(def_fp); X X /* get function to be performed */ X while ((func = menu(dbname, functs)) != EXIT) { X switch (func) { X case FIND: X find(fields, dbname); X continue; X case ADD: X add(fields, dbname); X continue; X case CHANGE: X change(fields, dbname); X continue; X case DELETE: X delete(fields, dbname); X continue; X case PRINT: X printdb(fields, dbname); X continue; X case MAINT: X maint(fields, dbname, ak_data); X continue; X default: X msg("Illegal function chosen"); X getout(); X } X } X getout(); X/*NOTREACHED*/ X} X X Xvoid Xsetupscr() X{ X X setbuf(stdout, NULL); X setbuf(stdin, NULL); X ioctl(0, TCGETA, &outstty); X instty.c_iflag = outstty.c_iflag; X instty.c_oflag = outstty.c_oflag; X instty.c_cflag = outstty.c_cflag; X instty.c_lflag = 0; X instty.c_line = outstty.c_line; X instty.c_cc[VINTR] = outstty.c_cc[VINTR]; X instty.c_cc[VQUIT] = outstty.c_cc[VQUIT]; X instty.c_cc[VERASE] = outstty.c_cc[VERASE]; X instty.c_cc[VKILL] = outstty.c_cc[VKILL]; X instty.c_cc[VMIN] = 1; X instty.c_cc[VTIME] = 0; X instty.c_cc[6] = outstty.c_cc[6]; X instty.c_cc[7] = outstty.c_cc[7]; X ioctl(0, TCSETAW, &instty); X setupterm((char*)0, 1, (int*)0); X setupkeys(); X putp(keypad_xmit); X} X Xgetout() X{ X X putp(keypad_local); X putp(clear_screen); X ioctl(0, TCSETAW, &outstty); X exit(0); X} X Xstatic Xusage(prog) Xchar *prog; X{ X printf("Usage: %s [-r] file\n", prog); X} SHAR_EOF chmod 0644 cardfile.c || echo "restore of cardfile.c fails" echo "x - extracting add.c (Text)" sed 's/^X//' << 'SHAR_EOF' > add.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)add.c 2.3 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* ADD.C */ X/* This subroutine is used to add a new record to the X** data base. It also requests the updating of the X** alternate key files. X*/ X X#include <stdio.h> X#include <errno.h> X#include "ascii.h" X#include "cardfile.h" X Xlong ftell(); Xchar *malloc(); Xextern int errno; Xextern int readonly; X Xadd(fields, dbname) Xstruct Fdata fields[]; Xchar *dbname; X{ X struct Fdata *fp; X struct Sdata add_screen[MAXFLDS+1], *sp; X int err; X char first[SWIDTH+1]; X char out_line[DBSIZE+1]; X FILE *filep; X char filename[FNSIZE]; X long offset; X char *buffer; X X if (readonly) { X msg("Database is readonly"); X return(1); X } X sprintf(first, "Add Records to %s Data Base", dbname); X buffer = malloc(BUFSIZE+1); X buffer[0] = '\0'; X sprintf(filename, "%s%s.db", datadir, dbname); X if((filep = fopen(filename, "a")) == NULL) { X if (errno == EACCES) { X strcpy(out_line, "You do not have permission to modify this database"); X } else { X sprintf(out_line, "Unable to open %s, errno=%d\n", filename, errno); X } X msg(out_line); X return(1); X } X fp = fields; X sp = add_screen; X while (fp->F_title[0] != 0) { X sp->S_title = fp->F_title; X sp->S_length = fp->F_length; X sp->S_result = malloc((unsigned)fp->F_length+1); X sp->S_dfault = 0; X ++fp; X ++sp; X } X sp->S_title = 0; X while (screen(first, add_screen, 0, 0) != 0) { X out_line[0] = '\0'; X err = 0; X fp = fields; X sp = add_screen; X while (sp->S_title) { X if (sp->S_result[0] == '\0' && fp->F_required == 'Y') { X sprintf(out_line, "Required field %s missing", fp->F_title); X msg(out_line); X ++err; X } X if (strchr(sp->S_result, ':')) { X msg("A ':' is not allowed in any field"); X ++err; X } X if (strlen(out_line) + strlen(sp->S_result) >= DBSIZE) { X msg("Record too long"); X ++err; X break; X } X strcat(out_line, sp->S_result); X strcat(out_line, ":"); X ++fp; X ++sp; X } X if (err) { X continue; X } X if (err > 0) X break; X out_line[strlen(out_line)-1] = '\0'; X offset = ftell(filep); X fprintf(filep, " :%s\n", out_line); X buildak(dbname, out_line, offset, fields, buffer); X } X sp = add_screen; X while (sp->S_title) { X free(sp->S_result); X ++sp; X } X fclose(filep); X writeak(dbname, buffer); X free(buffer); X return(0); X} SHAR_EOF chmod 0444 add.c || echo "restore of add.c fails" echo "x - extracting change.c (Text)" sed 's/^X//' << 'SHAR_EOF' > change.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)change.c 2.3 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* CHANGE.C */ X/* This module is used to find and change records X** in the data base matching the selection criteria. X** The user may select multiple values of one field X** which will be or'd and/or multiple fields which will X** be and'd. In other words, if field 1 is specified X** as value1:value2 and field3 is specified as value3. X** Records with field1==(value1 || value2) && field3==value3 X** will be selected for changing. X** It calls findrcds.c to do all the work of finding them. X** It does not actually change records, the flag(first) X** field is set to 'D' in the current record and a new X** record is added. To physically delete the old record, X** the data base must be compressed. X*/ X X#include <stdio.h> X#include <errno.h> X#include "ascii.h" X#include "cardfile.h" X#include <curses.h> X#include <term.h> X Xchar *malloc(), *getfield(); Xextern int readonly; X X Xchange(fields, dbname) Xstruct Fdata *fields; Xchar *dbname; X{ X char first[SWIDTH]; X int processc(); X X if (readonly) { X msg("Database is readonly"); X return(1); X } X sprintf(first, "Select Records from the %s Data Base to be Changed", X dbname); X findrcds(fields, dbname, processc, first); X return(0); X} X X Xprocessc(fields, rcd, dbfile, dbname) Xstruct Fdata *fields; Xchar *rcd; XFILE *dbfile; Xchar *dbname; X{ X char c, save[DBSIZE+1]; X long offset, ftell(); X char *strchr(); X static char fmt[256]; X int fn; X struct Fdata *fp; X X strcpy(save, rcd); X putp(clear_screen); /* clear screen */ X if (*fmt == '\0') { /* first time */ X fp = fields; X fn = 1; X while (fp->F_title[0]) { X sprintf(&fmt[strlen(fmt)], "%s: %%%d%%n", fp->F_title, fn); X ++fn; X ++fp; X } X } X putrcd("Record to be CHANGED", save, stdout, fmt, SWIDTH, 2); X putp(tparm(cursor_address, MSGLINE, 9)); X fputs("RETURN for next, ESC to abort, Ctrl-C to Change, Ctrl-B to reverse", X stdout); X noecho(); X while ((c=rawgetchar()) != LF && c != CR) { X if (c == ETX) { /* CTL-C entered */ X offset = ftell(dbfile) - strlen(rcd); X *strchr(rcd, '\n') = '\0'; /* truncate \n */ X if (doadd(dbname, fields, rcd, dbfile) == 0) { X fseek(dbfile, offset, 0); X putc('D', dbfile); X } X break; X } X if (c == ESC) { X echo(); X return(1); X } X if (c == STX) { /* CTRL-B */ X echo(); X return(-1); X } X rawputchar(BEL); X } X echo(); X return(0); X} X Xdoadd(dbname, fields, rcd, filep) Xchar *dbname; Xstruct Fdata *fields; Xchar *rcd; XFILE *filep; X{ X struct Sdata add_screen[MAXFLDS+1], *sp; X struct Fdata *fp; X int err; X char buffer[BUFSIZE+1]; X char outline[DBSIZE+1]; X long offset, ftell(); X X fseek(filep, 0L, 2); /* end of file */ X fp = fields; X getfield(rcd, ":"); /* step past flag */ X sp = add_screen; X while(fp->F_title[0]) { X sp->S_title = fp->F_title; X sp->S_length = fp->F_length; X sp->S_result = malloc((unsigned)fp->F_length); X sp->S_dfault = getfield(0, ":"); X ++fp; X ++sp; X } X sp->S_title = 0; X outline[0] = '\0'; X while (screen("Enter changes", add_screen, 0, 0) > 0) { X err = 0; X fp = fields; X sp = add_screen; X while(fp->F_title[0]) { X if (sp->S_result[0] == '\0' && fp->F_required == 'Y') { X sprintf(outline, "Required field %s missing", fp->F_title); X msg(outline); X err++; X } X if (strchr(sp->S_result, ':')) { X msg("A : is not allowed in any field"); X ++err; X } X if (strlen(outline) + strlen(sp->S_result) >= DBSIZE) { X msg("Record too long"); X ++err; X break; X } X strcat(outline, sp->S_result); X strcat(outline, ":"); X ++fp; X ++sp; X } X if (err == 0) X break; X outline[0] = '\0'; X } X if (err > 0 || outline[0] == '\0') { X sp = add_screen; X while(sp->S_title) X free((sp++)->S_result); X return(1); X } X outline[strlen(outline)-1] = '\0'; /* truncate trailing : */ X offset = ftell(filep); X if (fprintf(filep, " :%s\n", outline) == EOF) { X msg("Error writing DB file"); X getout(); X } X buffer[0] = '\0'; X buildak(dbname, outline, offset, fields, buffer); X writeak(dbname, buffer); X sp = add_screen; X while(sp->S_title) X free((sp++)->S_result); X return(0); X} SHAR_EOF chmod 0444 change.c || echo "restore of change.c fails" echo "x - extracting common.c (Text)" sed 's/^X//' << 'SHAR_EOF' > common.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)common.c 2.1 DeltaDate 2/13/89 ExtrDate 2/13/89"; X#endif X X/* COMMON.C */ X#include "stdio.h" X#include "ascii.h" X#include "cardfile.h" X#include <curses.h> X#include <term.h> X Xmsg(str) /* put a blinking message on the 23'th line of the crt */ Xchar *str; X{ X putp(tparm(cursor_address, MSGLINE, 9)); X putp(clr_eol); /* erase line */ X putp(enter_blink_mode); /* blink */ X fputs(str, stdout); X putp(exit_attribute_mode); /* back to normal */ X sleep(5); X} X X Xchar * Xgetfield(string, sepset) /* like strtok except contiguous seps */ Xchar *string, *sepset; /* result in a null field */ X{ X register char *p, *r; X static char *savept; X char *strpbrk(); X X p = (string == NULL)? savept : string; X if (p == 0) X return(NULL); X if (*p == '\0') X return(NULL); X if ((r = strpbrk(p, sepset)) == NULL) /* move past token */ X savept = 0; /* indicate this is the last token */ X else { X *r = '\0'; X savept = ++r; X } X return(p); X} X X Xhelp(help_msg) Xchar *help_msg; X{ X putp(clear_screen); /* clear screen */ X puts("\n"); X fputs(help_msg, stdout); X putp(tparm(cursor_address, MSGLINE, 9)); X fputs("Enter any character to continue.", stdout); X rawgetchar(); X putp(clear_screen); /* clear screen */ X} SHAR_EOF chmod 0644 common.c || echo "restore of common.c fails" echo "x - extracting compress.c (Text)" sed 's/^X//' << 'SHAR_EOF' > compress.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)compress.c 2.2 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* COMPRESS.C */ X/* This function will remove all records with a flag X** byte of 'D' from the data base. The alternate key X** files MUST be rebuilt after it finishes. X*/ X X#include "stdio.h" X#include "cardfile.h" X Xcompress(dbname) Xchar *dbname; X{ X FILE *in, *out; X char fname[FNSIZE]; X char tempname[FNSIZE]; X char rcd[DBSIZE+1]; X X sprintf(tempname, "%snewdb.$$$", datadir); X if ((out = fopen(tempname, "w")) == NULL) { X msg("Unable to create temporary file"); X getout(); X } X sprintf(fname, "%s%s.db", datadir, dbname); X if ((in = fopen(fname, "r")) == NULL) { X unlink(tempname); X msg("Unable to read DB file"); X getout(); X } X while (fgets(rcd, DBSIZE, in) != NULL) { X if (feof(in)) X break; X if (*rcd == 'D') /* record to be deleted */ X continue; X if(fputs(rcd, out) == EOF) { X unlink(tempname); X msg("Unable to write to temporary file"); X getout(); X } X } X fclose(in); X fclose(out); X unlink(fname); X link(tempname, fname); X unlink(tempname); X} SHAR_EOF chmod 0644 compress.c || echo "restore of compress.c fails" echo "x - extracting define.c (Text)" sed 's/^X//' << 'SHAR_EOF' > define.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)define.c 2.2 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* DEFINE.C */ X/* This subroutine is used to define a new data-base. X** It expects as input the base name (8 characters or less) X** of all files. X** X** It will create the definition file - name.def X** the primary file - name.db X** and the alternate key files - name.akN X*/ X X#include "cardfile.h" X#include "stdio.h" X#include "ascii.h" X#include <curses.h> X#include <term.h> X Xstruct Sdata def_screen[] = { X {"FIELD NAME",TSIZE," ",0}, X {"MAXIMUM LENGTH",3," ",0}, X {"REQUIRED?",1," ","N"}, X {"SEARCH KEY",1," ","Y"}, X {"SUBFIELD SEPARATORS",5," ",0}, X {0,0,0,0} X }; X Xstruct Fdata fields[MAXFLDS]; X Xdb_define(basename) Xchar *basename; X{ X int i; X char first[SWIDTH]; X char *last = "RETURN to exit"; X struct Fdata *fp; X int num_fields, num_ak; X char fname[FNSIZE]; X FILE *filep, *filep2; X int err; X int tsize; X X sprintf(first,"DEFINING %s DATA BASE", basename); X num_fields = 0; X fp = fields; X tsize = 0; X if (access((*datadir ? datadir : "."), 06) != 0) { X printf("Cannot write into %s\n", (*datadir ? datadir : ".")); X sleep(5); X return(1); X } X X /* X * Read each field entered X */ X while (screen(first, def_screen, last, 0)) { X err = 0; X strcpy(fp->F_title, def_screen[0].S_result); X if (strchr(fp->F_title, ':') != NULL) { X msg("TITLE field must not include :"); X ++err; X } X fp->F_length = atoi(def_screen[1].S_result); X if (fp->F_length <= 0 || fp->F_length >= FLDLEN) { X msg("LENGTH field must be less than 256"); X ++err; X } X if ((tsize += fp->F_length) > DBSIZE) { X msg("Total record length too large"); X ++err; X } X fp->F_required = toupper(def_screen[2].S_result[0]); X if (fp->F_required != 'Y' && fp->F_required != 'N') { X msg("REQUIRED must be Y or N"); X ++err; X } X fp->F_key = toupper(def_screen[3].S_result[0]); X if (fp->F_key != 'Y' && fp->F_key != 'N') { X msg("KEY must be Y or N"); X ++err; X } X strcpy(fp->F_seps, def_screen[4].S_result); X if (strchr(fp->F_seps, ':') != NULL) { X msg("SEPARATORS must not include :"); X ++err; X } X if (err) { X continue; X } X for (i=0; i<num_fields; i++) { X if (strcmp(fields[i].F_title, fp->F_title) == 0) { X msg("Field already defined"); X err++; X break; X } X } X if (err) X continue; X ++fp; X if (++num_fields > MAXFLDS) { X msg("Too many fields defined, aborting"); X return(1); X } X } X X putp(clear_screen); /* clear screen */ X if (num_fields == 0) { X msg("No fields defined, aborting"); X return(1); X } X sprintf(fname,"%s%s.def", datadir, basename); X if ((filep = fopen(fname,"w")) == NULL) { X msg("Unable to create DEF file"); X return(1); X } X sprintf(fname,"%s%s.db", datadir, basename); X if ((filep2 = fopen(fname,"w")) == NULL) { X msg("Unable to create DB file"); X sprintf(fname, "%s.def", basename); X unlink(fname); X return(1); X } X fclose(filep2); X fprintf(filep, "%s:%d\n", fname, num_fields); X num_ak = 0; X for (i=0; i<num_fields; i++) { X fp = &fields[i]; X if (fp->F_key == 'Y') X fp->F_key = '0' + num_ak++; X fprintf(filep, "%s:%c:%s:%c:%d\n", fp->F_title, fp->F_key, X fp->F_seps, fp->F_required, fp->F_length); X } X if (num_ak == 0) { X msg("At least 1 key field must be defined"); X unlink(fname); /* DB file */ X sprintf(fname, "%s.def", basename); X unlink(fname); X return(1); X } X fprintf(filep, "%d\n", num_ak); X num_ak = 0; X for (i=0; i<num_fields; i++) { X fp = &fields[i]; X if (fp->F_key != 'N') { /* Alternate key file required */ X sprintf(fname,"%s%s.ak%d", datadir, basename, num_ak++); X if ((filep2 = fopen(fname,"w")) == NULL) { X msg("Unable to create AK file\n"); X sprintf(fname, "%s.def", basename); X unlink(fname); X return(1); X } X fclose(filep2); X fprintf(filep, "%d:%s\n", i, fname); X } X } X fclose(filep); X return(0); X} SHAR_EOF chmod 0644 define.c || echo "restore of define.c fails" echo "x - extracting delete.c (Text)" sed 's/^X//' << 'SHAR_EOF' > delete.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)delete.c 2.3 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* DELETE.C */ X/* This module is used to find and delete records X** in the data base matching the selection criteria. X** The user may select multiple values of one field X** which will be or'd and/or multiple fields which will X** be and'd. In other words, if field 1 is specified X** as value1:value2 and field3 is specified as value3. X** Records with field1==(value1 || value2) && field3==value3 X** will be selected for deletion. X** It calls findrcds.c to do all the work. X** It does not actually delete records, the flag(first) X** field is set to 'D'. To physically delete them, the data X** base must be compressed. X*/ X X#include "stdio.h" X#include "ascii.h" X#include "cardfile.h" X#include <curses.h> X#include <term.h> X X Xextern int readonly; X Xdelete(fields, dbname) Xstruct Fdata *fields; Xchar *dbname; X{ X char first[SWIDTH]; X int processd(); X X if (readonly) { X msg("Database is readonly"); X return(1); X } X sprintf(first, "Select Records from the %s Data Base to be DELETED", X dbname); X findrcds(fields, dbname, processd, first); X return(0); X} X X X/*ARGSUSED*/ Xprocessd(fields, rcd, dbfile, dummy) Xstruct Fdata *fields; Xchar *rcd; XFILE *dbfile; Xchar *dummy; X{ X char c, save[DBSIZE+1]; X static char fmt[256]; X int fn; X struct Fdata *fp; X X strcpy(save, rcd); X putp(clear_screen); /* clear screen */ X if (*fmt == '\0') { /* first time */ X fp = fields; X fn = 1; X while (fp->F_title[0]) { X sprintf(&fmt[strlen(fmt)], "%s: %%%d%%n", fp->F_title, fn); X ++fn; X ++fp; X } X } X putrcd("Record to be DELETED", save, stdout, fmt, SWIDTH, 2); X putp(tparm(cursor_address, MSGLINE, 9)); X fputs("RETURN for next, ESC to abort, CTL-D to DELETE, CTL-B to reverse", X stdout); X noecho(); X while ((c=rawgetchar()) != LF && c != CR) { X if (c == EOT) { /* CTL-D entered */ X fseek(dbfile, (long)(-strlen(rcd)), 1); X putc('D', dbfile); X break; X } X if (c == ESC) { X echo(); X return(1); X } X if (c == STX) { X echo(); X return(-1); X } X rawputchar(BEL); X } X echo(); X return(0); X} SHAR_EOF chmod 0444 delete.c || echo "restore of delete.c fails" echo "x - extracting dumpdb.c (Text)" sed 's/^X//' << 'SHAR_EOF' > dumpdb.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)dumpdb.c 2.1 DeltaDate 2/13/89 ExtrDate 2/13/89"; X#endif X X/* DUMPDB.C */ X/* This subroutine is used to dump all records from X the data base and the alternate key files, including X deleted records. X*/ X X#include "stdio.h" X#include "ascii.h" X#include "cardfile.h" X#include <curses.h> X#include <term.h> X#include <signal.h> X Xstatic FILE *in, *out; Xstatic int quit = 0; Xstatic int abort(); Xstatic int (*old_sig)(); X Xdumpdb(dbname, ak_data, fields) Xchar *dbname; Xstruct AKdata *ak_data; Xstruct Fdata *fields; X{ X char ptitle[PWIDTH+1]; X int pnum, lnum; X char fname[FNSIZE]; X char rcd[DBSIZE+1]; X X if ((out = popen("lp", "w")) == NULL) { X msg("Unable to open printer"); X return(1); X } X sprintf(fname, "%s%s.db", datadir, dbname); X if ((in = fopen(fname, "r")) == NULL) { X pclose(out); X msg("Unable to open DB file"); X return(1); X } X putp(tparm(cursor_address, MSGLINE, 9)); X putp(clr_eol); X fputs("DEL to abort dump", stdout); X old_sig = signal(SIGINT, abort); X lnum = PLENGTH + 1; /* force top-of-page */ X pnum = 1; X sprintf(ptitle, "Records in %s.db", dbname); X while( !quit && (fgets(rcd, DBSIZE, in)) != NULL) { /* print DB file */ X rcd[strlen(rcd)-1] = '\0'; /* truncate \n */ X if (lnum + 3 + (strlen(rcd)+PWIDTH-1)/PWIDTH >= PLENGTH) { X newpage(&pnum, ptitle); X lnum = 3; X } X fputs(rcd, out); X putc('\n', out); X lnum += (strlen(rcd)+PWIDTH-1)/PWIDTH; X } X if (quit) X return(1); X fclose(in); X while (ak_data->A_fldnum >= 0) { X strcpy(fname, datadir); X strcat(fname, ak_data->A_akname); X if ((in = fopen(fname, "r")) == NULL) { X sprintf(ptitle, "Unable to open %s", fname); X msg(ptitle); X return(1); X } X lnum = PLENGTH + 1; X pnum = 1; X sprintf(ptitle, "Records in %s sorted on %s", X fname, fields[ak_data->A_fldnum].F_title); X /* print AK file */ X while( !quit && (fgets(rcd, DBSIZE, in)) != NULL) { X rcd[strlen(rcd)-1] = '\0'; /* truncate \n */ X if (lnum + 3 + (strlen(rcd)+PWIDTH-1)/PWIDTH >= PLENGTH) { X newpage(&pnum, ptitle); X lnum = 3; X } X fputs(rcd, out); X putc('\n', out); X lnum += (strlen(rcd)+PWIDTH-1)/PWIDTH; X } X if (quit) X return(1); X fclose(in); X ++ak_data; X } X pclose(out); X signal(SIGINT, old_sig); X return(0); X} X Xnewpage(pnum, title) Xint *pnum; Xchar *title; X{ X X putc(FF, out); X fprintf(out, "%-69sPAGE %4d\n\n\n", title, *pnum); X *pnum += 1; X} X X Xstatic Xabort() X{ X X signal(SIGINT, old_sig); X fputs("\n\nOUTPUT ABORTED!\n", out); X pclose(out); X fclose(in); X msg("Dump aborted"); X quit = 1; X} SHAR_EOF chmod 0644 dumpdb.c || echo "restore of dumpdb.c fails" echo "x - extracting extract.c (Text)" sed 's/^X//' << 'SHAR_EOF' > extract.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)extract.c 2.2 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* EXTRACT.C */ X/* This subroutine is used to write selected records from the X * database on to another disk file. X */ X X#include "stdio.h" X#include "cardfile.h" X#include "ascii.h" X#include <curses.h> X#include <term.h> X XFILE *ext_fp; Xchar ext_fname[FNSIZE]; Xint rec_cnt; Xstruct Sdata fn_screen[] = { X {"File Name", FNSIZE, ext_fname, NULL}, X {0, 0, 0, 0} X }; X Xextract(fields, dbname) Xstruct Fdata *fields; Xchar *dbname; X{ X char first[SWIDTH+1]; X int write_rcd(); X X sprintf(first, "Select records from the %s Data Base to be extracted", X dbname); X screen(first, fn_screen, "Enter File Name of Extract File", NULL); X if (ext_fname[0] == '\0') { X msg("No file name entered, using temp.ext"); X strcpy(ext_fname, "temp.ext"); X } X if ((ext_fp = fopen(ext_fname, "w")) == NULL) { X sprintf(first, "Unable to open file %s for output", ext_fname); X msg(first); X return(1); X } X rec_cnt = 0; X findrcds(fields, dbname, write_rcd, first); X fclose(ext_fp); X sprintf(first, "%d records written to %s", rec_cnt, ext_fname); X putp(tparm(cursor_address, MSGLINE, 9)); X putp(clr_eol); X fputs(first, stdout); X sleep(5); X return(0); X} X X X/*ARGSUSED*/ Xint Xwrite_rcd(dummy1, rcd, dummy2, dummy3) Xchar *rcd; Xchar *dummy1, *dummy2, *dummy3; X{ X ++rec_cnt; X return(fputs(rcd, ext_fp)); X} SHAR_EOF chmod 0444 extract.c || echo "restore of extract.c fails" echo "x - extracting find.c (Text)" sed 's/^X//' << 'SHAR_EOF' > find.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)find.c 2.2 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* FIND.C */ X/* This module is used to find and display records X** in the data base matching the selection criteria. X** The user may select multiple values of one field X** which will be or'd and/or multiple fields which will X** be and'd. In other words, if field 1 is specified X** as value1:value2 and field3 is specified as value3. X** Records with field1==(value1 || value2) && field3==value3 X** will be displayed. X** It calls findrcds.c to do all the work. X*/ X X#include "stdio.h" X#include "ascii.h" X#include "cardfile.h" X#include <curses.h> X#include <term.h> X X Xfind(fields, dbname) Xstruct Fdata *fields; Xchar *dbname; X{ X char first[SWIDTH]; X int display(); X X sprintf(first, "Display Records from the %s Data Base", dbname); X findrcds(fields, dbname, display, first); X return(0); X} X X X/*ARGSUSED*/ Xdisplay(fields, rcd, dummy1, dummy2) Xstruct Fdata *fields; Xchar *rcd; XFILE *dummy1; Xchar *dummy2; X{ X char c, save[DBSIZE+1]; X FILE *filep; X char *first = "Selected Records"; X static char fmt[256]; X int fn; X struct Fdata *fp; X X strcpy(save, rcd); X putp(clear_screen); /* clear screen */ X if (*fmt == '\0') { /* first time */ X fp = fields; X fn = 1; X while (fp->F_title[0]) { X sprintf(&fmt[strlen(fmt)], "%s: %%%d%%n", fp->F_title, fn); X ++fn; X ++fp; X } X } X putrcd(first, rcd, stdout, fmt, SWIDTH, 2); X putp(tparm(cursor_address, MSGLINE, 9)); X fputs("RETURN for next, ESC to abort, CTL-P to print, CTL-B to reverse", X stdout); X noecho(); X while ((c=rawgetchar()) != LF && c != CR) { X if (c == DLE) { /* CTL-P entered */ X if ((filep=popen("lp", "w")) == NULL) { X msg("Unable to open printer"); X continue; X } X strcpy(rcd, save); X putrcd(first, rcd, filep, fmt, PWIDTH, 1); X pclose(filep); X continue; X } X if (c == ESC) { X echo(); X return(1); X } X if (c == STX) { X echo(); X return(-1); X } X rawputchar(BEL); X } X echo(); X return(0); X} SHAR_EOF chmod 0444 find.c || echo "restore of find.c fails" echo "x - extracting findrcds.c (Text)" sed 's/^X//' << 'SHAR_EOF' > findrcds.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)findrcds.c 2.3 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* FINDRCDS.C */ X/* This module is used to find records in the data base X** matching the selection criteria. X** The user may select multiple values of one field X** connected by '&' or '|' which will be or'd and/or X** multiple fields which will be and'd. X** In other words, if field 1 is specified X** as value1|value2 and field3 is specified as value3. X** Records with field1==(value1 || value2) && field3==value3 X** will be selected. An external routine 'display', passed X** as a parameter will be called to process the selected records. X*/ X X#include <stdio.h> X#include <errno.h> X#include "ascii.h" X#include "cardfile.h" X Xextern int readonly; X Xstruct Keys { X char K_flag; /* 0 = good record X * 1 = found on this search argument X * 64 = physical end of key array X * 128 = first free key block */ X long K_offset; X}; X X Xfindrcds(fields, dbname, display, first) Xstruct Fdata *fields; Xchar *dbname; Xint (*display)(); Xchar *first; X{ X struct Fdata *fp; X struct Sdata find_screen[MAXFLDS+1], *sp; X int found; X FILE *filep; X char filename[FNSIZE]; X diskptr offset; X char *last = X "Separate alternate values with a | for 'or' or a & for 'and'"; X int knum, firstflag; X long getkey(); X char *keyval, *cp; X char *rcd; X char operator, nextop; X char out_line[SWIDTH+1]; X struct Keys keys[MAXKEYS+1], *kp; X int i; X int rc; X char *getslct(), *malloc(); X X fp = fields; X sp = find_screen; X while (fp->F_title[0] != '\0') { X if (fp->F_key == 'N') { X ++fp; X continue; X } X sp->S_title = fp->F_title; X sp->S_length = 2 * fp->F_length; X sp->S_result = malloc((unsigned)sp->S_length+1); X sp->S_dfault = 0; X ++fp; X ++sp; X } X sp->S_title = 0; X while (screen(first, find_screen, last, 0) != 0) { X firstflag = 1; X knum = -1; X keys[0].K_flag = 128; X for (i=1; i<MAXKEYS; ++i) X keys[i].K_flag = 0; X keys[MAXKEYS].K_flag = 64; X sp = find_screen; X while (sp->S_title) { /* loop on key field */ X ++knum; X if (*(sp->S_result) == '\0') { /* was key entered */ X ++sp; X continue; X } X sprintf(filename, "%s%s.ak%d", datadir, dbname, knum); X if ((filep = fopen(filename, "r")) == NULL) { X msg("Unable to open alternate key file"); X getout(); X } X operator = '&'; X if (firstflag) { X firstflag = 0; X operator = '|'; X } X cp = sp->S_result; X /* loop for each key for this field */ X while ((keyval = getslct(cp, "&|", &nextop)) != NULL) { X cp = NULL; X /* loop getting offsets for this value */ X while ((offset = getkey(filep, keyval)) >= 0L) { X keyval = 0; X if (operator == '|') { X if (orkey(offset, keys) == -1) X break; X } X if (operator == '&') { X andkey(offset, keys); X } X } X if (operator == '&') { X purgekey(keys); X } X operator = nextop; X } X fclose(filep); X ++sp; X } X sprintf(filename, "%s%s.db", datadir, dbname); X if (readonly) { X if ((filep = fopen(filename, "r")) == NULL) { X sprintf(out_line, "Unable to open %s, errno=%d\n", filename, errno); X msg(out_line); X return(1); X } X } else { X if ((filep = fopen(filename, "r+")) == NULL) { X if (errno == EACCES) { X strcpy(out_line, "You do not have permission to modify this database"); X } else { X sprintf(out_line, "Unable to open %s, errno=%d\n", filename, errno); X } X msg(out_line); X return(1); X } X } X found = 0; X rcd = malloc(DBSIZE); X kp = keys; X while (kp->K_flag == 0) { X if (kp->K_offset <0L) { X ++kp; X continue; X } X fseek(filep, kp->K_offset, 0); X if (fgets(rcd, DBSIZE, filep) == NULL) { X msg("Error reading DB file"); X getout(); X } X if (*rcd == 'D') { X kp->K_offset = -1L; X ++kp; X continue; X } X ++found; X if ((rc = display(fields, rcd, filep, dbname)) == 1) X break; X if (rc == -1) { /* reverse */ X while (kp > &keys[0]) { X --kp; X if (kp->K_offset >= 0L) X break; X } X } else { X ++kp; X } X } X fclose(filep); X free(rcd); X if (found == 0) { X msg("No records found"); X } X } X sp = find_screen; X while (sp->S_title) { X free(sp->S_result); X ++sp; X } X return(0); X} X X Xchar * Xgetslct(string, seps, sepval) /* identical to getfld except separator is */ Xchar *string, *seps, *sepval; /* returned thru sepval */ X{ X register char *p, *r; X static char *savept; X X p = (string == NULL)? savept : string; X if (p == 0) X return(NULL); X if (*p == '\0') X return(NULL); X if ((r = strpbrk(p, seps)) == NULL) { /* move past token */ X *sepval = '\0'; X savept = 0; /* indicate this is the last token */ X } else { X *sepval = *r; X *r = '\0'; X savept = ++r; X } X return(p); X} X X Xorkey(offset, keys) Xdiskptr offset; Xstruct Keys *keys; X{ X struct Keys *save; X X save = NULL; X while (keys->K_flag == 0) { X if (offset == keys->K_offset) { X return(0); X } X if (keys->K_offset < 0L && save == NULL) { X save = keys; /* found free point for insertion */ X } X ++keys; X } X if (save == NULL) { X if (keys->K_flag == 64) { X msg("Too many records selected"); X return(-1); X } X save = keys; X if ((++keys)->K_flag != 64) X keys->K_flag = 128; X } X save->K_flag = 0; X save->K_offset = offset; X return(0); X} X X#define MATCH 1 X Xandkey(offset, keys) Xdiskptr offset; Xstruct Keys *keys; X{ X X while(keys->K_flag < 64) { X if (offset == keys->K_offset) { X keys->K_flag = MATCH; X break; X } X ++keys; X } X} X X Xpurgekey(keys) Xstruct Keys *keys; X{ X X while(keys->K_flag < 64) { X if (keys->K_flag != MATCH) { X keys->K_offset = -1L; X } else { X keys->K_flag = 0; X } X ++keys; X } X} SHAR_EOF chmod 0444 findrcds.c || echo "restore of findrcds.c fails" echo "x - extracting getkey.c (Text)" sed 's/^X//' << 'SHAR_EOF' > getkey.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)getkey.c 2.1 DeltaDate 2/13/89 ExtrDate 2/13/89"; X#endif X X/* GETKEY.C */ X/* This subroutine is used to search an alternate key file X to find records matching the input value. It returns the X offset of the record in the DB file if found, or -1 if X no match was found. X*/ X#include "stdio.h" X#include "cardfile.h" X#include "ascii.h" X Xdiskptr Xgetkey(file, val) XFILE *file; Xchar *val; X{ X static char *match; X char rcd[AKSIZE]; X long atol(); X char *fgets(); X int rc; X X if (val != 0) { /* first time */ X match = val; X fseek(file, 0L, 0); X } X while (fgets(rcd, AKSIZE, file) != NULL) { X if ((rc = keymatch(rcd, match)) == -1) { X break; X } else if (rc == 1) { X return (atol(strchr(rcd, ':')+1)); X } X } X return (-1L); X} SHAR_EOF chmod 0644 getkey.c || echo "restore of getkey.c fails" echo "x - extracting keymatch.c (Text)" sed 's/^X//' << 'SHAR_EOF' > keymatch.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)keymatch.c 2.1 DeltaDate 2/13/89 ExtrDate 2/13/89"; X#endif X X/* KEYMATCH.C */ X/* This subroutine is used to test if the key in the AK file X** record matches the input value. Two options are supported, X** UNIX style regular expressions or simple matches. If 'RE' X** is defined regular expressions are allowed. X** If the input value is enclosed in quotes an exact match is X** required, otherwise upper/lower case is ignored. If simple X** matching is being done, an '*' at the end of an unquoted X** string matches everything. X**/ X#include "cardfile.h" X X#ifdef RE X# ifdef BSD Xextern char *re_comp(); Xextern int re_exec(); X# else Xextern char *regcmp(), *regex(); X# endif X#endif X Xkeymatch(akrcd, val) Xchar *akrcd, *val; X{ X register int vlen; X char convert[256]; X char key[256]; X static int lower; X#ifdef RE X static char lastval[256]; X# ifndef BSD X static char *cval; X# endif X#else X register char *cp = convert, X *kp = key; X#endif X X#ifdef RE X if (strcmp(val, lastval) != 0) { /* new pattern */ X vlen = strlen(val); X# ifndef BSD X if (cval) /* free last pattern if any */ X free (cval); X# endif X strcpy(lastval, val); X#else X vlen = strlen(val); X#endif X strcpy(convert, val); X if ((*convert == '"' && convert[vlen-1] == '"') X || (*convert == '\'' && convert[vlen-1] == '\'')) { X strcpy(convert, convert+1); X convert[vlen-2] = '\0'; X lower = 0; X } else { X strlower(convert); X lower = 1; X } X#ifdef RE X# ifdef BSD X if (re_comp(convert) != 0) { X# else X if ((cval = regcmp(convert, 0)) == 0) { X# endif X msg("Invalid search pattern"); X return(-1); X } X } X#endif X strcpy(key, akrcd); X *strchr(key, ':') = '\0'; X if (lower) { X strlower(key); X } X#ifdef RE X# ifdef BSD X if (re_exec(key) != 0) X return(1); X# else X if (regex(cval, key) != 0) X return(1); X# endif X return(0); X#else X while (*cp != '\0' && (lower == 0 || *cp != '*')) { X if (*cp != *kp) X return(0); X ++cp; X ++kp; X } X if (*cp == '\0' && *kp != '\0') X return(0); X return(1); X#endif X} X X Xstatic Xstrlower(str) Xregister char *str; X{ X X while (*str) { X if (*str >= 'A' && *str <= 'Z') X *str = *str - 'A' + 'a'; X ++str; X } X} SHAR_EOF chmod 0644 keymatch.c || echo "restore of keymatch.c fails" echo "x - extracting maint.c (Text)" sed 's/^X//' << 'SHAR_EOF' > maint.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)maint.c 2.1 DeltaDate 2/13/89 ExtrDate 2/13/89"; X#endif X X/* MAINT.C */ X/* This module is displays the maintenance menu and X controls the execution of the associated routines X*/ X#include "stdio.h" X#include "cardfile.h" X Xchar *mfuncts[] = {"EXIT ", X "DUMP ", X "COMPRESS ", X "REBUILD AK's", X "EXTRACT ", X 0 X }; X#define DUMP 1 X#define COMPRES 2 X#define RBUILD 3 X#define EXTRACT 4 X#define EXIT 0 X X Xmaint(fields, dbname, ak_data) Xstruct Fdata *fields; Xchar *dbname; Xstruct AKdata *ak_data; X{ X char first[SWIDTH]; X int func; X X sprintf(first, "Maintenance functions for %s", dbname); X while ((func = menu(first, mfuncts)) != EXIT) { X switch (func) { X case DUMP: X dumpdb(dbname, ak_data, fields); X continue; X case COMPRES: X compress(dbname); X rbuildak(dbname, ak_data, fields); X continue; X case RBUILD: X rbuildak(dbname, ak_data, fields); X continue; X case EXTRACT: X extract(fields, dbname); X continue; X default: X msg("Illegal function chosen"); X } X } X} SHAR_EOF chmod 0644 maint.c || echo "restore of maint.c fails" echo "x - extracting menu.c (Text)" sed 's/^X//' << 'SHAR_EOF' > menu.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)menu.c 2.1 DeltaDate 2/13/89 ExtrDate 2/13/89"; X#endif X X/* MENU.C */ X/* This subroutine is used to display a menu of commands X and wait for the user to select one. X It returns the number of the command selected. X*/ X#include "cardfile.h" X Xmenu(dbname, functs) Xchar *dbname; Xchar **functs; X{ X int func; X char dummy[2]; X struct Sdata *sp, *mscreen; X char *malloc(); X X mscreen = (struct Sdata*)malloc((MAXFLDS+1) * sizeof(struct Sdata)); X sp = mscreen; X do { X sp->S_title = *functs; X sp->S_length = 1; X sp->S_result = dummy; X sp->S_dfault = 0; X ++sp; X } while (*(functs++)); X screen(dbname, mscreen, 0, &func); X free((char*)mscreen); X return (func); X} SHAR_EOF chmod 0644 menu.c || echo "restore of menu.c fails" echo "x - extracting printdb.c (Text)" sed 's/^X//' << 'SHAR_EOF' > printdb.c && X#ifndef lint Xstatic char Sccsid[] = "@(#)printdb.c 2.2 DeltaDate 4/16/89 ExtrDate 4/16/89"; X#endif X X/* PRINTDB.C */ X/* This module format prints the data base in the order requested. X** NOTE: If a record contains multiple values for the key selected, X** it will be printed multiple times. X** Alternativly it will format print an extracted file (see extract.c) X** The formats allowed are: X** %NN - contents of field NN (1-maxfield) X** %NN(form) - contents of field NN in printf '%form' format X** %n - new-line X** %t - tab X** %f - form feed X** %t(NN) - tab to column NN X** %% - % X** NOTE: '.' ends a format specification %1.1 prints the contents of X** field 1 concatenated with a '1'. X**/ X#include "stdio.h" X#include "cardfile.h" X#include "ascii.h" X#include <curses.h> X#include <term.h> X#include <signal.h> X Xchar out_format[128], ext_file[15]; Xchar out_file[15], out_width[4]; Xstruct Sdata first_screen[] = { X {"Format", 127, out_format, 0}, X {"Extracted File", 14, ext_file, 0}, X {"Output File", 14, out_file, 0}, X {"Output Width", 3, out_width, 0}, X {0, 0, 0, 0} X }; X Xstatic int quit = 0; Xstatic int (*old_sig)(); Xstatic FILE *out; Xstatic abort(); X X Xprintdb(fields, dbname) Xstruct Fdata *fields; Xchar *dbname; X{ X char *keys[MAXAK+1]; X char keyv[MAXAK][TSIZE+1]; X int nkeys, i; X struct Fdata *fp; X char aknum[MAXAK]; X int keynum; X char title[SWIDTH]; X char fname[FNSIZE]; X FILE *akfile, *dbfile; X char akrec[AKSIZE+1], dbrec[DBSIZE+1]; X long offset, atol(); X int width; X char *last; X char hline[1024]; X X sprintf(title, "Print %s data base", dbname); X while (1) { X last = "Enter ? for help."; X screen(title, first_screen, last, 0); X if (strcmp(ext_file, "?") == 0) { X strcpy(hline, "Enter the name of the extract file generated"); X strcat(hline, " from\nthe maintenance menu. If you leave it"); X strcat(hline, " blank, the entire\ndata base will be dumped."); X help(hline); X continue; X } X if (strcmp(out_width, "?") == 0) { X help("Enter the width of the output device, defaults to 80."); X continue; X } X if (*out_width == '\0') { X width = PWIDTH; X } else { X width = atoi(out_width); X } X if (strcmp(out_file, "?") == 0) { X strcpy(hline, "Enter the name of the output file,"); X strcat(hline, " defaults to '|lp', the printer."); X help(hline); X continue; X } X if (*out_file == '\0') { X strcpy(out_file, "|lp"); X } X if (strcmp(out_format, "?") == 0) { X strcpy(hline, "Enter the output format. All characters will"); X strcat(hline, " be\nprinted as entered except for % sequences."); X strcat(hline, "\n\n\t%NN\t - contents of field NN (1-maxfield)"); X strcat(hline, "\n\t%NN(form) - contents of field NN in"); X strcat(hline, " printf '%form' format\n"); X strcat(hline, "\t%n\t- new-line\n"); X strcat(hline, "\t%t\t- tab\n"); X strcat(hline, "\t%t(NN)\t- tab to column NN\n"); X strcat(hline, "\t%f\t- form-feed\n"); X strcat(hline, "\t%%\t- %"); X help(hline); X continue; X } X if (*out_format == '\0') { /* not specified, use default */ X fp = fields; X i = 1; X while (fp->F_title[0]) { X sprintf(&out_format[strlen(out_format)], "%s: %%%d%%n", X fp->F_title, i); X ++i; X ++fp; X } X strcat(out_format, "%n%n%n"); X } X break; X } X if (ext_file[0] == '\0') { X msg("No extract specified, entire database will be dumped"); X } X fp = fields; X nkeys = 0; X if (ext_file[0] == '\0') { X while (fp->F_title[0] != '\0') { X if (fp->F_key != 'N') { X sprintf(keyv[nkeys], "%-20s", fp->F_title); X keys[nkeys] = keyv[nkeys]; X aknum[nkeys] = fp->F_key; X ++nkeys; X } X ++fp; X } X keys[nkeys] = "Cancel "; X aknum[nkeys] = '\0'; X ++nkeys; X keys[nkeys] = 0; X sprintf(title, "Print %s data base sorted by", dbname); X keynum = menu(title, keys); X if (aknum[keynum] == '\0') { /* cancel */ X return(1); X } X if (*out_file == '|') X out = popen(out_file+1, "w"); X else X out = fopen(out_file, "w"); X if (out == NULL) { X msg("Unable to open output file"); X return(1); X } X X sprintf(fname, "%s%s.ak%c", datadir, dbname, aknum[keynum]); X if ((akfile = fopen(fname, "r")) == NULL) { X msg("Unable to open alternate key file"); X getout(); X } X sprintf(fname, "%s%s.db", datadir, dbname); X if ((dbfile = fopen(fname, "r")) == NULL) { X msg("Unable to open db file"); X getout(); X } X putp(tparm(cursor_address, MSGLINE, 9)); X putp(clr_eol); /* clear line */ X fputs("DEL to abort printing", stdout); X old_sig = signal(SIGINT, abort); X X while(!quit && fgets(akrec, AKSIZE, akfile) != NULL) { X offset = atol(strchr(akrec, ':')+1); X if (offset < 0L) X continue; X fseek(dbfile, offset, 0); X if (fgets(dbrec, DBSIZE, dbfile) == NULL) { X msg("Bad offset found"); X continue; X } X if (*dbrec == 'D') /* check for delete flag */ X continue; X putrcd(0, dbrec, out, out_format, width, 1); X } X fclose(akfile); X } else { X if ((dbfile = fopen(ext_file, "r")) == NULL) { SHAR_EOF echo "End of part 1" echo "File printdb.c is continued in part 2" echo "2" > s2_seq_.tmp exit 0