/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* unit-test-pkix-parser.c: Test PKIX parser
Copyright (C) 2007 Stefan Walter
The Gnome Keyring Library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public License as
published by the Free Software Foundation; either version 2 of the
License, or (at your option) any later version.
The Gnome Keyring Library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public
License along with the Gnome Library; see the file COPYING.LIB. If not,
see <http://www.gnu.org/licenses/>.
Author: Stef Walter <stef@memberwebs.com>
*/
#include "config.h"
#include "egg/egg-error.h"
#include "egg/egg-secure-memory.h"
#include "egg/egg-testing.h"
#include "gcr/gcr-base.h"
#include "gcr/gcr-internal.h"
#include "gck/gck.h"
#include <glib.h>
#include <gcrypt.h>
#include <errno.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
/*
* Each test looks like (on one line):
* void unit_test_xxxxx (CuTest* cu)
*
* Each setup looks like (on one line):
* void unit_setup_xxxxx (void);
*
* Each teardown looks like (on one line):
* void unit_teardown_xxxxx (void);
*
* Tests be run in the order specified here.
*/
typedef struct {
GcrParser *parser;
const gchar* filedesc;
} Test;
static void
ensure_block_can_be_parsed (GcrDataFormat format,
GBytes *block)
{
GcrParser *parser;
gboolean result;
GError *error = NULL;
g_assert (block != NULL);
parser = gcr_parser_new ();
g_object_add_weak_pointer (G_OBJECT (parser), (gpointer *)&parser);
gcr_parser_format_disable (parser, -1);
gcr_parser_format_enable (parser, format);
result = gcr_parser_parse_bytes (parser, block, &error);
if (!result) {
g_critical ("The data returned from gcr_parser_get_parsed_block() "
"cannot be parsed: %s", error->message);
g_assert_not_reached ();
}
g_object_unref (parser);
g_assert (parser == NULL);
}
static void
parsed_item (GcrParser *par, gpointer user_data)
{
GckAttributes *attrs;
const gchar *description;
const gchar *label;
Test *test = user_data;
GBytes *block;
GcrDataFormat format;
g_assert (GCR_IS_PARSER (par));
g_assert (par == test->parser);
attrs = gcr_parser_get_parsed_attributes (test->parser);
g_assert (attrs);
description = gcr_parser_get_parsed_description (test->parser);
label = gcr_parser_get_parsed_label (test->parser);
block = gcr_parser_get_parsed_bytes (test->parser);
format = gcr_parser_get_parsed_format (test->parser);
ensure_block_can_be_parsed (format, block);
if (g_test_verbose ())
g_print ("%s: '%s'\n", description, label);
}
static gboolean
authenticate (GcrParser *par, gint state, gpointer user_data)
{
Test *test = user_data;
g_assert (GCR_IS_PARSER (par));
g_assert (par == test->parser);
switch (state) {
case 0:
gcr_parser_add_password (test->parser, "booo");
return TRUE;
case 1:
gcr_parser_add_password (test->parser, "usr0052");
return TRUE;
default:
g_printerr ("decryption didn't work for: %s", test->filedesc);
g_assert_not_reached ();
return FALSE;
};
}
static void
setup (Test *test, gconstpointer unused)
{
test->parser = gcr_parser_new ();
g_signal_connect (test->parser, "parsed", G_CALLBACK (parsed_item), test);
g_signal_connect (test->parser, "authenticate", G_CALLBACK (authenticate), test);
}
static void
teardown (Test *test, gconstpointer unused)
{
g_object_unref (test->parser);
}
static void
test_parse_one (Test *test,
gconstpointer user_data)
{
const gchar *path = user_data;
gchar *contents;
GError *error = NULL;
gboolean result;
GBytes *bytes;
gsize len;
if (!g_file_get_contents (path, &contents, &len, NULL))
g_assert_not_reached ();
test->filedesc = path;
bytes = g_bytes_new_take (contents, len);
result = gcr_parser_parse_bytes (test->parser, bytes, &error);
g_assert_no_error (error);
g_assert (result);
g_bytes_unref (bytes);
}
static void
on_parsed_compare_bytes (GcrParser *parser,
gpointer user_data)
{
GBytes *original = user_data;
GBytes *bytes;
gconstpointer data;
gsize n_data;
GcrParsed *parsed;
bytes = gcr_parser_get_parsed_bytes (parser);
g_assert (bytes != NULL);
g_assert (g_bytes_equal (original, bytes));
data = gcr_parser_get_parsed_block (parser, &n_data);
g_assert (data != NULL);
g_assert_cmpint (n_data, ==, g_bytes_get_size (original));
g_assert (memcmp (data, g_bytes_get_data (original, NULL), n_data) == 0);
parsed = gcr_parser_get_parsed (parser);
g_assert (parsed != NULL);
bytes = gcr_parsed_get_bytes (parsed);
g_assert (bytes != NULL);
g_assert (g_bytes_equal (original, bytes));
data = gcr_parsed_get_data (parsed, &n_data);
g_assert (data != NULL);
g_assert_cmpint (n_data, ==, g_bytes_get_size (original));
g_assert (memcmp (data, g_bytes_get_data (original, NULL), n_data) == 0);
}
static void
test_parsed_bytes (void)
{
GcrParser *parser = gcr_parser_new ();
gchar *contents;
GError *error = NULL;
gboolean result;
GBytes *bytes;
gsize len;
if (!g_file_get_contents (SRCDIR "/gcr/fixtures/cacert.org.cer", &contents, &len, NULL))
g_assert_not_reached ();
bytes = g_bytes_new_take (contents, len);
g_signal_connect (parser, "parsed", G_CALLBACK (on_parsed_compare_bytes), bytes);
result = gcr_parser_parse_bytes (parser, bytes, &error);
g_assert_no_error (error);
g_assert (result);
g_bytes_unref (bytes);
g_object_unref (parser);
}
static void
test_parse_null (void)
{
GcrParser *parser = gcr_parser_new ();
GError *error = NULL;
gboolean result;
result = gcr_parser_parse_data (parser, NULL, 0, &error);
g_assert_error (error, GCR_DATA_ERROR, GCR_ERROR_UNRECOGNIZED);
g_assert (!result);
g_error_free (error);
g_object_unref (parser);
}
static void
test_parse_empty (void)
{
GcrParser *parser = gcr_parser_new ();
GError *error = NULL;
gboolean result;
result = gcr_parser_parse_data (parser, (const guchar *)"", 0, &error);
g_assert_error (error, GCR_DATA_ERROR, GCR_ERROR_UNRECOGNIZED);
g_assert (!result);
g_error_free (error);
g_object_unref (parser);
}
static void
test_parse_stream (void)
{
GcrParser *parser = gcr_parser_new ();
GError *error = NULL;
gboolean result;
GFile *file;
GFileInputStream *fis;
gchar *contents;
gsize len;
GBytes *bytes;
file = g_file_new_for_path (SRCDIR "/gcr/fixtures/cacert.org.cer");
fis = g_file_read (file, NULL, &error);
g_assert_no_error (error);
if (!g_file_get_contents (SRCDIR "/gcr/fixtures/cacert.org.cer", &contents, &len, NULL))
g_assert_not_reached ();
bytes = g_bytes_new_take (contents, len);
g_signal_connect (parser, "parsed", G_CALLBACK (on_parsed_compare_bytes), bytes);
result = gcr_parser_parse_stream (parser, G_INPUT_STREAM (fis), NULL, &error);
g_assert_no_error (error);
g_assert (result);
g_bytes_unref (bytes);
g_object_unref (fis);
g_object_unref (file);
g_object_unref (parser);
}
static void
on_parsed_ref (GcrParser *parser,
gpointer user_data)
{
GcrParsed **parsed = user_data;
g_assert (parsed != NULL);
g_assert (*parsed == NULL);
*parsed = gcr_parsed_ref (gcr_parser_get_parsed (parser));
}
static void
test_parse_filename (void)
{
GcrParser *parser = gcr_parser_new ();
GcrParsed *parsed = NULL;
GError *error = NULL;
gboolean result;
gchar *contents;
gsize len;
GBytes *bytes;
if (!g_file_get_contents (SRCDIR "/gcr/fixtures/cacert.org.cer", &contents, &len, NULL))
g_assert_not_reached ();
bytes = g_bytes_new_take (contents, len);
gcr_parser_set_filename (parser, "cacert.org.cer");
g_signal_connect (parser, "parsed", G_CALLBACK (on_parsed_ref), &parsed);
result = gcr_parser_parse_bytes (parser, bytes, &error);
g_assert_cmpstr (gcr_parser_get_filename (parser), ==, "cacert.org.cer");
g_assert_no_error (error);
g_assert (result);
g_bytes_unref (bytes);
g_object_unref (parser);
g_assert (parsed != NULL);
g_assert_cmpstr (gcr_parsed_get_filename (parsed), ==, "cacert.org.cer");
gcr_parsed_unref (parsed);
}
int
main (int argc, char **argv)
{
const gchar *filename;
GError *error = NULL;
GPtrArray *strings;
GDir *dir;
gchar *path;
gchar *lower;
gchar *test;
int ret;
g_test_init (&argc, &argv, NULL);
g_set_prgname ("test-parser");
strings = g_ptr_array_new_with_free_func (g_free);
dir = g_dir_open (SRCDIR "/gcr/fixtures", 0, &error);
g_assert_no_error (error);
for (;;) {
filename = g_dir_read_name (dir);
if (!filename)
break;
if (filename[0] == '.')
continue;
path = g_build_filename (SRCDIR "/gcr/fixtures", filename, NULL);
if (g_file_test (path, G_FILE_TEST_IS_DIR)) {
g_free (path);
continue;
}
lower = g_ascii_strdown (filename, -1);
test = g_strdup_printf ("/gcr/parser/%s",
g_strcanon (lower, "abcdefghijklmnopqrstuvwxyz0123456789", '_'));
g_free (lower);
g_test_add (test, Test, path, setup, test_parse_one, teardown);
g_ptr_array_add (strings, path);
g_ptr_array_add (strings, test);
}
g_dir_close (dir);
g_test_add_func ("/gcr/parser/parse_null", test_parse_null);
g_test_add_func ("/gcr/parser/parse_empty", test_parse_empty);
g_test_add_func ("/gcr/parser/parse_stream", test_parse_stream);
g_test_add_func ("/gcr/parser/parsed_bytes", test_parsed_bytes);
g_test_add_func ("/gcr/parser/filename", test_parse_filename);
ret = g_test_run ();
g_ptr_array_free (strings, TRUE);
return ret;
}