X-Git-Url: http://git.indexdata.com/?p=yaz-moved-to-github.git;a=blobdiff_plain;f=util%2Fmarcdisp.c;h=8b72818108eabd416dfc702db6d266382296007d;hp=9a33ebf8b12cb80f6091b586543733ce040f1622;hb=4d531a1a9131d69c3b6c27fbac42837e22cff61c;hpb=63cafe41a93427118959a74201b3e331169a71d9 diff --git a/util/marcdisp.c b/util/marcdisp.c index 9a33ebf..8b72818 100644 --- a/util/marcdisp.c +++ b/util/marcdisp.c @@ -1,46 +1,100 @@ /* - * Copyright (c) 1995, Index Data + * Copyright (c) 1995-2003, Index Data * See the file LICENSE for details. - * Sebastian Hammer, Adam Dickmeiss - * - * $Log: marcdisp.c,v $ - * Revision 1.4 1995-09-29 17:12:34 quinn - * Smallish - * - * Revision 1.3 1995/09/27 15:03:03 quinn - * Modified function heads & prototypes. - * - * Revision 1.2 1995/05/16 08:51:12 quinn - * License, documentation, and memory fixes - * - * Revision 1.1 1995/04/10 10:28:46 quinn - * Added copy of CCL and MARC display * + * $Id: marcdisp.c,v 1.28 2003-01-06 08:20:28 adam Exp $ */ +#if HAVE_CONFIG_H +#include +#endif + #include #include #include -#include +#include +#include +#include + +struct yaz_marc_t_ { + WRBUF m_wr; + int xml; + int debug; +}; + +yaz_marc_t yaz_marc_create(void) +{ + yaz_marc_t mt = xmalloc(sizeof(*mt)); + mt->xml = YAZ_MARC_LINE; + mt->debug = 0; + mt->m_wr = wrbuf_alloc(); + return mt; +} -#define ISO2709_RS 035 -#define ISO2709_FS 036 -#define ISO2709_IDFS 037 +void yaz_marc_destroy(yaz_marc_t mt) +{ + if (!mt) + return ; + wrbuf_free (mt->m_wr, 1); + xfree (mt); +} -int atoi_n (const char *buf, int len) +static void marc_cdata (yaz_marc_t mt, const char *buf, size_t len, WRBUF wr) { - int val = 0; + size_t i; + for (i = 0; ixml) + { + switch (buf[i]) { + case '<': + wrbuf_puts(wr, "<"); + break; + case '>': + wrbuf_puts(wr, ">"); + break; + case '&': + wrbuf_puts(wr, "&"); + break; + default: + wrbuf_putc(wr, buf[i]); + } + } + else + wrbuf_putc(wr, buf[i]); + } +} - while (--len >= 0) +#if 0 +static void marc_cdata (yaz_marc_t mt, const char *buf, size_t len) +{ + if (!mt->cd) + marc_cdata2 (mt, buf, len); + else { - if (isdigit (*buf)) - val = val*10 + (*buf - '0'); - buf++; + char outbuf[12]; + size_t inbytesleft = len; + const char *inp = buf; + + while (inbytesleft) + { + size_t outbytesleft = sizeof(outbuf); + char *outp = outbuf; + size_t r = yaz_iconv (mt->cd, (char**) &inp, &inbytesleft, + &outp, &outbytesleft); + if (r == (size_t) (-1)) + { + int e = yaz_iconv_error(mt->cd); + if (e != YAZ_ICONV_E2BIG) + break; + } + marc_cdata2 (mt, outbuf, outp - outbuf); + } } - return val; } +#endif -int marc_display (const char *buf, FILE *outf) +int yaz_marc_decode_wrbuf (yaz_marc_t mt, const char *buf, int bsize, WRBUF wr) { int entry_p; int record_length; @@ -51,19 +105,109 @@ int marc_display (const char *buf, FILE *outf) int length_starting; int length_implementation; + wrbuf_rewind(wr); + record_length = atoi_n (buf, 5); if (record_length < 25) + { + if (mt->debug) + { + char str[40]; + + sprintf (str, "Record length %d - aborting\n", record_length); + wrbuf_puts (wr, str); + } return -1; - indicator_length = atoi_n (buf+10, 1); - identifier_length = atoi_n (buf+11, 1); - base_address = atoi_n (buf+12, 4); + } + /* ballout if bsize is known and record_length is than that */ + if (bsize != -1 && record_length > bsize) + return -1; + if (isdigit(buf[10])) + indicator_length = atoi_n (buf+10, 1); + else + indicator_length = 2; + if (isdigit(buf[11])) + identifier_length = atoi_n (buf+11, 1); + else + identifier_length = 2; + base_address = atoi_n (buf+12, 5); length_data_entry = atoi_n (buf+20, 1); length_starting = atoi_n (buf+21, 1); length_implementation = atoi_n (buf+22, 1); + if (mt->xml) + { + char str[80]; + int i; + switch(mt->xml) + { + case YAZ_MARC_SIMPLEXML: + wrbuf_puts (wr, "\n"); + break; + case YAZ_MARC_OAIMARC: + wrbuf_puts( + wr, + "\n", + buf[5], buf[6], buf[7]); + wrbuf_puts (wr, str); + break; + case YAZ_MARC_MARCXML: + wrbuf_printf( + wr, + "\n" + " %.24s\n", buf); + break; + } + } + if (mt->debug) + { + char str[40]; + + if (mt->xml) + wrbuf_puts (wr, "\n"); + } + for (entry_p = 24; buf[entry_p] != ISO2709_FS; ) + { entry_p += 3+length_data_entry+length_starting; + if (entry_p >= record_length) + return -1; + } base_address = entry_p+1; for (entry_p = 24; buf[entry_p] != ISO2709_FS; ) { @@ -72,44 +216,246 @@ int marc_display (const char *buf, FILE *outf) int end_offset; int i, j; char tag[4]; + int identifier_flag = 1; memcpy (tag, buf+entry_p, 3); entry_p += 3; tag[3] = '\0'; - fprintf (outf, "%s ", tag); data_length = atoi_n (buf+entry_p, length_data_entry); entry_p += length_data_entry; data_offset = atoi_n (buf+entry_p, length_starting); entry_p += length_starting; i = data_offset + base_address; end_offset = i+data_length-1; - if (memcmp (tag, "00", 2) && indicator_length) + + if (indicator_length < 4 && indicator_length > 0) + { + if (buf[i + indicator_length] != ISO2709_IDFS) + identifier_flag = 0; + } + else if (!memcmp (tag, "00", 2)) + identifier_flag = 0; + + switch(mt->xml) + { + case YAZ_MARC_LINE: + if (mt->debug) + wrbuf_puts (wr, "Tag: "); + wrbuf_puts (wr, tag); + wrbuf_puts (wr, " "); + break; + case YAZ_MARC_SIMPLEXML: + wrbuf_printf (wr, "xml) + { + case YAZ_MARC_LINE: + if (mt->debug) + wrbuf_puts (wr, " Ind: "); + wrbuf_putc (wr, buf[i]); + break; + case YAZ_MARC_SIMPLEXML: + wrbuf_printf (wr, " Indicator%d=\"%c\"", j+1, buf[i]); + break; + case YAZ_MARC_OAIMARC: + wrbuf_printf (wr, " i%d=\"%c\"", j+1, buf[i]); + break; + case YAZ_MARC_MARCXML: + wrbuf_printf (wr, " ind%d=\"%c\"", j+1, buf[i]); + } + } } - while (buf[i] != ISO2709_RS && buf[i] != ISO2709_FS && i < end_offset) - { - if (memcmp (tag, "00", 2) && identifier_length) - { - i++; - fprintf (outf, " $"); - for (j = 1; jxml) + { + wrbuf_puts (wr, ">"); + if (identifier_flag) + wrbuf_puts (wr, "\n"); + } + else + { + if (mt->debug && !mt->xml) + wrbuf_puts (wr, " Fields: "); + } + if (identifier_flag) + { + while (buf[i] != ISO2709_RS && buf[i] != ISO2709_FS && i < end_offset) + { + int i0; + i++; + switch(mt->xml) + { + case YAZ_MARC_LINE: + wrbuf_puts (wr, " $"); + for (j = 1; j"); + break; + case YAZ_MARC_OAIMARC: + wrbuf_puts (wr, " "); + break; + case YAZ_MARC_MARCXML: + wrbuf_puts (wr, " "); + break; + } + i0 = i; + while (buf[i] != ISO2709_RS && buf[i] != ISO2709_IDFS && + buf[i] != ISO2709_FS && i < end_offset) + i++; + marc_cdata(mt, buf + i0, i - i0, wr); + + if (mt->xml) + wrbuf_puts (wr, "\n"); + } + } + else + { + int i0 = i; + while (buf[i] != ISO2709_RS && buf[i] != ISO2709_FS && i < end_offset) + i++; + marc_cdata(mt, buf + i0, i - i0, wr); } - fprintf (outf, "\n"); + if (!mt->xml) + wrbuf_putc (wr, '\n'); if (i < end_offset) - fprintf (outf, "-- separator but not at end of field\n"); + wrbuf_puts (wr, " \n"); if (buf[i] != ISO2709_RS && buf[i] != ISO2709_FS) - fprintf (outf, "-- no separator at end of field\n"); + wrbuf_puts (wr, " \n"); + switch(mt->xml) + { + case YAZ_MARC_SIMPLEXML: + wrbuf_puts (wr, "\n"); + break; + case YAZ_MARC_OAIMARC: + if (identifier_flag) + wrbuf_puts (wr, " \n"); + else + wrbuf_puts (wr, " \n"); + break; + case YAZ_MARC_MARCXML: + if (identifier_flag) + wrbuf_puts (wr, " \n"); + else + wrbuf_puts (wr, " \n"); + break; + } + } + switch (mt->xml) + { + case YAZ_MARC_LINE: + wrbuf_puts (wr, ""); + break; + case YAZ_MARC_SIMPLEXML: + wrbuf_puts (wr, "\n"); + break; + case YAZ_MARC_OAIMARC: + wrbuf_puts (wr, "\n"); + break; + case YAZ_MARC_MARCXML: + wrbuf_puts (wr, "\n"); + break; } return record_length; } +int yaz_marc_decode_buf (yaz_marc_t mt, const char *buf, int bsize, + char **result, int *rsize) +{ + int r = yaz_marc_decode_wrbuf(mt, buf, bsize, mt->m_wr); + if (r > 0) + { + if (result) + *result = wrbuf_buf(mt->m_wr); + if (rsize) + *rsize = wrbuf_len(mt->m_wr); + } + return r; +} + +void yaz_marc_xml(yaz_marc_t mt, int xmlmode) +{ + if (mt) + mt->xml = xmlmode; +} + +void yaz_marc_debug(yaz_marc_t mt, int level) +{ + if (mt) + mt->debug = level; +} + +/* depricated */ +int yaz_marc_decode(const char *buf, WRBUF wr, int debug, int bsize, int xml) +{ + yaz_marc_t mt = yaz_marc_create(); + int r; + + mt->debug = debug; + mt->xml = xml; + r = yaz_marc_decode_wrbuf(mt, buf, bsize, wr); + yaz_marc_destroy(mt); + return r; +} + +/* depricated */ +int marc_display_wrbuf (const char *buf, WRBUF wr, int debug, int bsize) +{ + return yaz_marc_decode(buf, wr, debug, bsize, 0); +} + +/* depricated */ +int marc_display_exl (const char *buf, FILE *outf, int debug, int bsize) +{ + yaz_marc_t mt = yaz_marc_create(); + int r; + + mt->debug = debug; + r = yaz_marc_decode_wrbuf (mt, buf, bsize, mt->m_wr); + if (!outf) + outf = stdout; + if (r > 0) + fwrite (wrbuf_buf(mt->m_wr), 1, wrbuf_len(mt->m_wr), outf); + yaz_marc_destroy(mt); + return r; +} + +/* depricated */ +int marc_display_ex (const char *buf, FILE *outf, int debug) +{ + return marc_display_exl (buf, outf, debug, -1); +} + +/* depricated */ +int marc_display (const char *buf, FILE *outf) +{ + return marc_display_ex (buf, outf, 0); +} +