aboutsummaryrefslogblamecommitdiffstats
path: root/modules/book-config-ldap/evolution-book-config-ldap.c
blob: 3aa83de2433a09038198caa30bcb195b6ca0f988 (plain) (tree)





















                                                                             
                                    

                                              
                                      
                                         
































































































































































































































































































                                                                            
                                



                                  

                                                             

                                                              
                                                            








                                                                            






                                                                 
































































































































































































                                                                           
                               






























































































































































































































































































































                                                                            
                                                                                 
















                                                                            




                                                                        



















































                                                                            










                                                                           








































































                                                                            
/*
 * evolution-book-config-ldap.c
 *
 * This program 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 2 of the License, or (at your option) version 3.
 *
 * This program 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 the program; if not, see <http://www.gnu.org/licenses/>
 *
 */

#include <config.h>
#include <stdlib.h>
#include <glib/gi18n-lib.h>

#include <libebackend/libebackend.h>

#include <libevolution-utils/e-alert-dialog.h>
#include <misc/e-book-source-config.h>
#include <misc/e-source-config-backend.h>

#include "e-source-ldap.h"

#ifndef G_OS_WIN32
#include <ldap.h>
#ifndef SUNLDAP
#include <ldap_schema.h>
#endif
#else
#include <winldap.h>
#include "openldap-extract.h"
#endif

/* Combo box ordering */
#define LDAP_PORT  389
#define LDAPS_PORT 636
#define MSGC_PORT  3268
#define MSGCS_PORT 3269

typedef ESourceConfigBackend EBookConfigLDAP;
typedef ESourceConfigBackendClass EBookConfigLDAPClass;

typedef struct _Closure Closure;
typedef struct _Context Context;

struct _Closure {
    ESourceConfigBackend *backend;
    ESource *scratch_source;
};

struct _Context {
    GtkWidget *auth_combo;
    GtkWidget *auth_entry;
    GtkWidget *host_entry;
    GtkWidget *port_combo;
    GtkWidget *security_combo;
    GtkWidget *search_base_combo;
    GtkWidget *search_base_button;
    GtkWidget *search_scope_combo;
    GtkWidget *search_filter_entry;
    GtkWidget *limit_spinbutton;
    GtkWidget *can_browse_toggle;
};

/* Module Entry Points */
void e_module_load (GTypeModule *type_module);
void e_module_unload (GTypeModule *type_module);

/* Forward Declarations */
GType e_book_config_ldap_get_type (void);

G_DEFINE_DYNAMIC_TYPE (
    EBookConfigLDAP,
    e_book_config_ldap,
    E_TYPE_SOURCE_CONFIG_BACKEND)

static Closure *
book_config_ldap_closure_new (ESourceConfigBackend *backend,
                              ESource *scratch_source)
{
    Closure *closure;

    closure = g_slice_new (Closure);
    closure->backend = g_object_ref (backend);
    closure->scratch_source = g_object_ref (scratch_source);

    return closure;
}

static void
book_config_ldap_closure_free (Closure *closure)
{
    g_object_unref (closure->backend);
    g_object_unref (closure->scratch_source);

    g_slice_free (Closure, closure);
}

static void
book_config_ldap_context_free (Context *context)
{
    g_object_unref (context->auth_combo);
    g_object_unref (context->auth_entry);
    g_object_unref (context->host_entry);
    g_object_unref (context->port_combo);
    g_object_unref (context->security_combo);
    g_object_unref (context->search_base_combo);
    g_object_unref (context->search_base_button);
    g_object_unref (context->search_scope_combo);
    g_object_unref (context->search_filter_entry);
    g_object_unref (context->limit_spinbutton);
    g_object_unref (context->can_browse_toggle);

    g_slice_free (Context, context);
}

static GtkTreeModel *
book_config_ldap_root_dse_query (ESourceConfigBackend *backend,
                                 ESource *scratch_source)
{
    LDAP *ldap;
    LDAPMessage *result = NULL;
    GtkListStore *store = NULL;
    ESourceAuthentication *extension;
    struct timeval timeout;
    const gchar *alert_id = NULL;
    const gchar *extension_name;
    const gchar *host;
    gchar **values = NULL;
    gint ldap_error;
    gint option;
    gint version;
    guint16 port;
    gint ii;

    const gchar *attrs[] = { "namingContexts", NULL };

    /* FIXME This all runs synchronously in the main loop.
     *       We should do this in a separate thread behind
     *       async/finish functions.  May need to define
     *       some custom GError codes, or maybe just an
     *       LDAP GError domain that reuses LDAP result
     *       codes from <ldap.h>. */

    extension_name = E_SOURCE_EXTENSION_AUTHENTICATION;
    extension = e_source_get_extension (scratch_source, extension_name);

    host = e_source_authentication_get_host (extension);
    port = e_source_authentication_get_port (extension);

    timeout.tv_sec = 60;
    timeout.tv_usec = 0;

    ldap = ldap_init (host, port);
    if (ldap == NULL) {
        alert_id = "addressbook:ldap-init";
        goto exit;
    }

    version = LDAP_VERSION3;
    option = LDAP_OPT_PROTOCOL_VERSION;
    if (ldap_set_option (ldap, option, &version) != LDAP_SUCCESS) {
        /* XXX Define an alert for this. */
        g_warning ("Failed to set protocol version to LDAPv3");
        goto exit;
    }

    /* FIXME Use the user's actual authentication settings. */
    if (ldap_simple_bind_s (ldap, NULL, NULL) != LDAP_SUCCESS) {
        alert_id = "addressbook:ldap-auth";
        goto exit;
    }

    ldap_error = ldap_search_ext_s (
        ldap, LDAP_ROOT_DSE, LDAP_SCOPE_BASE,
        "(objectclass=*)", (gchar **) attrs, 0,
        NULL, NULL, &timeout, LDAP_NO_LIMIT, &result);
    if (ldap_error != LDAP_SUCCESS) {
        alert_id = "addressbook:ldap-search-base";
        goto exit;
    }

    values = ldap_get_values (ldap, result, "namingContexts");
    if (values == NULL || values[0] == NULL || *values[0] == '\0') {
        alert_id = "addressbook:ldap-search-base";
        goto exit;
    }

    store = gtk_list_store_new (1, G_TYPE_STRING);

    for (ii = 0; values[ii] != NULL; ii++) {
        GtkTreeIter iter;

        gtk_list_store_append (store, &iter);
        gtk_list_store_set (store, &iter, 0, values[ii], -1);
    }

exit:
    if (alert_id != NULL) {
        ESourceConfig *config;
        gpointer parent;

        config = e_source_config_backend_get_config (backend);

        parent = gtk_widget_get_toplevel (GTK_WIDGET (config));
        parent = gtk_widget_is_toplevel (parent) ? parent : NULL;

        e_alert_run_dialog_for_args (parent, alert_id, NULL);
    }

    if (values != NULL)
        ldap_value_free (values);

    if (result != NULL)
        ldap_msgfree (result);

    if (ldap != NULL)
        ldap_unbind_s (ldap);

    /* This may be NULL, so don't use a cast macro. */
    return (GtkTreeModel *) store;
}

static gboolean
book_config_ldap_port_to_active (GBinding *binding,
                                 const GValue *source_value,
                                 GValue *target_value,
                                 gpointer unused)
{
    guint port;
    gint active;

    port = g_value_get_uint (source_value);

    switch (port) {
        case 0:  /* initialize to LDAP_PORT */
        case LDAP_PORT:
            active = 0;
            break;

        case LDAPS_PORT:
            active = 1;
            break;

        case MSGC_PORT:
            active = 2;
            break;

        case MSGCS_PORT:
            active = 3;
            break;

        default:
            active = -1;
            break;
    }

    g_value_set_int (target_value, active);

    if (active == -1) {
        GObject *target;
        GtkWidget *entry;
        gchar *text;

        target = g_binding_get_target (binding);
        entry = gtk_bin_get_child (GTK_BIN (target));

        text = g_strdup_printf ("%u", port);
        gtk_entry_set_text (GTK_ENTRY (entry), text);
        g_free (text);
    }

    return TRUE;
}

