⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 file_cdb.c

📁 打魔兽战网的都知道他是什么
💻 C
字号:
/* * Copyright (C) 2003,2004 Dizzy  * * 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. */#include "common/setup_before.h"#include <stdio.h>#ifdef HAVE_STDDEF_H# include <stddef.h>#else# ifndef NULL#  define NULL ((void *)0)# endif#endif#ifdef STDC_HEADERS# include <stdlib.h>#else# ifdef HAVE_MALLOC_H#  include <malloc.h># endif#endif#ifdef HAVE_STRING_H# include <string.h>#else# ifdef HAVE_STRINGS_H#  include <strings.h># endif#endif#include "compat/strchr.h"#include "compat/strdup.h"#include "compat/strcasecmp.h"#include "compat/strncasecmp.h"#include <ctype.h>#ifdef HAVE_LIMITS_H# include <limits.h>#endif#include "compat/char_bit.h"#ifdef TIME_WITH_SYS_TIME# include <sys/time.h># include <time.h>#else# ifdef HAVE_SYS_TIME_H#  include <sys/time.h># else#  include <time.h># endif#endif#include <errno.h>#include "compat/strerror.h"#ifdef HAVE_SYS_TYPES_H# include <sys/types.h>#endif#ifdef HAVE_UNISTD_H# include <unistd.h>#else# ifdef WIN32#  include <io.h># endif#endif#include "compat/pdir.h"#include "common/eventlog.h"#include "prefs.h"#include "common/util.h"#include "common/field_sizes.h"#include "common/bnethash.h"#include "common/introtate.h"#define CLAN_INTERNAL_ACCESS#include "account.h"#include "common/hashtable.h"#include "storage.h"#include "storage_file.h"#include "common/list.h"#include "common/xalloc.h"#include "connection.h"#include "watch.h"#include "clan.h"#undef CLAN_INTERNAL_ACCESS#include "common/tag.h"#ifdef HAVE_FCNTL_H# include <fcntl.h>#else# ifdef HAVE_SYS_FILE_H#  include <sys/file.h># endif#endif#include "tinycdb/cdb.h"#include "common/elist.h"#include "attr.h"#include "common/setup_after.h"/* cdb file storage API functions */static int cdb_read_attrs(const char *filename, t_read_attr_func cb, void *data);static t_attr * cdb_read_attr(const char *filename, const char *key);static int cdb_write_attrs(const char *filename, const t_hlist *attributes);/* file_engine struct populated with the functions above */t_file_engine file_cdb = {    cdb_read_attr,    cdb_read_attrs,    cdb_write_attrs};/* start of actual cdb file storage code *///#define CDB_ON_DEMAND	1static int cdb_write_attrs(const char *filename, const t_hlist *attributes){    FILE           *cdbfile;    t_hlist	   *curr;    t_attr         *attr;    struct cdb_make cdbm;    if ((cdbfile = fopen(filename, "w+b")) == NULL) {	eventlog(eventlog_level_error, __FUNCTION__, "unable to open file \"%s\" for writing ",filename);	return -1;    }    cdb_make_start(&cdbm, cdbfile);    hlist_for_each(curr,attributes) {	attr = hlist_entry(curr, t_attr, link);	if (attr_get_key(attr) && attr_get_val(attr)) {	    if (strncmp("BNET\\CharacterDefault\\", attr_get_key(attr), 20) == 0) {		eventlog(eventlog_level_debug, __FUNCTION__, "skipping attribute key=\"%s\"",attr_get_key(attr));	    } else {		eventlog(eventlog_level_debug, __FUNCTION__, "saving attribute key=\"%s\" val=\"%s\"",attr_get_key(attr),attr_get_val(attr));		if (cdb_make_add(&cdbm, attr_get_key(attr), strlen(attr_get_key(attr)), attr_get_val(attr), strlen(attr_get_val(attr)))<0)		{		    eventlog(eventlog_level_error, __FUNCTION__, "got error on cdb_make_add ('%s' = '%s')", attr_get_key(attr), attr_get_val(attr));		    cdb_make_finish(&cdbm); /* try to bail out nicely */		    fclose(cdbfile);		    return -1;		}	    }	} else eventlog(eventlog_level_error, __FUNCTION__,"could not save attribute key=\"%s\"",attr_get_key(attr));	attr_clear_dirty(attr);    }    if (cdb_make_finish(&cdbm)<0) {	eventlog(eventlog_level_error, __FUNCTION__, "got error on cdb_make_finish");	fclose(cdbfile);	return -1;    }    if (fclose(cdbfile)<0) {	eventlog(eventlog_level_error, __FUNCTION__, "got error on fclose()");	return -1;    }    return 0;}#ifndef CDB_ON_DEMAND/* code adapted from tinycdb-0.73/cdb.c */static int fget(FILE * fd, unsigned char *b, cdbi_t len, cdbi_t *posp, cdbi_t limit){    if (posp && limit - *posp < len) {	eventlog(eventlog_level_error, __FUNCTION__, "invalid cdb database format");	return -1;    }    if (fread(b, 1, len, fd) != len) {	if (ferror(fd)) {	    eventlog(eventlog_level_error, __FUNCTION__, "got error reading from db file");	    return -1;	}	eventlog(eventlog_level_error, __FUNCTION__, "unable to read from cdb file, incomplete file");	return -1;    }    if (posp) *posp += len;    return 0;}static const char * fcpy(FILE *fd, cdbi_t len, cdbi_t *posp, cdbi_t limit, unsigned char * buf){    static char *str;    static unsigned strl;    unsigned int res = 0, no = 0;    if (strl < len + 1) {	char *tmp;	tmp = xmalloc(len + 1);	if (str) xfree((void*)str);	str = tmp;	strl = len + 1;    }    while(len - res > 0) {	if (len > 2048) no = 2048;	else no = len;	if (fget(fd, buf, no, posp, limit)) return NULL;	memmove(str + res, buf, no);	res += no;    }    if (res > strl - 1) {	eventlog(eventlog_level_error, __FUNCTION__, "BUG, this should not happen");	return NULL;    }    str[res] = '\0';    return str;}static int cdb_read_attrs(const char *filename, t_read_attr_func cb, void *data){    cdbi_t eod, klen, vlen;    cdbi_t pos = 0;    const char *key;    const char *val;    unsigned char buf[2048];    FILE *f;    if ((f = fopen(filename, "rb")) == NULL) {	eventlog(eventlog_level_error, __FUNCTION__, "got error opening file '%s'", filename);	return -1;    }    if (fget(f, buf, 2048, &pos, 2048)) goto err_fd;    eod = cdb_unpack(buf);    while(pos < eod) {	if (fget(f, buf, 8, &pos, eod)) goto err_fd;	klen = cdb_unpack(buf);	vlen = cdb_unpack(buf + 4);	if ((key = fcpy(f, klen, &pos, eod, buf)) == NULL) {	    eventlog(eventlog_level_error, __FUNCTION__, "error reading attribute key");	    goto err_fd;	}	key = xstrdup(key);	if ((val = fcpy(f, vlen, &pos, eod, buf)) == NULL) {	    eventlog(eventlog_level_error, __FUNCTION__, "error reading attribute val");	    goto err_key;	}//	eventlog(eventlog_level_trace, __FUNCTION__, "read atribute : '%s' -> '%s'", key, val);	if (cb(key, val, data))	    eventlog(eventlog_level_error, __FUNCTION__, "got error from callback on account file '%s'", filename);	xfree((void *)key);    }    fclose(f);    return 0;err_key:    xfree((void *)key);err_fd:    fclose(f);    return -1;}#else /* CDB_ON_DEMAND */static int cdb_read_attrs(const char *filename, t_read_attr_func cb, void *data){    return 0;}#endifstatic t_attr * cdb_read_attr(const char *filename, const char *key){#ifdef CDB_ON_DEMAND    FILE	*cdbfile;    t_attr	*attr;    char	*val;    unsigned	vlen = 1;//    eventlog(eventlog_level_trace, __FUNCTION__, "reading key '%s'", key);    if ((cdbfile = fopen(filename, "rb")) == NULL) {//	eventlog(eventlog_level_debug, __FUNCTION__, "unable to open file \"%s\" for reading ",filename);	return NULL;    }    if (cdb_seek(cdbfile, key, strlen(key), &vlen) <= 0) {//	eventlog(eventlog_level_debug, __FUNCTION__, "could not find key '%s'", key);	fclose(cdbfile);	return NULL;    }    /* FIXME: use attr_* API */    attr = xmalloc(sizeof(t_attr));    attr->key = xstrdup(key);    val = xmalloc(vlen + 1);    cdb_bread(cdbfile, val, vlen);    fclose(cdbfile);    val[vlen] = '\0';    attr->val = val;    attr->dirty = 0;//    eventlog(eventlog_level_trace, __FUNCTION__, "read key '%s' value '%s'", attr->key, attr->val);    return attr;#else    return NULL;#endif}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -