aboutsummaryrefslogblamecommitdiffstats
path: root/e-util/e-db3-utils.c
blob: c1da265453cbf4b4f49d0ff54ffe6dc57a3f9626 (plain) (tree)























































































































































































                                                                                            
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */

#include "config.h"

#include "e-db3-utils.h"

#include <db.h>

#include <errno.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <libgnome/gnome-defs.h>
#include <libgnome/gnome-util.h>

#if DB_VERSION_MAJOR != 3 || \
    DB_VERSION_MINOR != 1 || \
    DB_VERSION_PATCH != 17
#error Including wrong DB3.  Need libdb 3.1.17.
#endif

static char *
get_check_filename (const char *filename)
{
    return g_strdup_printf ("%s-upgrading", filename);
}

static char *
get_copy_filename (const char *filename)
{
    return g_strdup_printf ("%s-copy", filename);
}

static int
cp_file (const char *src, const char *dest)
{
    int i;
    int o;
    char buffer[1024];
    int length;
    int place;

    i = open (src, O_RDONLY);
    if (i == -1)
        return -1;
    o = creat (dest, S_IREAD | S_IWRITE);
    if (o == -1) {
        close (i);
        return -1;
    }
    while (1) {
        length = read (i, &buffer, sizeof (buffer));

        if (length == 0)
            break;

        if (length == -1) {
            if (errno == EINTR)
                continue;
            else {
                close (i);
                close (o);
                unlink (dest);
                return -1;
            }
        }

        place = 0;
        while (length != 0) {
            int count;
            count = write (o, buffer + place, length);
            if (count == -1) {
                if (errno == EINTR)
                    continue;
                else {
                    close (i);
                    close (o);
                    unlink (dest);
                    return -1;
                }
            }

            length -= count;
            place += count;
        }
    }
    if (close (i))
        return -1;
    if (close (o))
        return -1;
    return 0;
}

static int
touch_file (const char *file)
{
    int o;
    o = creat (file, S_IREAD | S_IWRITE);
    if (o == -1)
        return -1;

    if (close (o) == -1)
        return -1;

    return 0;
}

static int
resume_upgrade (const char *filename, const char *copy_filename, const char *check_filename)
{
    DB *db;
    int ret_val;

    ret_val = db_create (&db, NULL, 0);

    if (ret_val == 0)
        ret_val = cp_file (copy_filename, filename);

    if (ret_val == 0)
        ret_val = db->upgrade (db, filename, 0);

    if (ret_val == 0)
        ret_val = unlink (check_filename);
    if (ret_val == 0)
        ret_val = unlink (copy_filename);

    db->close (db, 0);

    return ret_val;
}

int
e_db3_utils_maybe_recover (const char *filename)
{
    int ret_val = 0;
    char *copy_filename;
    char *check_filename;

    copy_filename = get_copy_filename (filename);
    check_filename = get_check_filename (filename);

    if (g_file_exists (check_filename)) {
        ret_val = resume_upgrade(filename, copy_filename, check_filename);
    } else if (g_file_exists (copy_filename)) {
        unlink (copy_filename);
    }

    g_free (copy_filename);
    g_free (check_filename);
    return ret_val;
}

int
e_db3_utils_upgrade_format (const char *filename)
{
    char *copy_filename;
    char *check_filename;
    DB *db;
    int ret_val;

    ret_val = db_create (&db, NULL, 0);
    if (ret_val != 0)
        return ret_val;

    copy_filename = get_copy_filename (filename);
    check_filename = get_check_filename (filename);

    ret_val = cp_file (filename, copy_filename);

    if (ret_val == 0)
        ret_val = touch_file (check_filename);
    if (ret_val == 0)
        ret_val = db->upgrade (db, filename, 0);
    if (ret_val == 0)
        ret_val = unlink (check_filename);

    if (ret_val == 0)
        ret_val = unlink (copy_filename);

    db->close (db, 0);

    g_free (check_filename);
    g_free (copy_filename);
    return ret_val;
}