/*
 * Copyright (c) 2009, Rambler media
 * All rights reserved.
 *
 * 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.
 *
 * THIS SOFTWARE IS PROVIDED BY Rambler media ''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 Rambler 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.
 */

#include "config.h"
#include "main.h"
#include "util.h"
#include "view.h"
#include "expressions.h"
#include "cfg_file.h"
#include "map.h"

struct rspamd_view             *
init_view (memory_pool_t * pool)
{
	struct rspamd_view             *new;

	new = memory_pool_alloc0 (pool, sizeof (struct rspamd_view));

	new->pool = pool;
	new->from_hash = g_hash_table_new (rspamd_strcase_hash, rspamd_strcase_equal);
	new->symbols_hash = g_hash_table_new (rspamd_strcase_hash, rspamd_strcase_equal);
	new->rcpt_hash = g_hash_table_new (rspamd_strcase_hash, rspamd_strcase_equal);
	new->ip_tree = radix_tree_create ();
	new->client_ip_tree = radix_tree_create ();

	memory_pool_add_destructor (new->pool, (pool_destruct_func) g_hash_table_destroy, new->symbols_hash);

	return new;
}

gboolean
add_view_from (struct rspamd_view * view, gchar *line)
{
	struct rspamd_regexp           *re = NULL;

	if (add_map (line, read_host_list, fin_host_list, (void **)&view->from_hash)) {
		return TRUE;
	}
	else if ((re = parse_regexp (view->pool, line, TRUE)) != NULL) {
		view->from_re_list = g_list_prepend (view->from_re_list, re);
		return TRUE;
	}

	return FALSE;
}

gboolean
add_view_rcpt (struct rspamd_view * view, gchar *line)
{
	struct rspamd_regexp           *re = NULL;

	if (add_map (line, read_host_list, fin_host_list, (void **)&view->rcpt_hash)) {
		return TRUE;
	}
	else if ((re = parse_regexp (view->pool, line, TRUE)) != NULL) {
		view->rcpt_re_list = g_list_prepend (view->rcpt_re_list, re);
		return TRUE;
	}

	return FALSE;
}

gboolean
add_view_symbols (struct rspamd_view * view, gchar *line)
{
	struct rspamd_regexp           *re = NULL;
	GList                          *symbols;

	if (add_map (line, read_host_list, fin_host_list, (void **)&view->symbols_hash)) {
		return TRUE;
	}
	else if ((re = parse_regexp (view->pool, line, TRUE)) != NULL) {
		view->symbols_re_list = g_list_prepend (view->symbols_re_list, re);
		return TRUE;
	}
	else {
		/* Try to parse symbols line as comma separated list */
		symbols = parse_comma_list (view->pool, line);
		while (symbols) {
			g_hash_table_insert (view->symbols_hash, (gchar *)symbols->data, symbols->data);
			/* Symbols list would be free at pool destruction */
			symbols = g_list_next (symbols);
		}
	}

	return FALSE;

}

gboolean
add_view_ip (struct rspamd_view * view, gchar *line)
{
	if (add_map (line, read_radix_list, fin_radix_list, (void **)&view->ip_tree)) {
		return TRUE;
	}

	return FALSE;
}

gboolean
add_view_client_ip (struct rspamd_view * view, gchar *line)
{
	if (add_map (line, read_radix_list, fin_radix_list, (void **)&view->client_ip_tree)) {
		return TRUE;
	}

	return FALSE;
}


static struct rspamd_view             *
find_view_by_ip (GList * views, struct worker_task *task)
{
	GList                          *cur;
	struct rspamd_view             *v;

	if (task->from_addr.s_addr == INADDR_NONE) {
		return NULL;
	}

	cur = views;
	while (cur) {
		v = cur->data;
		if (radix32tree_find (v->ip_tree, ntohl (task->from_addr.s_addr)) != RADIX_NO_VALUE) {
			return v;
		}
		cur = g_list_next (cur);
	}

	return NULL;
}

static struct rspamd_view             *
find_view_by_client_ip (GList * views, struct worker_task *task)
{
	GList                          *cur;
	struct rspamd_view             *v;

	if (task->client_addr.s_addr == INADDR_NONE) {
		return NULL;
	}

	cur = views;
	while (cur) {
		v = cur->data;
		if (radix32tree_find (v->client_ip_tree, ntohl (task->client_addr.s_addr)) != RADIX_NO_VALUE) {
			msg_info ("found view for client ip %s", inet_ntoa (task->client_addr));
			return v;
		}
		cur = g_list_next (cur);
	}

	return NULL;
}

static struct rspamd_view             *
find_view_by_from (GList * views, struct worker_task *task)
{
	GList                          *cur, *cur_re;
	struct rspamd_view             *v;
	struct rspamd_regexp           *re;
	gchar                          *from_domain;

	if (task->from == NULL) {
		return NULL;
	}

	cur = views;
	while (cur) {
		v = cur->data;
		/* First try to lookup in hashtable domain name */
		if ((from_domain = strchr (task->from, '@')) != NULL) {
			from_domain ++;
			if (g_hash_table_lookup (v->from_hash, from_domain) != NULL) {
				msg_info ("found view for client from %s", task->from);
				return v;
			}
		}
		if (g_hash_table_lookup (v->from_hash, task->from) != NULL) {
			msg_info ("found view for client from %s", task->from);
			return v;
		}
		/* Then try to match re */
		cur_re = v->from_re_list;

		while (cur_re) {
			re = cur_re->data;
			if (g_regex_match (re->regexp, task->from, 0, NULL) == TRUE) {
				msg_info ("found view for client from %s", task->from);
				return v;
			}
			cur_re = g_list_next (cur_re);
		}
		cur = g_list_next (cur);
	}

	return NULL;
}

static inline gboolean
check_view_rcpt (struct rspamd_view *v, struct worker_task *task)
{
	GList                          *cur, *cur_re;
	gchar                           rcpt_user[256], *p;
	gint                            l;
	struct rspamd_regexp           *re;

	cur = task->rcpt;
	while (cur) {
		if ((p = strchr (cur->data, '@')) != NULL) {
			l = MIN (sizeof (rcpt_user) - 1, p - (gchar *)cur->data);
			memcpy (rcpt_user, cur->data, l);
			rcpt_user[l] = '\0';
			/* First try to lookup in hashtable */
			if (g_hash_table_lookup (v->rcpt_hash, rcpt_user) != NULL) {
				msg_info ("found view for client rcpt %s", rcpt_user);
				return TRUE;
			}
			/* Then try to match re */
			cur_re = v->rcpt_re_list;

			while (cur_re) {
				re = cur_re->data;
				if (g_regex_match (re->regexp, rcpt_user, 0, NULL) == TRUE) {
					msg_info ("found view for client rcpt %s", rcpt_user);
					return TRUE;
				}
				cur_re = g_list_next (cur_re);
			}
		}
		/* Now check the whole recipient */
		if (g_hash_table_lookup (v->rcpt_hash, cur->data) != NULL) {
			msg_info ("found view for client rcpt %s", rcpt_user);
			return TRUE;
		}
		/* Then try to match re */
		cur_re = v->rcpt_re_list;

		while (cur_re) {
			re = cur_re->data;
			if (g_regex_match (re->regexp, cur->data, 0, NULL) == TRUE) {
				msg_info ("found view for client rcpt %s", rcpt_user);
				return TRUE;
			}
			cur_re = g_list_next (cur_re);
		}
		cur = g_list_next (cur);
	}

	return FALSE;
}

static struct rspamd_view             *
find_view_by_rcpt (GList * views, struct worker_task *task)
{
	GList                          *cur;
	struct rspamd_view             *v;

	if (task->from == NULL) {
		return NULL;
	}

	cur = views;
	while (cur) {
		v = cur->data;
		if (check_view_rcpt (v, task)) {
			return v;
		}
		cur = g_list_next (cur);
	}

	return NULL;
}

static                          gboolean
match_view_symbol (struct rspamd_view *v, const gchar *symbol)
{
	GList                          *cur;
	struct rspamd_regexp           *re;

	/* Special case */
	if (symbol == NULL) {
		return TRUE;
	}
	/* First try to lookup in hashtable */
	if (g_hash_table_lookup (v->symbols_hash, symbol) != NULL) {
		return TRUE;
	}
	/* Then try to match re */
	cur = v->symbols_re_list;

	while (cur) {
		re = cur->data;
		if (g_regex_match (re->regexp, symbol, 0, NULL) == TRUE) {
			return TRUE;
		}
		cur = g_list_next (cur);
	}

	return FALSE;
}

gboolean
check_view (GList * views, const gchar *symbol, struct worker_task * task)
{
	struct rspamd_view             *selected = NULL;


	if (views == NULL || (task->view == NULL && task->view_checked == TRUE)) {
		/* If now views defined just return TRUE to check each symbol */
		return TRUE;
	}

	if (task->view != NULL) {
		goto check_symbol;
	}

	if ((selected = find_view_by_ip (views, task)) == NULL) {
		if ((selected = find_view_by_client_ip (views, task)) == NULL) {
			if ((selected = find_view_by_from (views, task)) == NULL) {
				if ((selected = find_view_by_rcpt (views, task)) == NULL) {
					/* No matching view for this task */
					task->view_checked = TRUE;
					return TRUE;
				}
			}
		}
	}

	task->view_checked = TRUE;
	task->view = selected;

  check_symbol:
	/* selected is now not NULL */
	if (task->view->skip_check) {
		return FALSE;
	}
	if (match_view_symbol (task->view, symbol)) {
		return TRUE;
	}

	return FALSE;
}

gboolean
check_skip (GList * views, struct worker_task * task)
{
	if (check_view (views, NULL, task) == FALSE) {
		return TRUE;
	}
	return FALSE;
}