// Matrix Construct
//
// Copyright (C) Matrix Construct Developers, Authors & Contributors
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.

#pragma once
#define HAVE_IRCD_UTIL_CALLBACKS_H

namespace ircd {
inline namespace util
{
	/// The purpose of callbacks is simply explained with an analogy out of
	/// browser-javascript land: it is window.onload.addEventListener() in lieu
	/// of assigning window.onload = function(). This is a list of multiple
	/// callbacks listening for invocation. The listener is responsible for
	/// both adding and removing itself using the std list interface.
	///
	/// The template provides an option for whether exceptions should propagate
	/// to the caller. If they propagate, all listeners after the exception won't
	/// be invoked.
	///
	template<class prototype,
	         bool exceptions = true>
	struct callbacks;

	template<class prototype>
	struct callbacks<prototype, true>;

	template<class prototype>
	struct callbacks<prototype, false>;
}}

template<class prototype>
struct ircd::util::callbacks<prototype, true>
:std::list<std::function<prototype>>
{
	template<class... args>
	void operator()(args&&... a) const
	{
		for(const auto &func : *this)
			func(std::forward<args>(a)...);
	}

	callbacks() = default;
	callbacks(callbacks &&) = default;
	callbacks(const callbacks &) = delete;
	callbacks &operator=(callbacks &&) = default;
	callbacks &operator=(const callbacks &) = delete;
};

template<class prototype>
struct ircd::util::callbacks<prototype, false>
:std::list<std::function<prototype>>
{
	template<class... args>
	void operator()(args&&... a) const
	{
		for(const auto &func : *this) try
		{
			func(std::forward<args>(a)...);
		}
		catch(const std::exception &e)
		{
			// Logging isn't available in ircd::util;
			// just silently drop and continue.
		}
	}

	callbacks() = default;
	callbacks(callbacks &&) = default;
	callbacks(const callbacks &) = delete;
	callbacks &operator=(callbacks &&) = default;
	callbacks &operator=(const callbacks &) = delete;
};