summaryrefslogtreecommitdiff
path: root/libmailutils/observer.c
diff options
context:
space:
mode:
Diffstat (limited to 'libmailutils/observer.c')
-rw-r--r--libmailutils/observer.c239
1 files changed, 239 insertions, 0 deletions
diff --git a/libmailutils/observer.c b/libmailutils/observer.c
new file mode 100644
index 000000000..51b02602f
--- /dev/null
+++ b/libmailutils/observer.c
@@ -0,0 +1,239 @@
+/* GNU Mailutils -- a suite of utilities for electronic mail
+ Copyright (C) 1999, 2000, 2004, 2005, 2007, 2010 Free Software
+ Foundation, Inc.
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 3 of the License, or (at your option) any later version.
+
+ This 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
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General
+ Public License along with this library; if not, write to the
+ Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301 USA */
+
+#ifdef HAVE_CONFIG_H
+# include <config.h>
+#endif
+
+#include <errno.h>
+#include <stdlib.h>
+
+#include <mailutils/list.h>
+#include <mailutils/iterator.h>
+#include <mailutils/errno.h>
+#include <mailutils/sys/observer.h>
+
+int
+mu_observer_create (mu_observer_t *pobserver, void *owner)
+{
+ mu_observer_t observer;
+ observer = calloc (sizeof (*observer), 1);
+ if (observer == NULL)
+ return ENOMEM;
+ observer->owner = owner;
+ *pobserver = observer;
+ return 0;
+}
+
+void
+mu_observer_destroy (mu_observer_t *pobserver, void *owner)
+{
+ if (pobserver && *pobserver)
+ {
+ mu_observer_t observer = *pobserver;
+ if (observer->owner == owner || observer->flags & MU_OBSERVER_NO_CHECK)
+ {
+ if (observer->_destroy)
+ observer->_destroy (observer, observer->_action_data);
+ free (observer);
+ }
+ *pobserver = NULL;
+ }
+}
+
+void *
+mu_observer_get_owner (mu_observer_t observer)
+{
+ return (observer) ? observer->owner : NULL;
+}
+
+int
+mu_observer_action (mu_observer_t observer, size_t type, void *data)
+{
+ if (observer == NULL)
+ return EINVAL;
+ if (observer->_action)
+ return observer->_action (observer, type, data, observer->_action_data);
+ return 0;
+}
+
+int
+mu_observer_set_action (mu_observer_t observer,
+ int (*_action) (mu_observer_t, size_t, void *, void *),
+ void *owner)
+{
+ if (observer == NULL)
+ return EINVAL;
+ if (observer->owner != owner)
+ return EACCES;
+ observer->_action = _action;
+ return 0;
+}
+
+int
+mu_observer_set_action_data (mu_observer_t observer, void *data, void *owner)
+{
+ if (observer == NULL)
+ return EINVAL;
+ if (observer->owner != owner)
+ return EACCES;
+ observer->_action_data = data;
+ return 0;
+}
+
+int
+mu_observer_set_destroy (mu_observer_t observer,
+ int (*_destroy) (mu_observer_t, void *),
+ void *owner)
+{
+ if (observer == NULL)
+ return EINVAL;
+ if (observer->owner != owner)
+ return EACCES;
+ observer->_destroy = _destroy;
+ return 0;
+}
+
+int
+mu_observer_set_flags (mu_observer_t observer, int flags)
+{
+ if (observer == NULL)
+ return EINVAL;
+ observer->flags |= flags;
+ return 0;
+}
+
+static void
+_free_event (void *ptr)
+{
+ event_t event = ptr;
+ mu_observer_destroy (&event->observer, NULL);
+ free (event);
+}
+
+int
+mu_observable_create (mu_observable_t *pobservable, void *owner)
+{
+ mu_observable_t observable;
+ int status;
+ if (pobservable == NULL)
+ return MU_ERR_OUT_PTR_NULL;
+ observable = calloc (sizeof (*observable), 1);
+ if (observable == NULL)
+ return ENOMEM;
+ status = mu_list_create (&observable->list);
+ if (status != 0 )
+ {
+ free (observable);
+ return status;
+ }
+ mu_list_set_destroy_item (observable->list, _free_event);
+ observable->owner = owner;
+ *pobservable = observable;
+ return 0;
+}
+
+void
+mu_observable_destroy (mu_observable_t *pobservable, void *owner)
+{
+ if (pobservable && *pobservable)
+ {
+ mu_observable_t observable = *pobservable;
+ if (observable->owner == owner)
+ {
+ mu_list_destroy (&observable->list);
+ free (observable);
+ }
+ *pobservable = NULL;
+ }
+}
+
+void *
+mu_observable_get_owner (mu_observable_t observable)
+{
+ return (observable) ? observable->owner : NULL;
+}
+
+int
+mu_observable_attach (mu_observable_t observable, size_t type, mu_observer_t observer)
+{
+ event_t event;
+ if (observable == NULL || observer == NULL)
+ return EINVAL;
+ event = calloc (1, sizeof (*event));
+ if (event == NULL)
+ return ENOMEM;
+ event->type = type;
+ event->observer = observer;
+ return mu_list_append (observable->list, event);
+}
+
+int
+mu_observable_detach (mu_observable_t observable, mu_observer_t observer)
+{
+ mu_iterator_t iterator;
+ int status;
+ event_t event = NULL;
+
+ if (observable == NULL || observer == NULL)
+ return EINVAL;
+ status = mu_list_get_iterator (observable->list, &iterator);
+ if (status != 0)
+ return status;
+ status = MU_ERR_NOENT;
+ for (mu_iterator_first (iterator); !mu_iterator_is_done (iterator);
+ mu_iterator_next (iterator))
+ {
+ event = NULL;
+ mu_iterator_current (iterator, (void **)&event);
+ if (event && event->observer == observer)
+ {
+ mu_iterator_ctl (iterator, mu_itrctl_delete, NULL);
+ status = 0;
+ break;
+ }
+ }
+ mu_iterator_destroy (&iterator);
+ return status;
+}
+
+int
+mu_observable_notify (mu_observable_t observable, int type, void *data)
+{
+ mu_iterator_t iterator;
+ event_t event = NULL;
+ int status = 0;
+ if (observable == NULL)
+ return EINVAL;
+ status = mu_list_get_iterator (observable->list, &iterator);
+ if (status != 0)
+ return status;
+ for (mu_iterator_first (iterator); !mu_iterator_is_done (iterator);
+ mu_iterator_next (iterator))
+ {
+ event = NULL;
+ mu_iterator_current (iterator, (void **)&event);
+ if (event && event->type & type)
+ {
+ status |= mu_observer_action (event->observer, type, data);
+ }
+ }
+ mu_iterator_destroy (&iterator);
+ return status;
+}

Return to:

Send suggestions and report system problems to the System administrator.