static gboolean
book_config_ldap_active_to_port (GBinding *binding,
                                 const GValue *source_value,
                                 GValue *target_value,
                                 gpointer unused)
{
    guint port = LDAP_PORT;
    gint active;

    active = g_value_get_int (source_value);

    switch (active) {
        case 0:
            port = LDAP_PORT;
            break;

        case 1:
            port = LDAPS_PORT;
            break;

        case 2:
            port = MSGC_PORT;
            break;

        case 3:
            port = MSGCS_PORT;
            break;

        default:
            active = -1;
            break;
    }

    if (active == -1) {
        GObject *target;
        GtkWidget *entry;
        const gchar *text;
        glong v_long;

        target = g_binding_get_target (binding);
        entry = gtk_bin_get_child (GTK_BIN (target));
        text = gtk_entry_get_text (GTK_ENTRY (entry));

        v_long = text ? strtol (text, NULL, 10) : 0;
        if (v_long != 0 && v_long == CLAMP (v_long, 0, G_MAXUINT16))
            port = (guint) v_long;
    }

    g_value_set_uint (target_value, port);

    return TRUE;
}

static void
book_config_ldap_port_combo_changed (GtkComboBox *combo_box)
{
    if (gtk_combo_box_get_active (combo_box) == -1)
        g_object_notify (G_OBJECT (combo_box), "active");
}

static gboolean
book_config_ldap_port_to_security (GBinding *binding,
                                   const GValue *source_value,
                                   GValue *target_value,
                                   gpointer unused)
{
    switch (g_value_get_int (source_value)) {
        case 0:  /* LDAP_PORT -> StartTLS */
            g_value_set_int (
                target_value,
                E_SOURCE_LDAP_SECURITY_STARTTLS);
            return TRUE;

        case 1:  /* LDAPS_PORT -> LDAP over SSL */
            g_value_set_int (
                target_value,
                E_SOURCE_LDAP_SECURITY_LDAPS);
            return TRUE;

        case 2:  /* MSGC_PORT -> StartTLS */
            g_value_set_int (
                target_value,
                E_SOURCE_LDAP_SECURITY_STARTTLS);
            return TRUE;

        case 3:  /* MSGCS_PORT -> LDAP over SSL */
            g_value_set_int (
                target_value,
                E_SOURCE_LDAP_SECURITY_LDAPS);
            return TRUE;

        default:
            break;
    }

    return FALSE;
}

static void
book_config_ldap_search_base_button_clicked_cb (GtkButton *button,
                                                Closure *closure)
{
    Context *context;
    GtkComboBox *combo_box;
    GtkTreeModel *model;
    const gchar *uid;

    uid = e_source_get_uid (closure->scratch_source);
    context = g_object_get_data (G_OBJECT (closure->backend), uid);
    g_return_if_fail (context != NULL);

    model = book_config_ldap_root_dse_query (
        closure->backend, closure->scratch_source);

    combo_box = GTK_COMBO_BOX (context->search_base_combo);
    gtk_combo_box_set_model (combo_box, model);
    gtk_combo_box_set_active (combo_box, 0);

    if (model != NULL)
        g_object_unref (model);
}

static gboolean
book_config_ldap_query_port_tooltip_cb (GtkComboBox *combo_box,
                                        gint x,
                                        gint y,
                                        gboolean keyboard_mode,
                                        GtkTooltip *tooltip)
{
    GtkTreeModel *model;
    GtkTreeIter iter;
    gchar *text;

    /* XXX This only works if the port number was selected from
     *     the drop down menu.  No tooltip is shown if the user
     *     types the port number, even if the same port number
     *     is listed in the drop down menu.  That's fixable but
     *     the code would be a lot messier, and is arguably a
     *     job for GtkComboBox. */

    if (!gtk_combo_box_get_active_iter (combo_box, &iter))
        return FALSE;

    model = gtk_combo_box_get_model (combo_box);
    gtk_tree_model_get (model, &iter, 1, &text, -1);
    gtk_tooltip_set_text (tooltip, text);
    g_free (text);

    return TRUE;
}

