/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* Authors: Jeffrey Stedfast <fejj@ximian.com>
*
* Copyright 2002 Ximian, Inc. (www.ximian.com)
*
* 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 Street #330, Boston, MA 02111-1307, USA.
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <string.h>
#include "camel-multipart-encrypted.h"
#include "camel-mime-filter-crlf.h"
#include "camel-stream-filter.h"
#include "camel-stream-mem.h"
#include "camel-stream-fs.h"
#include "camel-mime-utils.h"
#include "camel-mime-part.h"
static void camel_multipart_encrypted_class_init (CamelMultipartEncryptedClass *klass);
static void camel_multipart_encrypted_init (gpointer object, gpointer klass);
static void camel_multipart_encrypted_finalize (CamelObject *object);
static void set_mime_type_field (CamelDataWrapper *data_wrapper, CamelContentType *mime_type);
static CamelMultipartClass *parent_class = NULL;
CamelType
camel_multipart_encrypted_get_type (void)
{
static CamelType type = CAMEL_INVALID_TYPE;
if (type == CAMEL_INVALID_TYPE) {
type = camel_type_register (camel_multipart_get_type (),
"CamelMultipartEncrypted",
sizeof (CamelMultipartEncrypted),
sizeof (CamelMultipartEncryptedClass),
(CamelObjectClassInitFunc) camel_multipart_encrypted_class_init,
NULL,
(CamelObjectInitFunc) camel_multipart_encrypted_init,
(CamelObjectFinalizeFunc) camel_multipart_encrypted_finalize);
}
return type;
}
static void
camel_multipart_encrypted_class_init (CamelMultipartEncryptedClass *klass)
{
CamelDataWrapperClass *camel_data_wrapper_class = CAMEL_DATA_WRAPPER_CLASS (klass);
parent_class = (CamelMultipartClass *) camel_multipart_get_type ();
/* virtual method overload */
camel_data_wrapper_class->set_mime_type_field = set_mime_type_field;
}
static void
camel_multipart_encrypted_init (gpointer object, gpointer klass)
{
CamelMultipartEncrypted *multipart = (CamelMultipartEncrypted *) object;
camel_data_wrapper_set_mime_type (CAMEL_DATA_WRAPPER (multipart), "multipart/encrypted");
multipart->decrypted = NULL;
}
static void
camel_multipart_encrypted_finalize (CamelObject *object)
{
CamelMultipartEncrypted *mpe = (CamelMultipartEncrypted *) object;
g_free (mpe->protocol);
if (mpe->decrypted)
camel_object_unref (mpe->decrypted);
}
/* we snoop the mime type to get the protocol */
static void
set_mime_type_field (CamelDataWrapper *data_wrapper, CamelContentType *mime_type)
{
CamelMultipartEncrypted *mpe = (CamelMultipartEncrypted *) data_wrapper;
if (mime_type) {
const char *protocol;
protocol = camel_content_type_param (mime_type, "protocol");
g_free (mpe->protocol);
mpe->protocol = g_strdup (protocol);
}
((CamelDataWrapperClass *) parent_class)->set_mime_type_field (data_wrapper, mime_type);
}
/**
* camel_multipart_encrypted_new:
*
* Create a new CamelMultipartEncrypted object.
*
* A MultipartEncrypted should be used to store and create parts of
* type "multipart/encrypted".
*
* Returns a new CamelMultipartEncrypted
**/
CamelMultipartEncrypted *
camel_multipart_encrypted_new (void)
{
CamelMultipartEncrypted *multipart;
multipart = (CamelMultipartEncrypted *) camel_object_new (CAMEL_MULTIPART_ENCRYPTED_TYPE);
return multipart;
}
int
camel_multipart_encrypted_encrypt (CamelMultipartEncrypted *mpe, CamelMimePart *content,
CamelCipherContext *cipher, const char *userid,
GPtrArray *recipients, CamelException *ex)
{
abort();
#if 0
CamelMimePart *version_part, *encrypted_part;
CamelContentType *mime_type;
CamelDataWrapper *wrapper;
CamelStream *stream;
g_return_val_if_fail (CAMEL_IS_MULTIPART_ENCRYPTED (mpe), -1);
g_return_val_if_fail (CAMEL_IS_CIPHER_CONTEXT (cipher), -1);
g_return_val_if_fail (cipher->encrypt_protocol != NULL, -1);
g_return_val_if_fail (CAMEL_IS_MIME_PART (content), -1);
/* encrypt the content stream */
encrypted_part = camel_mime_part_new();
if (camel_cipher_encrypt (cipher, userid, recipients, content, encrypted_part, ex) == -1) {
camel_object_unref(encrypted_part);
return -1;
}
/* construct the version part */
stream = camel_stream_mem_new ();
camel_stream_write_string (stream, "Version: 1\n");
camel_stream_reset (stream);
version_part = camel_mime_part_new ();
wrapper = camel_data_wrapper_new ();
camel_data_wrapper_set_mime_type (wrapper, cipher->encrypt_protocol);
camel_data_wrapper_construct_from_stream (wrapper, stream);
camel_object_unref (stream);
camel_medium_set_content_object ((CamelMedium *) version_part, wrapper);
camel_object_unref (wrapper);
/* save the version and encrypted parts */
/* FIXME: make sure there aren't any other parts?? */
camel_multipart_add_part (CAMEL_MULTIPART (mpe), version_part);
camel_object_unref (version_part);
camel_multipart_add_part (CAMEL_MULTIPART (mpe), encrypted_part);
camel_object_unref (encrypted_part);
/* cache the decrypted content */
camel_object_ref (content);
mpe->decrypted = content;
/* set the content-type params for this multipart/encrypted part */
mime_type = camel_content_type_new ("multipart", "encrypted");
camel_content_type_set_param (mime_type, "protocol", cipher->encrypt_protocol);
camel_data_wrapper_set_mime_type_field (CAMEL_DATA_WRAPPER (mpe), mime_type);
camel_content_type_unref (mime_type);
camel_multipart_set_boundary ((CamelMultipart *) mpe, NULL);
#endif
return 0;
}
CamelMimePart *
camel_multipart_encrypted_decrypt (CamelMultipartEncrypted *mpe,
CamelCipherContext *cipher,
CamelException *ex)
{
CamelMimePart *version_part, *encrypted_part, *decrypted_part;
CamelContentType *mime_type;
CamelCipherValidity *valid;
CamelDataWrapper *wrapper;
const char *protocol;
char *content_type;
g_return_val_if_fail (CAMEL_IS_MULTIPART_ENCRYPTED (mpe), NULL);
g_return_val_if_fail (CAMEL_IS_CIPHER_CONTEXT (cipher), NULL);
g_return_val_if_fail (cipher->encrypt_protocol != NULL, NULL);
if (mpe->decrypted) {
/* we seem to have already decrypted the part */
camel_object_ref (mpe->decrypted);
return mpe->decrypted;
}
protocol = mpe->protocol;
if (protocol) {
/* make sure the protocol matches the cipher encrypt protocol */
if (g_ascii_strcasecmp (cipher->encrypt_protocol, protocol) != 0) {
camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
_("Failed to decrypt MIME part: protocol error"));
return NULL;
}
} else {
/* *shrug* - I guess just go on as if they match? */
protocol = cipher->encrypt_protocol;
}
/* make sure the protocol matches the version part's content-type */
version_part = camel_multipart_get_part (CAMEL_MULTIPART (mpe), CAMEL_MULTIPART_ENCRYPTED_VERSION);
wrapper = camel_medium_get_content_object (CAMEL_MEDIUM (version_part));
content_type = camel_data_wrapper_get_mime_type (wrapper);
if (g_ascii_strcasecmp (content_type, protocol) != 0) {
camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
_("Failed to decrypt MIME part: protocol error"));
g_free (content_type);
return NULL;
}
g_free (content_type);
/* get the encrypted part (second part) */
encrypted_part = camel_multipart_get_part (CAMEL_MULTIPART (mpe), CAMEL_MULTIPART_ENCRYPTED_CONTENT);
mime_type = camel_mime_part_get_content_type (encrypted_part);
if (!camel_content_type_is (mime_type, "application", "octet-stream")) {
camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
_("Failed to decrypt MIME part: invalid structure"));
return NULL;
}
decrypted_part = camel_mime_part_new();
valid = camel_cipher_decrypt(cipher, encrypted_part, decrypted_part, ex);
if (valid) {
camel_object_ref(decrypted_part);
mpe->decrypted = decrypted_part;
camel_cipher_validity_free(valid);
} else {
camel_object_ref(decrypted_part);
decrypted_part = NULL;
}
return decrypted_part;
}