diff --git a/apps/app_voicemail.c b/apps/app_voicemail.c
index 05b0cfb096a031ef9d1db3ac457145d056296834..538b6cd1bcda542209df42353ff737423298bbef 100644
--- a/apps/app_voicemail.c
+++ b/apps/app_voicemail.c
@@ -47,6 +47,7 @@
  */
 
 /*** MODULEINFO
+	<conflict>res_mwi_external</conflict>
 	<use type="module">res_adsi</use>
 	<use type="module">res_smdi</use>
 	<support_level>core</support_level>
diff --git a/configs/sorcery.conf.sample b/configs/sorcery.conf.sample
index ab802d450a3a45ff3c2c143a63050abfa79771c7..ee130036039a6e68572f984ef3d9ee8fcee1abd4 100644
--- a/configs/sorcery.conf.sample
+++ b/configs/sorcery.conf.sample
@@ -48,6 +48,13 @@ test=memory
 test/cache=test
 test=memory
 
+;
+; The following object mapping is the default mapping of external MWI mailbox
+; objects to give persistence to the message counts.
+;
+;[res_mwi_external]
+;mailboxes=astdb,mwi_external
+
 ;
 ; The following object mappings set PJSIP objects to use realtime database mappings from extconfig
 ; with the table names used when automatically generating configuration from the alembic script.
diff --git a/include/asterisk/res_mwi_external.h b/include/asterisk/res_mwi_external.h
new file mode 100644
index 0000000000000000000000000000000000000000..7698a1bd69d91caa3fa896758787651d039f392c
--- /dev/null
+++ b/include/asterisk/res_mwi_external.h
@@ -0,0 +1,226 @@
+/*
+ * Asterisk -- An open source telephony toolkit.
+ *
+ * Copyright (C) 2013, Digium, Inc.
+ *
+ * Richard Mudgett <rmudgett@digium.com>
+ *
+ * See http://www.asterisk.org for more information about
+ * the Asterisk project. Please do not directly contact
+ * any of the maintainers of this project for assistance;
+ * the project provides a web site, mailing lists and IRC
+ * channels for your use.
+ *
+ * This program is free software, distributed under the terms of
+ * the GNU General Public License Version 2. See the LICENSE file
+ * at the top of the source tree.
+ */
+
+/*!
+ * \file
+ * \brief Core external MWI support.
+ *
+ * \author Richard Mudgett <rmudgett@digium.com>
+ *
+ * See Also:
+ * \arg \ref AstCREDITS
+ */
+
+#ifndef _ASTERISK_RES_MWI_EXTERNAL_H
+#define _ASTERISK_RES_MWI_EXTERNAL_H
+
+#if defined(__cplusplus) || defined(c_plusplus)
+extern "C"
+{
+#endif
+
+/* ------------------------------------------------------------------- */
+
+/*!
+ * \brief Increase the external MWI resource module reference count.
+ * \since 12.1.0
+ *
+ * \return Nothing
+ */
+void ast_mwi_external_ref(void);
+
+/*!
+ * \brief Decrease the external MWI resource module reference count.
+ * \since 12.1.0
+ *
+ * \return Nothing
+ */
+void ast_mwi_external_unref(void);
+
+struct ast_mwi_mailbox_object;
+
+/*! \brief Convienience unref function for mailbox object. */
+#define ast_mwi_mailbox_unref(mailbox) ao2_ref((struct ast_mwi_mailbox_object *) mailbox, -1)
+
+/*!
+ * \brief Allocate an external MWI object.
+ * \since 12.1.0
+ *
+ * \param mailbox_id Name of mailbox.
+ *
+ * \retval object on success.  The object is an ao2 object.
+ * \retval NULL on error.
+ */
+struct ast_mwi_mailbox_object *ast_mwi_mailbox_alloc(const char *mailbox_id);
+
+/*!
+ * \brief Get mailbox id.
+ * \since 12.1.0
+ *
+ * \param mailbox Object to get id.
+ *
+ * \return mailbox_id of the object.
+ *
+ * \note This should never return NULL unless there is a bug in sorcery.
+ */
+const char *ast_mwi_mailbox_get_id(const struct ast_mwi_mailbox_object *mailbox);
+
+/*!
+ * \brief Get the number of new messages.
+ * \since 12.1.0
+ *
+ * \param mailbox Object to get number of new messages.
+ *
+ * \return Number of new messages.
+ */
+unsigned int ast_mwi_mailbox_get_msgs_new(const struct ast_mwi_mailbox_object *mailbox);
+
+/*!
+ * \brief Get the number of old messages.
+ * \since 12.1.0
+ *
+ * \param mailbox Object to get number of old messages.
+ *
+ * \return Number of old messages.
+ */
+unsigned int ast_mwi_mailbox_get_msgs_old(const struct ast_mwi_mailbox_object *mailbox);
+
+/*!
+ * \brief Copy the external MWI counts object.
+ * \since 12.1.0
+ *
+ * \param mailbox What to copy.
+ *
+ * \retval copy on success.  The object is an ao2 object.
+ * \retval NULL on error.
+ */
+struct ast_mwi_mailbox_object *ast_mwi_mailbox_copy(const struct ast_mwi_mailbox_object *mailbox);
+
+/*!
+ * \brief Set the number of new messages.
+ * \since 12.1.0
+ *
+ * \param mailbox Object to set number of new messages.
+ * \param num_msgs Number of messages to set.
+ *
+ * \return Nothing
+ */
+void ast_mwi_mailbox_set_msgs_new(struct ast_mwi_mailbox_object *mailbox, unsigned int num_msgs);
+
+/*!
+ * \brief Set the number of old messages.
+ * \since 12.1.0
+ *
+ * \param mailbox Object to set number of old messages.
+ * \param num_msgs Number of messages to set.
+ *
+ * \return Nothing
+ */
+void ast_mwi_mailbox_set_msgs_old(struct ast_mwi_mailbox_object *mailbox, unsigned int num_msgs);
+
+/*!
+ * \brief Update the external MWI counts with the given object.
+ * \since 12.1.0
+ *
+ * \param mailbox What to update.
+ *
+ * \retval 0 on success.
+ * \retval -1 on error.
+ */
+int ast_mwi_mailbox_update(struct ast_mwi_mailbox_object *mailbox);
+
+/*!
+ * \brief Delete matching external MWI object.
+ * \since 12.1.0
+ *
+ * \param mailbox_id Name of mailbox to delete.
+ *
+ * \retval 0 on success.
+ * \retval -1 on error.
+ */
+int ast_mwi_mailbox_delete(const char *mailbox_id);
+
+/*!
+ * \brief Delete all external MWI objects selected by the regular expression.
+ * \since 12.1.0
+ *
+ * \param regex Regular expression in extended syntax.  (NULL is same as "")
+ *
+ * \note The provided regex is treated as extended case sensitive.
+ *
+ * \retval 0 on success.
+ * \retval -1 on error.
+ */
+int ast_mwi_mailbox_delete_by_regex(const char *regex);
+
+/*!
+ * \brief Delete all external MWI objects.
+ * \since 12.1.0
+ *
+ * \retval 0 on success.
+ * \retval -1 on error.
+ */
+int ast_mwi_mailbox_delete_all(void);
+
+/*!
+ * \brief Get matching external MWI object.
+ * \since 12.1.0
+ *
+ * \param mailbox_id Name of mailbox to retrieve.
+ *
+ * \retval requested mailbox on success.  The object is an ao2 object.
+ * \retval NULL on error or no mailbox.
+ *
+ * \note The object must be treated as read-only.
+ */
+const struct ast_mwi_mailbox_object *ast_mwi_mailbox_get(const char *mailbox_id);
+
+/*!
+ * \brief Get all external MWI objects selected by the regular expression.
+ * \since 12.1.0
+ *
+ * \param regex Regular expression in extended syntax.  (NULL is same as "")
+ *
+ * \note The provided regex is treated as extended case sensitive.
+ *
+ * \retval container of struct ast_mwi_mailbox_object on success.
+ * \retval NULL on error.
+ *
+ * \note The objects in the container must be treated as read-only.
+ */
+struct ao2_container *ast_mwi_mailbox_get_by_regex(const char *regex);
+
+/*!
+ * \brief Get all external MWI objects.
+ * \since 12.1.0
+ *
+ * \retval container of struct ast_mwi_mailbox_object on success.
+ * \retval NULL on error.
+ *
+ * \note The objects in the container must be treated as read-only.
+ */
+struct ao2_container *ast_mwi_mailbox_get_all(void);
+
+
+/* ------------------------------------------------------------------- */
+
+#if defined(__cplusplus) || defined(c_plusplus)
+}
+#endif
+
+#endif	/* _ASTERISK_RES_MWI_EXTERNAL_H */
diff --git a/res/res_mwi_external.c b/res/res_mwi_external.c
new file mode 100644
index 0000000000000000000000000000000000000000..a7531c5cbf652b4bea485d06488f5918817ea987
--- /dev/null
+++ b/res/res_mwi_external.c
@@ -0,0 +1,960 @@
+/*
+ * Asterisk -- An open source telephony toolkit.
+ *
+ * Copyright (C) 2013, Digium, Inc.
+ *
+ * Richard Mudgett <rmudgett@digium.com>
+ *
+ * See http://www.asterisk.org for more information about
+ * the Asterisk project. Please do not directly contact
+ * any of the maintainers of this project for assistance;
+ * the project provides a web site, mailing lists and IRC
+ * channels for your use.
+ *
+ * This program is free software, distributed under the terms of
+ * the GNU General Public License Version 2. See the LICENSE file
+ * at the top of the source tree.
+ */
+
+/*!
+ * \file
+ * \brief Core external MWI support.
+ *
+ * \details
+ * The module manages the persistent message counts cache and supplies
+ * an API to allow the protocol specific modules to control the counts
+ * or a subset.
+ *
+ * \author Richard Mudgett <rmudgett@digium.com>
+ *
+ * See Also:
+ * \arg \ref AstCREDITS
+ */
+
+/*** MODULEINFO
+	<defaultenabled>no</defaultenabled>
+	<conflict>app_voicemail</conflict>
+	<support_level>core</support_level>
+ ***/
+
+/*** DOCUMENTATION
+	<configInfo name="res_mwi_external" language="en_US">
+		<synopsis>Core external MWI support</synopsis>
+		<configFile name="sorcery.conf">
+			<configObject name="mailboxes">
+				<synopsis>Persistent cache of external MWI Mailboxs.</synopsis>
+				<description>
+					<para>Allows the alteration of sorcery backend mapping for
+					the persistent cache of external MWI mailboxes.</para>
+				</description>
+			</configObject>
+		</configFile>
+	</configInfo>
+ ***/
+
+
+#include "asterisk.h"
+
+ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
+
+#include "asterisk/app.h"
+#include "asterisk/module.h"
+#include "asterisk/res_mwi_external.h"
+#include "asterisk/sorcery.h"
+#include "asterisk/cli.h"
+
+/* ------------------------------------------------------------------- */
+
+/*!
+ * Define to include CLI commands to manipulate the external MWI mailboxes.
+ * Useful for testing the module functionality.
+ */
+//#define MWI_DEBUG_CLI		1
+
+#define MWI_ASTDB_PREFIX	"mwi_external"
+#define MWI_MAILBOX_TYPE	"mailboxes"
+
+struct ast_mwi_mailbox_object {
+	SORCERY_OBJECT(details);
+	/*! Number of new messages in mailbox. */
+	unsigned int msgs_new;
+	/*! Number of old messages in mailbox. */
+	unsigned int msgs_old;
+};
+
+static struct ast_sorcery *mwi_sorcery;
+
+void ast_mwi_external_ref(void)
+{
+	ast_module_ref(ast_module_info->self);
+}
+
+void ast_mwi_external_unref(void)
+{
+	ast_module_unref(ast_module_info->self);
+}
+
+/*!
+ * \internal
+ * \brief Post an update event to the MWI counts.
+ * \since 12.1.0
+ *
+ * \return Nothing
+ */
+static void mwi_post_event(const struct ast_mwi_mailbox_object *mailbox)
+{
+	ast_publish_mwi_state(ast_sorcery_object_get_id(mailbox), NULL,
+		mailbox->msgs_new, mailbox->msgs_old);
+}
+
+static void mwi_observe_update(const void *obj)
+{
+	mwi_post_event(obj);
+}
+
+/*!
+ * \internal
+ * \brief Post a count clearing event to the MWI counts.
+ * \since 12.1.0
+ *
+ * \return Nothing
+ */
+static void mwi_observe_delete(const void *obj)
+{
+	const struct ast_mwi_mailbox_object *mailbox = obj;
+
+	if (!mailbox->msgs_new && !mailbox->msgs_old) {
+		/* No need to post a count clearing event. */
+		return;
+	}
+
+	/* Post a count clearing event. */
+	ast_publish_mwi_state(ast_sorcery_object_get_id(mailbox), NULL, 0, 0);
+}
+
+static const struct ast_sorcery_observer mwi_observers = {
+	.created = mwi_observe_update,
+	.updated = mwi_observe_update,
+	.deleted = mwi_observe_delete,
+};
+
+/*! \brief Internal function to allocate a mwi object */
+static void *mwi_sorcery_object_alloc(const char *id)
+{
+	return ast_sorcery_generic_alloc(sizeof(struct ast_mwi_mailbox_object), NULL);
+}
+
+/*!
+ * \internal
+ * \brief Initialize sorcery for external MWI.
+ * \since 12.1.0
+ *
+ * \retval 0 on success.
+ * \retval -1 on error.
+ */
+static int mwi_sorcery_init(void)
+{
+	int res;
+
+	mwi_sorcery = ast_sorcery_open();
+	if (!mwi_sorcery) {
+		ast_log(LOG_ERROR, "MWI external: Sorcery failed to open.\n");
+		return -1;
+	}
+
+	/* Map the external MWI wizards. */
+	res = !!ast_sorcery_apply_config(mwi_sorcery, "res_mwi_external");
+	res &= !!ast_sorcery_apply_default(mwi_sorcery, MWI_MAILBOX_TYPE, "astdb",
+		MWI_ASTDB_PREFIX);
+	if (res) {
+		ast_log(LOG_ERROR, "MWI external: Sorcery could not setup wizards.\n");
+		return -1;
+	}
+
+	res = ast_sorcery_object_register(mwi_sorcery, MWI_MAILBOX_TYPE,
+		mwi_sorcery_object_alloc, NULL, NULL);
+	if (res) {
+		ast_log(LOG_ERROR, "MWI external: Sorcery could not register object type '%s'.\n",
+			MWI_MAILBOX_TYPE);
+		return -1;
+	}
+
+	/* Define the MWI_MAILBOX_TYPE object fields. */
+	res |= ast_sorcery_object_field_register_nodoc(mwi_sorcery, MWI_MAILBOX_TYPE,
+		"msgs_new", "0", OPT_UINT_T, 0, FLDSET(struct ast_mwi_mailbox_object, msgs_new));
+	res |= ast_sorcery_object_field_register_nodoc(mwi_sorcery, MWI_MAILBOX_TYPE,
+		"msgs_old", "0", OPT_UINT_T, 0, FLDSET(struct ast_mwi_mailbox_object, msgs_old));
+	return res ? -1 : 0;
+}
+
+struct ao2_container *ast_mwi_mailbox_get_all(void)
+{
+	return ast_sorcery_retrieve_by_fields(mwi_sorcery, MWI_MAILBOX_TYPE,
+		AST_RETRIEVE_FLAG_MULTIPLE | AST_RETRIEVE_FLAG_ALL, NULL);
+}
+
+struct ao2_container *ast_mwi_mailbox_get_by_regex(const char *regex)
+{
+	return ast_sorcery_retrieve_by_regex(mwi_sorcery, MWI_MAILBOX_TYPE, regex ?: "");
+}
+
+const struct ast_mwi_mailbox_object *ast_mwi_mailbox_get(const char *mailbox_id)
+{
+	if (ast_strlen_zero(mailbox_id)) {
+		return NULL;
+	}
+
+	return ast_sorcery_retrieve_by_id(mwi_sorcery, MWI_MAILBOX_TYPE, mailbox_id);
+}
+
+struct ast_mwi_mailbox_object *ast_mwi_mailbox_alloc(const char *mailbox_id)
+{
+	if (ast_strlen_zero(mailbox_id)) {
+		return NULL;
+	}
+
+	return ast_sorcery_alloc(mwi_sorcery, MWI_MAILBOX_TYPE, mailbox_id);
+}
+
+struct ast_mwi_mailbox_object *ast_mwi_mailbox_copy(const struct ast_mwi_mailbox_object *mailbox)
+{
+	return ast_sorcery_copy(mwi_sorcery, mailbox);
+}
+
+const char *ast_mwi_mailbox_get_id(const struct ast_mwi_mailbox_object *mailbox)
+{
+	return ast_sorcery_object_get_id(mailbox);
+}
+
+unsigned int ast_mwi_mailbox_get_msgs_new(const struct ast_mwi_mailbox_object *mailbox)
+{
+	return mailbox->msgs_new;
+}
+
+unsigned int ast_mwi_mailbox_get_msgs_old(const struct ast_mwi_mailbox_object *mailbox)
+{
+	return mailbox->msgs_old;
+}
+
+void ast_mwi_mailbox_set_msgs_new(struct ast_mwi_mailbox_object *mailbox, unsigned int num_msgs)
+{
+	mailbox->msgs_new = num_msgs;
+}
+
+void ast_mwi_mailbox_set_msgs_old(struct ast_mwi_mailbox_object *mailbox, unsigned int num_msgs)
+{
+	mailbox->msgs_old = num_msgs;
+}
+
+int ast_mwi_mailbox_update(struct ast_mwi_mailbox_object *mailbox)
+{
+	const struct ast_mwi_mailbox_object *exists;
+	int res;
+
+	exists = ast_sorcery_retrieve_by_id(mwi_sorcery, MWI_MAILBOX_TYPE,
+		ast_sorcery_object_get_id(mailbox));
+	if (exists) {
+		res = ast_sorcery_update(mwi_sorcery, mailbox);
+		ast_mwi_mailbox_unref(exists);
+	} else {
+		res = ast_sorcery_create(mwi_sorcery, mailbox);
+	}
+	return res;
+}
+
+/*!
+ * \internal
+ * \brief Delete a mailbox.
+ * \since 12.1.0
+ *
+ * \param mailbox Mailbox object to delete from sorcery.
+ *
+ * \return Nothing
+ */
+static void mwi_mailbox_delete(struct ast_mwi_mailbox_object *mailbox)
+{
+	ast_sorcery_delete(mwi_sorcery, mailbox);
+}
+
+/*!
+ * \internal
+ * \brief Delete all mailboxes in container.
+ * \since 12.1.0
+ *
+ * \param mailboxes Mailbox objects to delete from sorcery.
+ *
+ * \return Nothing
+ */
+static void mwi_mailbox_delete_all(struct ao2_container *mailboxes)
+{
+	struct ast_mwi_mailbox_object *mailbox;
+	struct ao2_iterator iter;
+
+	iter = ao2_iterator_init(mailboxes, AO2_ITERATOR_UNLINK);
+	for (; (mailbox = ao2_iterator_next(&iter)); ast_mwi_mailbox_unref(mailbox)) {
+		mwi_mailbox_delete(mailbox);
+	}
+	ao2_iterator_destroy(&iter);
+}
+
+int ast_mwi_mailbox_delete_all(void)
+{
+	struct ao2_container *mailboxes;
+
+	mailboxes = ast_mwi_mailbox_get_all();
+	if (mailboxes) {
+		mwi_mailbox_delete_all(mailboxes);
+		ao2_ref(mailboxes, -1);
+	}
+	return 0;
+}
+
+int ast_mwi_mailbox_delete_by_regex(const char *regex)
+{
+	struct ao2_container *mailboxes;
+
+	mailboxes = ast_mwi_mailbox_get_by_regex(regex);
+	if (mailboxes) {
+		mwi_mailbox_delete_all(mailboxes);
+		ao2_ref(mailboxes, -1);
+	}
+	return 0;
+}
+
+int ast_mwi_mailbox_delete(const char *mailbox_id)
+{
+	const struct ast_mwi_mailbox_object *mailbox;
+
+	if (ast_strlen_zero(mailbox_id)) {
+		return -1;
+	}
+
+	mailbox = ast_mwi_mailbox_get(mailbox_id);
+	if (mailbox) {
+		mwi_mailbox_delete((struct ast_mwi_mailbox_object *) mailbox);
+		ast_mwi_mailbox_unref(mailbox);
+	}
+	return 0;
+}
+
+enum folder_map {
+	FOLDER_INVALID = 0,
+	FOLDER_INBOX = 1,
+	FOLDER_OLD = 2,
+};
+
+/*!
+ * \internal
+ * \brief Determine if the requested folder is valid for external MWI support.
+ * \since 12.1.0
+ *
+ * \param folder Folder name to check (NULL is valid).
+ *
+ * \return Enum of the supported folder.
+ */
+static enum folder_map mwi_folder_map(const char *folder)
+{
+	enum folder_map which_folder;
+
+	if (ast_strlen_zero(folder) || !strcasecmp(folder, "INBOX")) {
+		which_folder = FOLDER_INBOX;
+	} else if (!strcasecmp(folder, "Old")) {
+		which_folder = FOLDER_OLD;
+	} else {
+		which_folder = FOLDER_INVALID;
+	}
+	return which_folder;
+}
+
+/*!
+ * \internal
+ * \brief Gets the number of messages that exist in a mailbox folder.
+ * \since 12.1.0
+ *
+ * \param mailbox_id The mailbox name.
+ * \param folder The folder to look in.  Default is INBOX if not provided.
+ *
+ * \return The number of messages in the mailbox folder (zero or more).
+ */
+static int mwi_messagecount(const char *mailbox_id, const char *folder)
+{
+	const struct ast_mwi_mailbox_object *mailbox;
+	int num_msgs;
+	enum folder_map which_folder;
+
+	which_folder = mwi_folder_map(folder);
+	if (which_folder == FOLDER_INVALID) {
+		return 0;
+	}
+
+	mailbox = ast_mwi_mailbox_get(mailbox_id);
+	if (!mailbox) {
+		return 0;
+	}
+	num_msgs = 0;
+	switch (which_folder) {
+	case FOLDER_INVALID:
+		break;
+	case FOLDER_INBOX:
+		num_msgs = mailbox->msgs_new;
+		break;
+	case FOLDER_OLD:
+		num_msgs = mailbox->msgs_old;
+		break;
+	}
+	ast_mwi_mailbox_unref(mailbox);
+
+	return num_msgs;
+}
+
+/*!
+ * \internal
+ * \brief Determines if the given folder has messages.
+ * \since 12.1.0
+ *
+ * \param mailboxes Comma or & delimited list of mailboxes.
+ * \param folder The folder to look in.  Default is INBOX if not provided.
+ *
+ * \retval 1 if the folder has one or more messages.
+ * \retval 0 otherwise.
+ */
+static int mwi_has_voicemail(const char *mailboxes, const char *folder)
+{
+	char *parse;
+	char *mailbox_id;
+	enum folder_map which_folder;
+
+	which_folder = mwi_folder_map(folder);
+	if (which_folder == FOLDER_INVALID) {
+		return 0;
+	}
+
+	/* For each mailbox in the list. */
+	parse = ast_strdupa(mailboxes);
+	while ((mailbox_id = strsep(&parse, ",&"))) {
+		const struct ast_mwi_mailbox_object *mailbox;
+		int num_msgs;
+
+		/* Get the specified mailbox. */
+		mailbox = ast_mwi_mailbox_get(mailbox_id);
+		if (!mailbox) {
+			continue;
+		}
+
+		/* Done if the found mailbox has any messages. */
+		num_msgs = 0;
+		switch (which_folder) {
+		case FOLDER_INVALID:
+			break;
+		case FOLDER_INBOX:
+			num_msgs = mailbox->msgs_new;
+			break;
+		case FOLDER_OLD:
+			num_msgs = mailbox->msgs_old;
+			break;
+		}
+		ast_mwi_mailbox_unref(mailbox);
+		if (num_msgs) {
+			return 1;
+		}
+	}
+
+	return 0;
+}
+
+/*!
+ * \internal
+ * \brief Gets the number of messages that exist for the mailbox list.
+ * \since 12.1.0
+ *
+ * \param mailboxes Comma or space delimited list of mailboxes.
+ * \param newmsgs Where to put the count of new messages. (Can be NULL)
+ * \param oldmsgs Where to put the count of old messages. (Can be NULL)
+ *
+ * \details
+ * Simultaneously determines the count of new and old
+ * messages.  The total messages would then be the sum of these.
+ *
+ * \retval 0 on success
+ * \retval -1 on failure
+ */
+static int mwi_inboxcount(const char *mailboxes, int *newmsgs, int *oldmsgs)
+{
+	char *parse;
+	char *mailbox_id;
+
+	if (!newmsgs && !oldmsgs) {
+		/* Nowhere to accumulate counts */
+		return 0;
+	}
+
+	/* For each mailbox in the list. */
+	parse = ast_strdupa(mailboxes);
+	while ((mailbox_id = strsep(&parse, ", "))) {
+		const struct ast_mwi_mailbox_object *mailbox;
+
+		/* Get the specified mailbox. */
+		mailbox = ast_mwi_mailbox_get(mailbox_id);
+		if (!mailbox) {
+			continue;
+		}
+
+		/* Accumulate the counts. */
+		if (newmsgs) {
+			*newmsgs += mailbox->msgs_new;
+		}
+		if (oldmsgs) {
+			*oldmsgs += mailbox->msgs_old;
+		}
+
+		ast_mwi_mailbox_unref(mailbox);
+	}
+
+	return 0;
+}
+
+/*!
+ * \internal
+ * \brief Gets the number of messages that exist for the mailbox list.
+ * \since 12.1.0
+ *
+ * \param mailboxes Comma or space delimited list of mailboxes.
+ * \param urgentmsgs Where to put the count of urgent messages. (Can be NULL)
+ * \param newmsgs Where to put the count of new messages. (Can be NULL)
+ * \param oldmsgs Where to put the count of old messages. (Can be NULL)
+ *
+ * \details
+ * Simultaneously determines the count of new, old, and urgent
+ * messages.  The total messages would then be the sum of these
+ * three.
+ *
+ * \retval 0 on success
+ * \retval -1 on failure
+ */
+static int mwi_inboxcount2(const char *mailboxes, int *urgentmsgs, int *newmsgs, int *oldmsgs)
+{
+	/*
+	 * This module does not support urgentmsgs.  Just ignore them.
+	 * The global API call has already set the count to zero.
+	 */
+	return mwi_inboxcount(mailboxes, newmsgs, oldmsgs);
+}
+
+static const struct ast_vm_functions vm_table = {
+	.module_version = VM_MODULE_VERSION,
+	.module_name = AST_MODULE,
+
+	.has_voicemail = mwi_has_voicemail,
+	.inboxcount = mwi_inboxcount,
+	.inboxcount2 = mwi_inboxcount2,
+	.messagecount = mwi_messagecount,
+};
+
+#if defined(MWI_DEBUG_CLI)
+static char *complete_mailbox(const char *word, int state)
+{
+	struct ao2_iterator iter;
+	int wordlen = strlen(word);
+	int which = 0;
+	char *ret = NULL;
+	char *regex;
+	const struct ast_mwi_mailbox_object *mailbox;
+	RAII_VAR(struct ao2_container *, mailboxes, NULL, ao2_cleanup);
+
+	regex = ast_alloca(2 + wordlen);
+	sprintf(regex, "^%s", word);/* Safe */
+
+	mailboxes = ast_mwi_mailbox_get_by_regex(regex);
+	if (!mailboxes) {
+		return NULL;
+	}
+
+	iter = ao2_iterator_init(mailboxes, 0);
+	for (; (mailbox = ao2_iterator_next(&iter)); ast_mwi_mailbox_unref(mailbox)) {
+		if (++which > state) {
+			ret = ast_strdup(ast_sorcery_object_get_id(mailbox));
+			ast_mwi_mailbox_unref(mailbox);
+			break;
+		}
+	}
+	ao2_iterator_destroy(&iter);
+
+	return ret;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+static char *handle_mwi_delete_all(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+	switch (cmd) {
+	case CLI_INIT:
+		e->command = "mwi delete all";
+		e->usage =
+			"Usage: mwi delete all\n"
+			"       Delete all external MWI mailboxes.\n";
+		return NULL;
+	case CLI_GENERATE:
+		return NULL;
+	}
+
+	ast_mwi_mailbox_delete_all();
+	ast_cli(a->fd, "Deleted all external MWI mailboxes.\n");
+	return CLI_SUCCESS;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+static char *handle_mwi_delete_like(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+	const char *regex;
+
+	switch (cmd) {
+	case CLI_INIT:
+		e->command = "mwi delete like";
+		e->usage =
+			"Usage: mwi delete like <pattern>\n"
+			"       Delete external MWI mailboxes matching a regular expression.\n";
+		return NULL;
+	case CLI_GENERATE:
+		return NULL;
+	}
+
+	if (a->argc != 4) {
+		return CLI_SHOWUSAGE;
+	}
+	regex = a->argv[3];
+
+	ast_mwi_mailbox_delete_by_regex(regex);
+	ast_cli(a->fd, "Deleted external MWI mailboxes matching '%s'.\n", regex);
+	return CLI_SUCCESS;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+static char *handle_mwi_delete_mailbox(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+	const char *mailbox_id;
+
+	switch (cmd) {
+	case CLI_INIT:
+		e->command = "mwi delete mailbox";
+		e->usage =
+			"Usage: mwi delete mailbox <mailbox_id>\n"
+			"       Delete a specific external MWI mailbox.\n";
+		return NULL;
+	case CLI_GENERATE:
+		if (a->pos == 3) {
+			return complete_mailbox(a->word, a->n);
+		}
+		return NULL;
+	}
+
+	if (a->argc != 4) {
+		return CLI_SHOWUSAGE;
+	}
+	mailbox_id = a->argv[3];
+
+	ast_mwi_mailbox_delete(mailbox_id);
+	ast_cli(a->fd, "Deleted external MWI mailbox '%s'.\n", mailbox_id);
+
+	return CLI_SUCCESS;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#define FORMAT_MAILBOX_HDR "%6s %6s %s\n"
+#define FORMAT_MAILBOX_ROW "%6u %6u %s\n"
+
+#if defined(MWI_DEBUG_CLI)
+/*!
+ * \internal
+ * \brief Print a mailbox list line to CLI.
+ * \since 12.1.0
+ *
+ * \param cli_fd File descriptor for CLI output.
+ * \param mailbox What to list.
+ *
+ * \return Nothing
+ */
+static void mwi_cli_print_mailbox(int cli_fd, const struct ast_mwi_mailbox_object *mailbox)
+{
+	ast_cli(cli_fd, FORMAT_MAILBOX_ROW, mailbox->msgs_new, mailbox->msgs_old,
+		ast_sorcery_object_get_id(mailbox));
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+/*!
+ * \internal
+ * \brief List all mailboxes in the given container.
+ * \since 12.1.0
+ *
+ * \param cli_fd File descriptor for CLI output.
+ * \param mailboxes What to list.
+ *
+ * \return Nothing
+ */
+static void mwi_cli_list_mailboxes(int cli_fd, struct ao2_container *mailboxes)
+{
+	struct ao2_iterator iter;
+	const struct ast_mwi_mailbox_object *mailbox;
+
+	ast_cli(cli_fd, FORMAT_MAILBOX_HDR, "New", "Old", "Mailbox");
+
+	iter = ao2_iterator_init(mailboxes, 0);
+	for (; (mailbox = ao2_iterator_next(&iter)); ast_mwi_mailbox_unref(mailbox)) {
+		mwi_cli_print_mailbox(cli_fd, mailbox);
+	}
+	ao2_iterator_destroy(&iter);
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#undef FORMAT_MAILBOX_HDR
+#undef FORMAT_MAILBOX_ROW
+
+#if defined(MWI_DEBUG_CLI)
+static char *handle_mwi_list_all(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+	struct ao2_container *mailboxes;
+
+	switch (cmd) {
+	case CLI_INIT:
+		e->command = "mwi list all";
+		e->usage =
+			"Usage: mwi list all\n"
+			"       List all external MWI mailboxes.\n";
+		return NULL;
+	case CLI_GENERATE:
+		return NULL;
+	}
+
+	mailboxes = ast_mwi_mailbox_get_all();
+	if (!mailboxes) {
+		ast_cli(a->fd, "Failed to retrieve external MWI mailboxes.\n");
+		return CLI_SUCCESS;
+	}
+	mwi_cli_list_mailboxes(a->fd, mailboxes);
+	ao2_ref(mailboxes, -1);
+	return CLI_SUCCESS;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+static char *handle_mwi_list_like(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+	struct ao2_container *mailboxes;
+	const char *regex;
+
+	switch (cmd) {
+	case CLI_INIT:
+		e->command = "mwi list like";
+		e->usage =
+			"Usage: mwi list like <pattern>\n"
+			"       List external MWI mailboxes matching a regular expression.\n";
+		return NULL;
+	case CLI_GENERATE:
+		return NULL;
+	}
+
+	if (a->argc != 4) {
+		return CLI_SHOWUSAGE;
+	}
+	regex = a->argv[3];
+
+	mailboxes = ast_mwi_mailbox_get_by_regex(regex);
+	if (!mailboxes) {
+		ast_cli(a->fd, "Failed to retrieve external MWI mailboxes.\n");
+		return CLI_SUCCESS;
+	}
+	mwi_cli_list_mailboxes(a->fd, mailboxes);
+	ao2_ref(mailboxes, -1);
+	return CLI_SUCCESS;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+static char *handle_mwi_show_mailbox(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+	const struct ast_mwi_mailbox_object *mailbox;
+	const char *mailbox_id;
+
+	switch (cmd) {
+	case CLI_INIT:
+		e->command = "mwi show mailbox";
+		e->usage =
+			"Usage: mwi show mailbox <mailbox_id>\n"
+			"       Show a specific external MWI mailbox.\n";
+		return NULL;
+	case CLI_GENERATE:
+		if (a->pos == 3) {
+			return complete_mailbox(a->word, a->n);
+		}
+		return NULL;
+	}
+
+	if (a->argc != 4) {
+		return CLI_SHOWUSAGE;
+	}
+	mailbox_id = a->argv[3];
+
+	mailbox = ast_mwi_mailbox_get(mailbox_id);
+	if (mailbox) {
+		ast_cli(a->fd,
+			"Mailbox: %s\n"
+			"NewMessages: %u\n"
+			"OldMessages: %u\n",
+			ast_sorcery_object_get_id(mailbox),
+			mailbox->msgs_new,
+			mailbox->msgs_old);
+
+		ast_mwi_mailbox_unref(mailbox);
+	} else {
+		ast_cli(a->fd, "External MWI mailbox '%s' not found.\n", mailbox_id);
+	}
+
+	return CLI_SUCCESS;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+static char *handle_mwi_update_mailbox(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+	struct ast_mwi_mailbox_object *mailbox;
+	const char *mailbox_id;
+	unsigned int num_new;
+	unsigned int num_old;
+
+	switch (cmd) {
+	case CLI_INIT:
+		e->command = "mwi update mailbox";
+		e->usage =
+			"Usage: mwi update mailbox <mailbox_id> [<new> [<old>]]\n"
+			"       Update a specific external MWI mailbox.\n";
+		return NULL;
+	case CLI_GENERATE:
+		if (a->pos == 3) {
+			return complete_mailbox(a->word, a->n);
+		}
+		return NULL;
+	}
+
+	if (a->argc < 4 || 6 < a->argc) {
+		return CLI_SHOWUSAGE;
+	}
+	mailbox_id = a->argv[3];
+
+	num_new = 0;
+	if (4 < a->argc) {
+		const char *count_new = a->argv[4];
+
+		if (sscanf(count_new, "%u", &num_new) != 1) {
+			ast_cli(a->fd, "Invalid NewMessages: '%s'.\n", count_new);
+			return CLI_SHOWUSAGE;
+		}
+	}
+
+	num_old = 0;
+	if (5 < a->argc) {
+		const char *count_old = a->argv[5];
+
+		if (sscanf(count_old, "%u", &num_old) != 1) {
+			ast_cli(a->fd, "Invalid OldMessages: '%s'.\n", count_old);
+			return CLI_SHOWUSAGE;
+		}
+	}
+
+	mailbox = ast_mwi_mailbox_alloc(mailbox_id);
+	if (mailbox) {
+		ast_mwi_mailbox_set_msgs_new(mailbox, num_new);
+		ast_mwi_mailbox_set_msgs_old(mailbox, num_old);
+		if (ast_mwi_mailbox_update(mailbox)) {
+			ast_cli(a->fd, "Could not update mailbox %s.\n",
+				ast_sorcery_object_get_id(mailbox));
+		} else {
+			ast_cli(a->fd, "Updated mailbox %s.\n", ast_sorcery_object_get_id(mailbox));
+		}
+
+		ast_mwi_mailbox_unref(mailbox);
+	}
+
+	return CLI_SUCCESS;
+}
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+#if defined(MWI_DEBUG_CLI)
+static struct ast_cli_entry mwi_cli[] = {
+	AST_CLI_DEFINE(handle_mwi_delete_all, "Delete all external MWI mailboxes"),
+	AST_CLI_DEFINE(handle_mwi_delete_like, "Delete external MWI mailboxes matching regex"),
+	AST_CLI_DEFINE(handle_mwi_delete_mailbox, "Delete a specific external MWI mailbox"),
+	AST_CLI_DEFINE(handle_mwi_list_all, "List all external MWI mailboxes"),
+	AST_CLI_DEFINE(handle_mwi_list_like, "List external MWI mailboxes matching regex"),
+	AST_CLI_DEFINE(handle_mwi_show_mailbox, "Show a specific external MWI mailbox"),
+	AST_CLI_DEFINE(handle_mwi_update_mailbox, "Update a specific external MWI mailbox"),
+};
+#endif	/* defined(MWI_DEBUG_CLI) */
+
+/*!
+ * \internal
+ * \brief Post initial MWI count events.
+ * \since 12.1.0
+ *
+ * \return Nothing
+ */
+static void mwi_initial_events(void)
+{
+	struct ao2_container *mailboxes;
+	const struct ast_mwi_mailbox_object *mailbox;
+	struct ao2_iterator iter;
+
+	/* Get all mailbox counts. */
+	mailboxes = ast_mwi_mailbox_get_all();
+	if (!mailboxes) {
+		return;
+	}
+
+	/* Post all mailbox counts. */
+	iter = ao2_iterator_init(mailboxes, AO2_ITERATOR_UNLINK);
+	for (; (mailbox = ao2_iterator_next(&iter)); ast_mwi_mailbox_unref(mailbox)) {
+		mwi_post_event(mailbox);
+	}
+	ao2_iterator_destroy(&iter);
+
+	ao2_ref(mailboxes, -1);
+}
+
+static int unload_module(void)
+{
+	ast_vm_unregister(vm_table.module_name);
+#if defined(MWI_DEBUG_CLI)
+	ast_cli_unregister_multiple(mwi_cli, ARRAY_LEN(mwi_cli));
+#endif	/* defined(MWI_DEBUG_CLI) */
+	ast_sorcery_observer_remove(mwi_sorcery, MWI_MAILBOX_TYPE, &mwi_observers);
+
+	ast_sorcery_unref(mwi_sorcery);
+	mwi_sorcery = NULL;
+
+	return 0;
+}
+
+static int load_module(void)
+{
+	if (mwi_sorcery_init()
+		|| ast_sorcery_observer_add(mwi_sorcery, MWI_MAILBOX_TYPE, &mwi_observers)
+#if defined(MWI_DEBUG_CLI)
+		|| ast_cli_register_multiple(mwi_cli, ARRAY_LEN(mwi_cli))
+#endif	/* defined(MWI_DEBUG_CLI) */
+		|| ast_vm_register(&vm_table)) {
+		unload_module();
+		return AST_MODULE_LOAD_DECLINE;
+	}
+
+	/* Post initial MWI count events. */
+	mwi_initial_events();
+
+	return AST_MODULE_LOAD_SUCCESS;
+}
+
+AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_GLOBAL_SYMBOLS | AST_MODFLAG_LOAD_ORDER, "Core external MWI resource",
+	.load = load_module,
+	.unload = unload_module,
+	.load_pri = AST_MODPRI_CHANNEL_DEPEND - 5,
+);
diff --git a/res/res_mwi_external.exports.in b/res/res_mwi_external.exports.in
new file mode 100644
index 0000000000000000000000000000000000000000..c8231974b3c886a10625665782ee968bb3272b59
--- /dev/null
+++ b/res/res_mwi_external.exports.in
@@ -0,0 +1,6 @@
+{
+	global:
+		LINKER_SYMBOL_PREFIXast_*;
+	local:
+		*;
+};