server/src/kernel/textstore.c

191 lines
4.7 KiB
C
Raw Normal View History

2010-08-08 10:06:34 +02:00
/* vi: set ts=2:
+-------------------+ Enno Rehling <enno@eressea.de>
| Eressea PBEM host | Christian Schlittchen <corwin@amber.kn-bremen.de>
| (c) 1998 - 2008 | Katja Zedel <katze@felidae.kn-bremen.de>
+-------------------+
This program may not be used, modified or distributed
without prior permission by the authors of Eressea.
*/
#include <platform.h>
#include "config.h"
#include "textstore.h"
#include "save.h"
#include "version.h"
#include <util/base36.h>
#include <util/log.h>
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <libxml/encoding.h>
#define NULL_TOKEN '@'
2011-03-07 08:02:35 +01:00
static int txt_w_brk(struct storage *store)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
putc('\n', (FILE *) store->userdata);
2010-08-08 10:06:34 +02:00
return 1;
}
2011-03-07 08:02:35 +01:00
static int txt_w_id(struct storage *store, int arg)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
return fprintf((FILE *) store->userdata, "%s ", itoa36(arg));
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static int txt_r_id(struct storage *store)
2010-08-08 10:06:34 +02:00
{
char id[8];
2011-03-07 08:02:35 +01:00
fscanf((FILE *) store->userdata, "%7s", id);
2010-08-08 10:06:34 +02:00
return atoi36(id);
}
2011-03-07 08:02:35 +01:00
static int txt_w_int(struct storage *store, int arg)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
return fprintf((FILE *) store->userdata, "%d ", arg);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static int txt_r_int(struct storage *store)
2010-08-08 10:06:34 +02:00
{
int result;
2011-03-07 08:02:35 +01:00
fscanf((FILE *) store->userdata, "%d", &result);
2010-08-08 10:06:34 +02:00
return result;
}
2011-03-07 08:02:35 +01:00
static int txt_w_flt(struct storage *store, float arg)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
return fprintf((FILE *) store->userdata, "%f ", arg);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static float txt_r_flt(struct storage *store)
2010-08-08 10:06:34 +02:00
{
double result;
2011-03-08 08:44:20 +01:00
fscanf((FILE *) store->userdata, "%lf", &result);
2010-08-08 10:06:34 +02:00
return (float)result;
}
2011-03-07 08:02:35 +01:00
static int txt_w_tok(struct storage *store, const char *tok)
2010-08-08 10:06:34 +02:00
{
int result;
2011-03-07 08:02:35 +01:00
if (tok == NULL || tok[0] == 0) {
result = fputc(NULL_TOKEN, (FILE *) store->userdata);
2010-08-08 10:06:34 +02:00
} else {
#ifndef NDEBUG
2011-03-07 08:02:35 +01:00
const char *find = strchr(tok, ' ');
if (!find)
find = strchr(tok, NULL_TOKEN);
2010-08-08 10:06:34 +02:00
assert(!find || !"reserved character in token");
#endif
2011-03-07 08:02:35 +01:00
assert(tok[0] != ' ');
result = fputs(tok, (FILE *) store->userdata);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
fputc(' ', (FILE *) store->userdata);
2010-08-08 10:06:34 +02:00
return result;
}
2011-03-07 08:02:35 +01:00
static char *txt_r_tok(struct storage *store)
2010-08-08 10:06:34 +02:00
{
char result[256];
2011-03-07 08:02:35 +01:00
fscanf((FILE *) store->userdata, "%256s", result);
if (result[0] == NULL_TOKEN || result[0] == 0) {
2010-08-08 10:06:34 +02:00
return NULL;
}
return strdup(result);
}
2011-03-07 08:02:35 +01:00
static void txt_r_tok_buf(struct storage *store, char *result, size_t size)
2010-08-08 10:06:34 +02:00
{
char format[16];
2011-03-07 08:02:35 +01:00
if (result && size > 0) {
format[0] = '%';
sprintf(format + 1, "%us", size);
fscanf((FILE *) store->userdata, format, result);
if (result[0] == NULL_TOKEN) {
2010-08-08 10:06:34 +02:00
result[0] = 0;
}
} else {
/* trick to skip when no result expected */
2011-03-07 08:02:35 +01:00
fscanf((FILE *) store->userdata, "%*s");
2010-08-08 10:06:34 +02:00
}
}
2011-03-07 08:02:35 +01:00
static int txt_w_str(struct storage *store, const char *str)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
int result = fwritestr((FILE *) store->userdata, str);
fputc(' ', (FILE *) store->userdata);
return result + 1;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static char *txt_r_str(struct storage *store)
2010-08-08 10:06:34 +02:00
{
char buffer[DISPLAYSIZE];
/* you should not use this */
2011-03-07 08:02:35 +01:00
freadstr((FILE *) store->userdata, store->encoding, buffer, sizeof(buffer));
2010-08-08 10:06:34 +02:00
return strdup(buffer);
}
2011-03-07 08:02:35 +01:00
static void txt_r_str_buf(struct storage *store, char *result, size_t size)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
freadstr((FILE *) store->userdata, store->encoding, result, size);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static int txt_open(struct storage *store, const char *filename, int mode)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
const char *modes[] = { 0, "rt", "wt", "at" };
FILE *F = fopen(filename, modes[mode]);
2010-08-08 10:06:34 +02:00
store->userdata = F;
if (F) {
2011-03-08 07:31:06 +01:00
const unsigned char utf8_bom[4] = { 0xef, 0xbb, 0xbf };
2011-03-07 08:02:35 +01:00
if (mode == IO_READ) {
2010-08-08 10:06:34 +02:00
char token[8];
/* recognize UTF8 BOM */
store->r_tok_buf(store, token, sizeof(token));
2011-03-07 08:02:35 +01:00
if (memcmp(token, utf8_bom, 3) == 0) {
if (enc_gamedata != XML_CHAR_ENCODING_UTF8) {
2010-08-08 10:06:34 +02:00
log_warning(("Found UTF-8 BOM, assuming unicode game data.\n"));
store->encoding = XML_CHAR_ENCODING_UTF8;
}
2011-03-07 08:02:35 +01:00
store->version = atoi(token + 3);
2010-08-08 10:06:34 +02:00
} else {
2011-03-07 08:02:35 +01:00
if (store->encoding == XML_CHAR_ENCODING_NONE) {
2010-08-08 10:06:34 +02:00
store->encoding = XML_CHAR_ENCODING_8859_1;
log_warning(("No BOM, assuming 8859-1 game data.\n"));
}
store->version = atoi(token);
}
2011-03-07 08:02:35 +01:00
} else if (store->encoding == XML_CHAR_ENCODING_UTF8) {
2011-03-08 08:44:20 +01:00
fputs((const char *)utf8_bom, F);
2010-08-08 10:06:34 +02:00
fprintf(F, "%d\n", RELEASE_VERSION);
}
}
2011-03-07 08:02:35 +01:00
return (F == NULL);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static int txt_w_bin(struct storage *store, void *arg, size_t size)
2010-08-08 10:06:34 +02:00
{
assert(!"not implemented!");
return 0;
}
2011-03-07 08:02:35 +01:00
static void txt_r_bin(struct storage *store, void *result, size_t size)
2010-08-08 10:06:34 +02:00
{
assert(!"not implemented!");
}
2011-03-07 08:02:35 +01:00
static int txt_close(struct storage *store)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
return fclose((FILE *) store->userdata);
2010-08-08 10:06:34 +02:00
}
const storage text_store = {
txt_w_brk,
txt_w_int, txt_r_int,
txt_w_flt, txt_r_flt,
txt_w_id, txt_r_id,
txt_w_tok, txt_r_tok, txt_r_tok_buf,
txt_w_str, txt_r_str, txt_r_str_buf,
2011-03-07 08:02:35 +01:00
txt_w_bin, txt_r_bin,
2010-08-08 10:06:34 +02:00
txt_open, txt_close,
0, 0, NULL
};