static GtkWidget *
book_config_build_port_combo (void)
{
    GtkWidget *widget;
    GtkComboBox *combo_box;
    GtkCellRenderer *renderer;
    GtkListStore *store;
    GtkTreeIter iter;

    store = gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_STRING);

    gtk_list_store_append (store, &iter);
    gtk_list_store_set (
        store, &iter,
        0, G_STRINGIFY (LDAP_PORT),
        1, _("Standard LDAP Port"), -1);

    gtk_list_store_append (store, &iter);
    gtk_list_store_set (
        store, &iter,
        0, G_STRINGIFY (LDAPS_PORT),
        1, _("LDAP over SSL (deprecated)"), -1);

    gtk_list_store_append (store, &iter);
    gtk_list_store_set (
        store, &iter,
        0, G_STRINGIFY (MSGC_PORT),
        1, _("Microsoft Global Catalog"), -1);

    gtk_list_store_append (store, &iter);
    gtk_list_store_set (
        store, &iter,
        0, G_STRINGIFY (MSGCS_PORT),
        1, _("Microsoft Global Catalog over SSL"), -1);

    widget = gtk_combo_box_new_with_entry ();

    combo_box = GTK_COMBO_BOX (widget);
    gtk_combo_box_set_model (combo_box, GTK_TREE_MODEL (store));
    gtk_combo_box_set_entry_text_column (combo_box, 0);

    renderer = gtk_cell_renderer_text_new ();
    g_object_set (renderer, "sensitive", FALSE, NULL);
    gtk_cell_layout_pack_start (
        GTK_CELL_LAYOUT (widget), renderer, FALSE);
    gtk_cell_layout_add_attribute (
        GTK_CELL_LAYOUT (widget), renderer, "text", 1);

    gtk_widget_set_has_tooltip (widget, TRUE);

    g_signal_connect (
        widget, "query-tooltip",
        G_CALLBACK (book_config_ldap_query_port_tooltip_cb), NULL);

    g_object_unref (store);

    return widget;
}

static void
book_config_ldap_insert_notebook_widget (GtkWidget *vbox,
                                         GtkSizeGroup *size_group,
                                         const gchar *caption,
                                         GtkWidget *widget)
{
    GtkWidget *hbox;
    GtkWidget *label;

    /* This is similar to e_source_config_insert_widget(),
     * but instead adds the widget to the LDAP notebook. */

    hbox = gtk_hbox_new (FALSE, 12);
    gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 0);
    gtk_widget_show (hbox);

    label = gtk_label_new (caption);
    gtk_misc_set_alignment (GTK_MISC (label), 1.0, 0.5);
    gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
    gtk_size_group_add_widget (size_group, label);
    gtk_widget_show (label);

    gtk_box_pack_start (GTK_BOX (hbox), widget, TRUE, TRUE, 0);
}

