/*
 * Copyright (c) 2013 Red Hat Inc.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 *     * Redistributions of source code must retain the above
 *       copyright notice, this list of conditions and the
 *       following disclaimer.
 *     * Redistributions in binary form must reproduce the
 *       above copyright notice, this list of conditions and
 *       the following disclaimer in the documentation and/or
 *       other materials provided with the distribution.
 *     * The names of contributors to this software may not be
 *       used to endorse or promote products derived from this
 *       software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
 * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
 * DAMAGE.
 *
 * Author: Stef Walter <stefw@redhat.com>
 */

#include "config.h"
#include "test.h"

#include <stdlib.h>
#include <stdio.h>
#include <string.h>

#include "compat.h"
#include "debug.h"
#include "lexer.h"
#include "message.h"

typedef struct {
	int tok_type;
	const char *name;
	const char *value;
} expected_tok;

static void
check_lex_msg (const char *file,
               int line,
               const char *function,
               const expected_tok *expected,
               const char *input,
               bool failure)
{
	p11_lexer lexer;
	size_t len;
	bool failed;
	int i;

	p11_lexer_init (&lexer, "test", input, strlen (input));
	for (i = 0; p11_lexer_next (&lexer, &failed); i++) {
		if (expected[i].tok_type != lexer.tok_type)
			p11_test_fail (file, line, function,
			               "lexer token type does not match: (%d != %d)",
			               expected[i].tok_type, lexer.tok_type);
		switch (lexer.tok_type) {
		case TOK_FIELD:
			if (strcmp (expected[i].name, lexer.tok.field.name) != 0)
				p11_test_fail (file, line, function,
				               "field name doesn't match: (%s != %s)",
				               expected[i].name, lexer.tok.field.name);
			if (strcmp (expected[i].value, lexer.tok.field.value) != 0)
				p11_test_fail (file, line, function,
				               "field value doesn't match: (%s != %s)",
				               expected[i].value, lexer.tok.field.value);
			break;
		case TOK_SECTION:
			if (strcmp (expected[i].name, lexer.tok.field.name) != 0)
				p11_test_fail (file, line, function,
				               "section name doesn't match: (%s != %s)",
				               expected[i].name, lexer.tok.field.name);
			break;
		case TOK_PEM:
			len = strlen (expected[i].name);
			if (lexer.tok.pem.length < len ||
			    strncmp (lexer.tok.pem.begin, expected[i].name, len) != 0) {
				p11_test_fail (file, line, function,
				               "wrong type of PEM block: %s",
				               expected[i].name);
			}
			break;
		case TOK_EOF:
			p11_test_fail (file, line, function, "eof should not be recieved");
			break;
		}
	}

	if (failure && !failed)
		p11_test_fail (file, line, function, "lexing didn't fail");
	else if (!failure && failed)
		p11_test_fail (file, line, function, "lexing failed");
	if (TOK_EOF != expected[i].tok_type)
		p11_test_fail (file, line, function, "premature end of lexing");

	p11_lexer_done (&lexer);
}

#define check_lex_success(expected, input) \
	check_lex_msg (__FILE__, __LINE__, __FUNCTION__, expected, input, false)

#define check_lex_failure(expected, input) \
	check_lex_msg (__FILE__, __LINE__, __FUNCTION__, expected, input, true)

static void
test_basic (void)
{
	const char *input = "[the header]\n"
	                    "field: value\n"
	                    "-----BEGIN BLOCK1-----\n"
	                    "aYNNXqshlVxCdo8QfKeXh3GUzd/yn4LYIVgQrx4a\n"
	                    "-----END BLOCK1-----\n";

	const expected_tok expected[] = {
		{ TOK_SECTION, "the header" },
		{ TOK_FIELD, "field", "value" },
		{ TOK_PEM, "-----BEGIN BLOCK1-----\n", },
		{ TOK_EOF }
	};

	check_lex_success (expected, input);
}

static void
test_corners (void)
{
	const char *input = "\r\n"                 /* blankline */
	                    " [the header]\r\n"    /* bad line endings */
	                    "  field: value  \r\n" /* whitespace */
	                    "number:    2\n"       /* extra space*/
	                    "number    :3\n"       /* extra space*/
	                    "number  :  4\n"       /* extra space*/
	                    "\n"
	                    " # A comment \n"
	                    "not-a-comment: # value\n"
	                    "-----BEGIN BLOCK1-----\r\n"
	                    "aYNNXqshlVxCdo8QfKeXh3GUzd/yn4LYIVgQrx4a\r\n"
	                    "-----END BLOCK1-----"; /* no new line */

	const expected_tok expected[] = {
		{ TOK_SECTION, "the header" },
		{ TOK_FIELD, "field", "value" },
		{ TOK_FIELD, "number", "2" },
		{ TOK_FIELD, "number", "3" },
		{ TOK_FIELD, "number", "4" },
		{ TOK_FIELD, "not-a-comment", "# value" },
		{ TOK_PEM, "-----BEGIN BLOCK1-----\r\n", },
		{ TOK_EOF }
	};

	check_lex_success (expected, input);
}

static void
test_following (void)
{
	const char *input = "-----BEGIN BLOCK1-----\n"
	                    "aYNNXqshlVxCdo8QfKeXh3GUzd/yn4LYIVgQrx4a\n"
	                    "-----END BLOCK1-----\n"
	                    "field: value";

	const expected_tok expected[] = {
		{ TOK_PEM, "-----BEGIN BLOCK1-----\n", },
		{ TOK_FIELD, "field", "value" },
		{ TOK_EOF }
	};

	check_lex_success (expected, input);
}

static void
test_bad_pem (void)
{
	const char *input = "field: value\n"
	                    "-----BEGIN BLOCK1-----\n"
	                    "aYNNXqshlVxCdo8QfKeXh3GUzd/yn4LYIVgQrx4a\n";

	const expected_tok expected[] = {
		{ TOK_FIELD, "field", "value" },
		{ TOK_EOF }
	};

	p11_message_quiet ();

	check_lex_failure (expected, input);

	p11_message_loud ();
}

static void
test_bad_section (void)
{
	const char *input = "field: value\n"
	                    "[section\n"
	                    "bad]\n";

	const expected_tok expected[] = {
		{ TOK_FIELD, "field", "value" },
		{ TOK_EOF }
	};

	p11_message_quiet ();

	check_lex_failure (expected, input);

	p11_message_loud ();
}

static void
test_bad_value (void)
{
	const char *input = "field_value\n"
	                    "[section\n"
	                    "bad]\n";

	const expected_tok expected[] = {
		{ TOK_EOF }
	};

	p11_message_quiet ();

	check_lex_failure (expected, input);

	p11_message_loud ();
}

int
main (int argc,
      char *argv[])
{
	p11_test (test_basic, "/lexer/basic");
	p11_test (test_corners, "/lexer/corners");
	p11_test (test_following, "/lexer/following");
	p11_test (test_bad_pem, "/lexer/bad-pem");
	p11_test (test_bad_section, "/lexer/bad-section");
	p11_test (test_bad_value, "/lexer/bad-value");
	return p11_test_run (argc, argv);
}