/*
- * $Id: serv_migrate.c 7274 2009-03-27 15:11:14Z ajc $
- * Copyright (c) 2000-2009 by the citadel.org development team
+ * $Id$
*
* This module dumps and/or loads the Citadel database in XML format.
*
+ * Copyright (c) 1987-2010 by the citadel.org team
+ *
* 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 3, or (at your option)
- * any later version.
+ * the Free Software Foundation; either version 3 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
*
* 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., 51 Franklin Street, Fifth Floor,
- * Boston, MA 02110-1301, USA.
- *
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include "sysdep.h"
#include "database.h"
#include "msgbase.h"
#include "user_ops.h"
-#include "room_ops.h"
#include "control.h"
#include "euidindex.h"
/* message list goes inside this tag */
- getroom(&CC->room, buf->QRname);
+ CtdlGetRoom(&CC->room, buf->QRname);
client_write("<room_messages>", 15);
client_write("<FRname>", 8); xml_strout(CC->room.QRname); client_write("</FRname>\n", 10);
client_write("<FRmsglist>", 11);
char cmd[SIZ];
migr_global_message_list = fopen(migr_tempfilename1, "w");
if (migr_global_message_list != NULL) {
- ForEachRoom(migr_export_rooms_backend, NULL);
+ CtdlForEachRoom(migr_export_rooms_backend, NULL);
fclose(migr_global_message_list);
}
for (i=0; i < MAXFLOORS; ++i) {
client_write("<floor>\n", 8);
cprintf("<f_num>%d</f_num>\n", i);
- getfloor(&qfbuf, i);
+ CtdlGetFloor(&qfbuf, i);
buf = &qfbuf;
cprintf("<f_flags>%u</f_flags>\n", buf->f_flags);
client_write("<f_name>", 8); xml_strout(buf->f_name); client_write("</f_name>\n", 10);
struct CtdlMessage *msg;
struct ser_ret smr;
+ /* We can use a static buffer here because there will never be more than
+ * one of this operation happening at any given time, and it's really best
+ * to just keep it allocated once instead of torturing malloc/free.
+ * Call this function with msgnum "-1" to free the buffer when finished.
+ */
+ static int encoded_alloc = 0;
+ static char *encoded_msg = NULL;
+
+ if (msgnum < 0) {
+ if ((encoded_alloc == 0) && (encoded_msg != NULL)) {
+ free(encoded_msg);
+ encoded_alloc = 0;
+ encoded_msg = NULL;
+ }
+ return;
+ }
+
+ /* Ok, here we go ... */
+
msg = CtdlFetchMessage(msgnum, 1);
if (msg == NULL) return; /* fail silently */
serialize_message(&smr, msg);
CtdlFreeMessage(msg);
- int encoded_len = 0;
- int encoded_alloc = smr.len * 14 / 10; /* well-tested formula for predicting encoded size */
- char *encoded_msg = malloc(encoded_alloc);
+ /* Predict the buffer size we need. Expand the buffer if necessary. */
+ int encoded_len = smr.len * 15 / 10 ;
+ if (encoded_len > encoded_alloc) {
+ encoded_alloc = encoded_len;
+ encoded_msg = realloc(encoded_msg, encoded_alloc);
+ }
- if (encoded_msg != NULL) {
+ if (encoded_msg == NULL) {
+ /* Questionable hack that hopes it'll work next time and we only lose one message */
+ encoded_alloc = 0;
+ }
+ else {
+ /* Once we do the encoding we know the exact size */
encoded_len = CtdlEncodeBase64(encoded_msg, (char *)smr.ser, smr.len, 1);
client_write(encoded_msg, encoded_len);
- free(encoded_msg);
}
free(smr.ser);
char buf[SIZ];
long msgnum;
int count = 0;
- t_context *Ctx;
+ CitContext *Ctx;
Ctx = CC;
migr_global_message_list = fopen(migr_tempfilename1, "r");
CtdlLogPrintf(CTDL_INFO, "Exported %d messages.\n", count);
else
CtdlLogPrintf(CTDL_ERR, "Export aborted due to client disconnect! \n");
+
+ migr_export_message(-1L); /* This frees the encoding buffer */
}
void migr_do_export(void) {
struct config *buf;
buf = &config;
- t_context *Ctx;
+ CitContext *Ctx;
Ctx = CC;
cprintf("%d Exporting all Citadel databases.\n", LISTING_FOLLOWS);
}
-void migr_xml_end(void *data, const char *el, const char **attr) {
+void migr_xml_end(void *data, const char *el) {
char *ptr;
int msgcount = 0;
long msgnum = 0L;
else if (!strcasecmp(el, "u_fullname")) safestrncpy(usbuf.fullname, migr_chardata, sizeof usbuf.fullname);
else if (!strcasecmp(el, "user")) {
- putuser(&usbuf);
+ CtdlPutUser(&usbuf);
CtdlLogPrintf(CTDL_INFO, "Imported user: %s\n", usbuf.fullname);
}
else if (!strcasecmp(el, "QRdefaultview")) qrbuf.QRdefaultview = atoi(migr_chardata);
else if (!strcasecmp(el, "room")) {
- putroom(&qrbuf);
+ CtdlPutRoom(&qrbuf);
CtdlLogPrintf(CTDL_INFO, "Imported room: %s\n", qrbuf.QRname);
}
}
}
if (msgcount > 0) {
- CtdlSaveMsgPointersInRoom(FRname, msglist, msgcount, 0, NULL);
+ CtdlSaveMsgPointersInRoom(FRname, msglist, msgcount, 0, NULL, 1);
}
free(msglist);
msglist = NULL;
else if (!strcasecmp(el, "f_ep_expire_value")) flbuf.f_ep.expire_value = atoi(migr_chardata);
else if (!strcasecmp(el, "floor")) {
- putfloor(&flbuf, floornum);
+ CtdlPutFloor(&flbuf, floornum);
CtdlLogPrintf(CTDL_INFO, "Imported floor #%d (%s)\n", floornum, flbuf.f_name);
}
+/*
+ * Dump out the pathnames of directories which can be copied "as is"
+ */
+void migr_do_listdirs(void) {
+ cprintf("%d Don't forget these:\n", LISTING_FOLLOWS);
+ cprintf("bio|%s\n", ctdl_bio_dir);
+ cprintf("files|%s\n", ctdl_file_dir);
+ cprintf("userpics|%s\n", ctdl_usrpic_dir);
+ cprintf("messages|%s\n", ctdl_message_dir);
+ cprintf("netconfigs|%s\n", ctdl_netcfg_dir);
+ cprintf("keys|%s\n", ctdl_key_dir);
+ cprintf("images|%s\n", ctdl_image_dir);
+ cprintf("info|%s\n", ctdl_info_dir);
+ cprintf("000\n");
+}
+
+
/*
* Common code appears in this section
*/
else if (!strcasecmp(cmd, "import")) {
migr_do_import();
}
+ else if (!strcasecmp(cmd, "listdirs")) {
+ migr_do_listdirs();
+ }
else {
cprintf("%d illegal command\n", ERROR + ILLEGAL_VALUE);
}
}
/* return our Subversion id for the Log */
- return "$Id: serv_migrate.c 7274 2009-03-27 15:11:14Z ajc $";
+ return "$Id$";
}