static void
book_config_ldap_insert_widgets (ESourceConfigBackend *backend,
                                 ESource *scratch_source)
{
    ESourceConfig *config;
    ESourceExtension *extension;
    GtkSizeGroup *size_group;
    GtkNotebook *notebook;
    GtkWidget *container;
    GtkWidget *widget;
    GtkWidget *page;
    GtkWidget *hbox;
    Context *context;
    PangoAttribute *attr;
    PangoAttrList *attr_list;
    const gchar *extension_name;
    const gchar *tab_label;
    const gchar *uid;
    gboolean is_new_source;

    context = g_slice_new (Context);
    uid = e_source_get_uid (scratch_source);
    config = e_source_config_backend_get_config (backend);

    g_object_set_data_full (
        G_OBJECT (backend), uid, context,
        (GDestroyNotify) book_config_ldap_context_free);

    e_book_source_config_add_offline_toggle (
        E_BOOK_SOURCE_CONFIG (config), scratch_source);

    container = e_source_config_get_page (config, scratch_source);

    /* Extra padding between the notebook and the options above. */
    widget = gtk_alignment_new (0.0, 0.0, 1.0, 1.0);
    gtk_alignment_set_padding (GTK_ALIGNMENT (widget), 6, 0, 0, 0);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_notebook_new ();
    gtk_container_add (GTK_CONTAINER (container), widget);
    gtk_widget_show (widget);

    notebook = GTK_NOTEBOOK (widget);

    /* For bold section headers. */
    attr_list = pango_attr_list_new ();
    attr = pango_attr_weight_new (PANGO_WEIGHT_BOLD);
    pango_attr_list_insert (attr_list, attr);

    /* Page 1 */

    tab_label = _("Connecting to LDAP");
    page = gtk_vbox_new (FALSE, 12);
    gtk_container_set_border_width (GTK_CONTAINER (page), 12);
    gtk_notebook_append_page (notebook, page, NULL);
    gtk_notebook_set_tab_label_text (notebook, page, tab_label);
    gtk_widget_show (page);

    size_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);

    /* Page 1 : Server Information */

    widget = gtk_vbox_new (FALSE, 6);
    gtk_box_pack_start (GTK_BOX (page), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_label_new (_("Server Information"));
    gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.5);
    gtk_label_set_attributes (GTK_LABEL (widget), attr_list);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    widget = gtk_alignment_new (0.0, 0.0, 1.0, 1.0);
    gtk_alignment_set_padding (GTK_ALIGNMENT (widget), 0, 0, 12, 0);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_vbox_new (FALSE, 6);
    gtk_container_add (GTK_CONTAINER (container), widget);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_entry_new ();
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Server:"), widget);
    context->host_entry = g_object_ref (widget);
    gtk_widget_show (widget);

    widget = book_config_build_port_combo ();
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Port:"), widget);
    context->port_combo = g_object_ref (widget);
    gtk_widget_show (widget);

    /* This must follow the order of ESourceLDAPSecurity. */
    widget = gtk_combo_box_text_new ();
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget),
        _("None"));
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget),
        _("LDAP over SSL (deprecated)"));
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget),
        _("StartTLS (recommended)"));
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Encryption:"), widget);
    context->security_combo = g_object_ref (widget);
    gtk_widget_show (widget);

    g_object_bind_property_full (
        context->port_combo, "active",
        context->security_combo, "active",
        G_BINDING_DEFAULT,
        book_config_ldap_port_to_security,
        NULL,  /* binding is one-way */
        NULL, (GDestroyNotify) NULL);

    /* If this is a new source, initialize security to StartTLS. */
    if (e_source_config_get_original_source (config) == NULL)
        gtk_combo_box_set_active (GTK_COMBO_BOX (widget), 2);

    /* Page 1 : Authentication */

    widget = gtk_vbox_new (FALSE, 6);
    gtk_box_pack_start (GTK_BOX (page), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_label_new (_("Authentication"));
    gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.5);
    gtk_label_set_attributes (GTK_LABEL (widget), attr_list);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    widget = gtk_alignment_new (0.0, 0.0, 1.0, 1.0);
    gtk_alignment_set_padding (GTK_ALIGNMENT (widget), 0, 0, 12, 0);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_vbox_new (FALSE, 6);
    gtk_container_add (GTK_CONTAINER (container), widget);
    gtk_widget_show (widget);

    container = widget;

    /* This must follow the order of ESourceLDAPAuthentication. */
    widget = gtk_combo_box_text_new ();
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget),
        _("Anonymous"));
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget),
        _("Using email address"));
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget),
        _("Using distinguished name (DN)"));
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Method:"), widget);
    context->auth_combo = g_object_ref (widget);
    gtk_widget_show (widget);

    gtk_widget_set_tooltip_text (
        widget, _("This is the method Evolution will use to "
        "authenticate you.  Note that setting this to \"Using "
        "email address\" requires anonymous access to your LDAP "
        "server."));

    widget = gtk_entry_new ();
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Username:"), widget);
    context->auth_entry = g_object_ref (widget);
    gtk_widget_show (widget);

    g_object_unref (size_group);

    /* Page 2 */

    tab_label = _("Using LDAP");
    page = gtk_vbox_new (FALSE, 12);
    gtk_container_set_border_width (GTK_CONTAINER (page), 12);
    gtk_notebook_append_page (notebook, page, NULL);
    gtk_notebook_set_tab_label_text (notebook, page, tab_label);
    gtk_widget_show (page);

    size_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);

    /* Page 2 : Searching */

    widget = gtk_vbox_new (FALSE, 6);
    gtk_box_pack_start (GTK_BOX (page), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_label_new (_("Searching"));
    gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.5);
    gtk_label_set_attributes (GTK_LABEL (widget), attr_list);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    widget = gtk_alignment_new (0.0, 0.0, 1.0, 1.0);
    gtk_alignment_set_padding (GTK_ALIGNMENT (widget), 0, 0, 12, 0);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_vbox_new (FALSE, 6);
    gtk_container_add (GTK_CONTAINER (container), widget);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_combo_box_new_with_entry ();
    gtk_combo_box_set_entry_text_column (GTK_COMBO_BOX (widget), 0);
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Search Base:"), widget);
    context->search_base_combo = g_object_ref (widget);
    gtk_widget_show (widget);

    widget = gtk_button_new_with_label (
        _("Find Possible Search Bases"));
    gtk_button_set_image (
        GTK_BUTTON (widget), gtk_image_new_from_stock (
        GTK_STOCK_FIND, GTK_ICON_SIZE_BUTTON));
    book_config_ldap_insert_notebook_widget (
        container, size_group, NULL, widget);
    context->search_base_button = g_object_ref (widget);
    gtk_widget_show (widget);

    /* Only sensitive when we have complete
     * server and authentication details. */
    g_object_bind_property (
        config, "complete",
        context->search_base_button, "sensitive",
        G_BINDING_DEFAULT);

    g_signal_connect_data (
        widget, "clicked",
        G_CALLBACK (book_config_ldap_search_base_button_clicked_cb),
        book_config_ldap_closure_new (backend, scratch_source),
        (GClosureNotify) book_config_ldap_closure_free, 0);

    /* This must follow the order of ESourceLDAPScope. */
    widget = gtk_combo_box_text_new ();
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget), _("One Level"));
    gtk_combo_box_text_append_text (
        GTK_COMBO_BOX_TEXT (widget), _("Subtree"));
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Search Scope:"), widget);
    context->search_scope_combo = g_object_ref (widget);
    gtk_widget_show (widget);

    gtk_widget_set_tooltip_text (
        widget, _("The search scope defines how deep you would "
        "like the search to extend down the directory tree.  A "
        "search scope of \"Subtree\" will include all entries "
        "below your search base.  A search scope of \"One Level\" "
        "will only include the entries one level beneath your "
        "search base."));

    widget = gtk_entry_new ();
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Search Filter:"), widget);
    context->search_filter_entry = g_object_ref (widget);
    gtk_widget_show (widget);

    /* Page 2 : Downloading */

    widget = gtk_vbox_new (FALSE, 6);
    gtk_box_pack_start (GTK_BOX (page), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_label_new (_("Downloading"));
    gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.5);
    gtk_label_set_attributes (GTK_LABEL (widget), attr_list);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    widget = gtk_alignment_new (0.0, 0.0, 1.0, 1.0);
    gtk_alignment_set_padding (GTK_ALIGNMENT (widget), 0, 0, 12, 0);
    gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_vbox_new (FALSE, 6);
    gtk_container_add (GTK_CONTAINER (container), widget);
    gtk_widget_show (widget);

    container = widget;

    widget = gtk_hbox_new (FALSE, 6);
    book_config_ldap_insert_notebook_widget (
        container, size_group, _("Limit:"), widget);
    gtk_widget_show (widget);

    hbox = widget;

    widget = gtk_spin_button_new_with_range (0, G_MAXUINT, 1);
    gtk_spin_button_set_numeric (GTK_SPIN_BUTTON (widget), TRUE);
    gtk_box_pack_start (GTK_BOX (hbox), widget, TRUE, TRUE, 0);
    context->limit_spinbutton = g_object_ref (widget);
    gtk_widget_show (widget);

    widget = gtk_label_new (_("contacts"));
    gtk_box_pack_start (GTK_BOX (hbox), widget, FALSE, FALSE, 0);
    gtk_widget_show (widget);

    widget = gtk_check_button_new_with_label (
        _("Browse until limit is reached"));
    book_config_ldap_insert_notebook_widget (
        container, size_group, NULL, widget);
    context->can_browse_toggle = g_object_ref (widget);
    gtk_widget_show (widget);

    g_object_unref (size_group);

    pango_attr_list_unref (attr_list);

    /* Bind widgets to extension properties. */

    extension_name = E_SOURCE_EXTENSION_AUTHENTICATION;
    is_new_source = !e_source_has_extension (scratch_source, extension_name);
    extension = e_source_get_extension (scratch_source, extension_name);

    g_object_bind_property (
        extension, "host",
        context->host_entry, "text",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    g_object_bind_property_full (
        extension, "port",
        context->port_combo, "active",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE,
        book_config_ldap_port_to_active,
        book_config_ldap_active_to_port,
        NULL, (GDestroyNotify) NULL);

    /* "active" doesn't change when setting custom port in entry,
       thus check also on the "changed" signal */
    g_signal_connect (context->port_combo, "changed",
        G_CALLBACK (book_config_ldap_port_combo_changed), NULL);

    g_object_bind_property (
        extension, "user",
        context->auth_entry, "text",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    extension_name = E_SOURCE_EXTENSION_LDAP_BACKEND;
    extension = e_source_get_extension (scratch_source, extension_name);

    g_object_bind_property (
        extension, "authentication",
        context->auth_combo, "active",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    g_object_bind_property (
        extension, "can-browse",
        context->can_browse_toggle, "active",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    g_object_bind_property (
        extension, "limit",
        context->limit_spinbutton, "value",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    widget = gtk_bin_get_child (GTK_BIN (context->search_base_combo));

    g_object_bind_property (
        extension, "root-dn",
        widget, "text",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    g_object_bind_property (
        extension, "scope",
        context->search_scope_combo, "active",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    g_object_bind_property (
        extension, "filter",
        context->search_filter_entry, "text",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    g_object_bind_property (
        extension, "security",
        context->security_combo, "active",
        G_BINDING_BIDIRECTIONAL |
        G_BINDING_SYNC_CREATE);

    /* initialize values from UI into extension, if the source
       is a fresh new source; bindings will take care of proper
       values setting into extension properties
    */
    if (is_new_source) {
        g_object_notify (G_OBJECT (context->host_entry), "text");
        g_object_notify (G_OBJECT (context->port_combo), "active");
        g_object_notify (G_OBJECT (context->auth_entry), "text");
        g_object_notify (G_OBJECT (context->auth_combo), "active");
    }
}

static gboolean
book_config_ldap_check_complete (ESourceConfigBackend *backend,
                                 ESource *scratch_source)
{
    ESourceLDAPAuthentication auth;
    ESourceExtension *extension;
    const gchar *extension_name;
    const gchar *host;
    const gchar *user;
    guint16 port;

    extension_name = E_SOURCE_EXTENSION_LDAP_BACKEND;
    extension = e_source_get_extension (scratch_source, extension_name);

    auth = e_source_ldap_get_authentication (E_SOURCE_LDAP (extension));

    extension_name = E_SOURCE_EXTENSION_AUTHENTICATION;
    extension = e_source_get_extension (scratch_source, extension_name);

    host = e_source_authentication_get_host (
        E_SOURCE_AUTHENTICATION (extension));
    port = e_source_authentication_get_port (
        E_SOURCE_AUTHENTICATION (extension));
    user = e_source_authentication_get_user (
        E_SOURCE_AUTHENTICATION (extension));

    if (host == NULL || *host == '\0' || port == 0)
        return FALSE;

    if (auth != E_SOURCE_LDAP_AUTHENTICATION_NONE)
        if (user == NULL || *user == '\0')
            return FALSE;

    return TRUE;
}

static void
e_book_config_ldap_class_init (ESourceConfigBackendClass *class)
{
    EExtensionClass *extension_class;

    extension_class = E_EXTENSION_CLASS (class);
    extension_class->extensible_type = E_TYPE_BOOK_SOURCE_CONFIG;

    class->parent_uid = "ldap-stub";
    class->backend_name = "ldap";
    class->insert_widgets = book_config_ldap_insert_widgets;
    class->check_complete = book_config_ldap_check_complete;
}

static void
e_book_config_ldap_class_finalize (ESourceConfigBackendClass *class)
{
}

static void
e_book_config_ldap_init (ESourceConfigBackend *backend)
{
}

G_MODULE_EXPORT void
e_module_load (GTypeModule *type_module)
{
    e_source_ldap_type_register (type_module);
    e_book_config_ldap_register_type (type_module);
}

G_MODULE_EXPORT void
e_module_unload (GTypeModule *type_module)
{
}