/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Author: * Chris Toshok (toshok@ximian.com) * * Copyright 2000, Ximian, Inc. */ #define DEBUG #include "config.h" #include #include #include #include #include #include #ifdef DEBUG #define LDAP_DEBUG #define LDAP_DEBUG_ADD #endif #include #ifdef DEBUG #undef LDAP_DEBUG #endif #if LDAP_VENDOR_VERSION > 20000 #define OPENLDAP2 #else #define OPENLDAP1 #endif #ifdef OPENLDAP2 #include "ldap_schema.h" #endif #include #include #include #include "pas-backend-ldap.h" #include "pas-backend-card-sexp.h" #include "pas-book.h" #include "pas-card-cursor.h" #include #define LDAP_MAX_SEARCH_RESPONSES 100 /* interval for our poll_ldap timeout */ #define LDAP_POLL_INTERVAL 20 /* timeout for ldap_result */ #define LDAP_RESULT_TIMEOUT_MILLIS 10 /* smart grouping stuff */ #define GROUPING_INITIAL_SIZE 1 #define GROUPING_MAXIMUM_SIZE 200 /* the next two are in milliseconds */ #define GROUPING_MINIMUM_WAIT 0 /* we never send updates faster than this, to avoid totally spamming the UI */ #define GROUPING_MAXIMUM_WAIT 250 /* we always send updates (if there are pending cards) when we hit this */ #define TV_TO_MILLIS(timeval) ((timeval).tv_sec * 1000 + (timeval).tv_usec / 1000) /* the objectClasses we need */ #define TOP "top" #define PERSON "person" #define ORGANIZATIONALPERSON "organizationalPerson" #define INETORGPERSON "inetOrgPerson" #define EVOLUTIONPERSON "evolutionPerson" static gchar *query_prop_to_ldap(gchar *query_prop); static PASBackendClass *pas_backend_ldap_parent_class; typedef struct _PASBackendLDAPCursorPrivate PASBackendLDAPCursorPrivate; typedef struct _PASBackendLDAPBookView PASBackendLDAPBookView; typedef struct LDAPOp LDAPOp; struct _PASBackendLDAPPrivate { char *uri; gboolean connected; GList *clients; gchar *ldap_host; gchar *ldap_rootdn; int ldap_port; int ldap_scope; int ldap_limit; GList *book_views; LDAP *ldap; EList *supported_fields; /* whether or not there's support for the objectclass we need to store all our additional fields */ gboolean evolutionPersonSupported; gboolean evolutionPersonChecked; gboolean writable; /* whether or not there's a request in process on our LDAP* */ LDAPOp *current_op; GList *pending_ops; int op_idle; }; struct _PASBackendLDAPCursorPrivate { PASBackend *backend; PASBook *book; GList *elements; long num_elements; }; struct _PASBackendLDAPBookView { PASBookView *book_view; PASBackendLDAPPrivate *blpriv; gchar *search; PASBackendCardSExp *card_sexp; int search_timeout; int search_msgid; LDAPOp *search_op; /* grouping stuff */ GList *pending_adds; /* the cards we're sending */ int num_pending_adds; /* the number waiting to be sent */ int target_pending_adds; /* the cutoff that forces a flush to the client, if it happens before the timeout */ int num_sent_this_time; /* the number of cards we sent to the client before the most recent timeout */ int num_sent_last_time; /* the number of cards we sent to the client before the previous timeout */ glong grouping_time_start; /* used by poll_ldap to only send the status messages once */ gboolean notified_receiving_results; }; typedef gboolean (*LDAPOpHandler)(PASBackend *backend, LDAPOp *op); typedef void (*LDAPOpDtor)(PASBackend *backend, LDAPOp *op); struct LDAPOp { LDAPOpHandler handler; LDAPOpDtor dtor; PASBackend *backend; PASBook *book; PASBookView *view; }; static void ldap_op_init (LDAPOp *op, PASBackend *backend, PASBook *book, PASBookView *view, LDAPOpHandler handler, LDAPOpDtor dtor); static void ldap_op_process_current (PASBackend *backend); static void ldap_op_process (LDAPOp *op); static void ldap_op_restart (LDAPOp *op); static gboolean ldap_op_process_on_idle (PASBackend *backend); static void ldap_op_finished (LDAPOp *op); static ECardSimple *build_card_from_entry (LDAP *ldap, LDAPMessage *e, GList **existing_objectclasses); static void email_populate (ECardSimple *card, char **values); struct berval** email_ber (ECardSimple *card); static gboolean email_compare (ECardSimple *ecard1, ECardSimple *ecard2); static void homephone_populate (ECardSimple *card, char **values); struct berval** homephone_ber (ECardSimple *card); static gboolean homephone_compare (ECardSimple *ecard1, ECardSimple *ecard2); static void business_populate (ECardSimple *card, char **values); struct berval** business_ber (ECardSimple *card); static gboolean business_compare (ECardSimple *ecard1, ECardSimple *ecard2); static void anniversary_populate (ECardSimple *card, char **values); struct berval** anniversary_ber (ECardSimple *card); static gboolean anniversary_compare (ECardSimple *ecard1, ECardSimple *ecard2); static void birthday_populate (ECardSimple *card, char **values); struct berval** birthday_ber (ECardSimple *card); static gboolean birthday_compare (ECardSimple *ecard1, ECardSimple *ecard2); struct prop_info { ECardSimpleField field_id; char *query_prop; char *ldap_attr; #define PROP_TYPE_STRING 0x01 #define PROP_TYPE_COMPLEX 0x02 #define PROP_DN 0x04 #define PROP_EVOLVE 0x08 int prop_type; /* the remaining items are only used for the TYPE_COMPLEX props */ /* used when reading from the ldap server populates ECard with the values in **values. */ void (*populate_ecard_func)(ECardSimple *card, char **values); /* used when writing to an ldap server. returns a NULL terminated array of berval*'s */ struct berval** (*ber_func)(ECardSimple *card); /* used to compare list attributes */ gboolean (*compare_func)(ECardSimple *card1, ECardSimple *card2); } prop_info[] = { #define COMPLEX_PROP(fid,q,a,ctor,ber,cmp) {fid, q, a, PROP_TYPE_COMPLEX, ctor, ber, cmp} #define E_COMPLEX_PROP(fid,q,a,ctor,ber,cmp) {fid, q, a, PROP_TYPE_COMPLEX | PROP_EVOLVE, ctor, ber, cmp} #define STRING_PROP(fid,q,a) {fid, q, a, PROP_TYPE_STRING} #define E_STRING_PROP(fid,q,a) {fid, q, a, PROP_TYPE_STRING | PROP_EVOLVE} /* name fields */ STRING_PROP (E_CARD_SIMPLE_FIELD_FULL_NAME, "full_name", "cn" ), STRING_PROP (E_CARD_SIMPLE_FIELD_FAMILY_NAME, "family_name", "sn" ), /* email addresses */ COMPLEX_PROP (E_CARD_SIMPLE_FIELD_EMAIL, "email", "mail", email_populate, email_ber, email_compare), /* phone numbers */ E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_PRIMARY, "primary_phone", "primaryPhone"), COMPLEX_PROP (E_CARD_SIMPLE_FIELD_PHONE_BUSINESS, "business_phone", "telephoneNumber", business_populate, business_ber, business_compare), COMPLEX_PROP (E_CARD_SIMPLE_FIELD_PHONE_HOME, "home_phone", "homePhone", homephone_populate, homephone_ber, homephone_compare), STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_MOBILE, "mobile_phone", "mobile"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_CAR, "car_phone", "carPhone"), STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_FAX, "business_fax", "facsimileTelephoneNumber"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_HOME_FAX, "home_fax", "homeFacsimileTelephoneNumber"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_OTHER, "other_phone", "otherPhone"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_OTHER, "other_fax", "otherFacsimileTelephoneNumber"), STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_ISDN, "isdn", "internationaliSDNNumber"), STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_PAGER, "pager", "pager"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_RADIO, "radio", "radio"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_TELEX, "telex", "telex"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_ASSISTANT, "assistant_phone", "assistantPhone"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_COMPANY, "company_phone", "companyPhone"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_CALLBACK, "callback_phone", "callbackPhone"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_PHONE_TTYTTD, "tty", "tty"), /* org information */ STRING_PROP (E_CARD_SIMPLE_FIELD_ORG, "org", "o"), STRING_PROP (E_CARD_SIMPLE_FIELD_ORG_UNIT, "org_unit", "ou"), STRING_PROP (E_CARD_SIMPLE_FIELD_OFFICE, "office", "roomNumber"), STRING_PROP (E_CARD_SIMPLE_FIELD_TITLE, "title", "title"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_ROLE, "role", "businessRole"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_MANAGER, "manager", "managerName"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_ASSISTANT, "assistant", "assistantName"), /* addresses */ STRING_PROP (E_CARD_SIMPLE_FIELD_ADDRESS_BUSINESS, "business_address", "postalAddress"), STRING_PROP (E_CARD_SIMPLE_FIELD_ADDRESS_HOME, "home_address", "homePostalAddress"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_ADDRESS_OTHER, "other_address", "otherPostalAddress"), /* misc fields */ STRING_PROP (E_CARD_SIMPLE_FIELD_URL, "url", "labeledURI"), /* map nickname to displayName */ STRING_PROP (E_CARD_SIMPLE_FIELD_NICKNAME, "nickname", "displayName"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_SPOUSE, "spouse", "spouseName"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_NOTE, "note", "note"), E_COMPLEX_PROP (E_CARD_SIMPLE_FIELD_ANNIVERSARY, "anniversary", "anniversary", anniversary_populate, anniversary_ber, anniversary_compare), E_COMPLEX_PROP (E_CARD_SIMPLE_FIELD_BIRTH_DATE, "birth_date", "birthDate", birthday_populate, birthday_ber, birthday_compare), E_STRING_PROP (E_CARD_SIMPLE_FIELD_MAILER, "mailer", "mailer"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_FILE_AS, "file_as", "fileAs"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_CATEGORIES, "categories", "categories"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_CALURI, "caluri", "calendarURI"), E_STRING_PROP (E_CARD_SIMPLE_FIELD_FBURL, "fburl", "freeBusyURI") /* E_CARD_SIMPLE_FIELD_NAME_OR_ORG, */ #undef E_STRING_PROP #undef STRING_PROP #undef E_COMPLEX_PROP #undef COMPLEX_PROP }; static int num_prop_infos = sizeof(prop_info) / sizeof(prop_info[0]); static void view_destroy(GtkObject *object, gpointer data) { CORBA_Environment ev; GNOME_Evolution_Addressbook_Book corba_book; PASBook *book = (PASBook *)data; PASBackendLDAP *bl; GList *list; bl = PAS_BACKEND_LDAP(pas_book_get_backend(book)); for (list = bl->priv->book_views; list; list = g_list_next(list)) { PASBackendLDAPBookView *view = list->data; if (view->book_view == PAS_BOOK_VIEW(object)) { if (view->search_timeout != 0) { /* we have a search running on the ldap connection. remove the timeout handler and anbandon the msg id */ g_source_remove(view->search_timeout); if (view->search_msgid != -1) ldap_abandon (bl->priv->ldap, view->search_msgid); /* if the search op is the current op, finish it. else, remove it from the list and nuke it ourselves. */ if (view->search_op == bl->priv->current_op) ldap_op_finished (view->search_op); else { bl->priv->pending_ops = g_list_remove (bl->priv->pending_ops, view->search_op); view->search_op->dtor (view->search_op->backend, view->search_op); } } g_free (view->search); gtk_object_unref (GTK_OBJECT (view->card_sexp)); g_free (view); bl->priv->book_views = g_list_remove_link(bl->priv->book_views, list); g_list_free_1(list); break; } } corba_book = bonobo_object_corba_objref(BONOBO_OBJECT(book)); CORBA_exception_init(&ev); GNOME_Evolution_Addressbook_Book_unref(corba_book, &ev); if (ev._major != CORBA_NO_EXCEPTION) { g_warning("view_destroy: Exception unreffing " "corba book.\n"); } CORBA_exception_free(&ev); } static void add_to_supported_fields (PASBackendLDAP *bl, char **attrs, GHashTable *attr_hash) { int i; for (i = 0; attrs[i]; i ++) { char *query_prop = g_hash_table_lookup (attr_hash, attrs[i]); if (query_prop) { e_list_append (bl->priv->supported_fields, query_prop); /* handle the list attributes here */ if (!strcmp (query_prop, "email")) { e_list_append (bl->priv->supported_fields, "email_2"); e_list_append (bl->priv->supported_fields, "email_3"); } else if (!strcmp (query_prop, "business_phone")) { e_list_append (bl->priv->supported_fields, "business_phone_2"); } else if (!strcmp (query_prop, "home_phone")) { e_list_append (bl->priv->supported_fields, "home_phone_2"); } } } } static void add_oc_attributes_to_supported_fields (PASBackendLDAP *bl, LDAPObjectClass *oc) { int i; GHashTable *attr_hash = g_hash_table_new (g_str_hash, g_str_equal); for (i = 0; i < num_prop_infos; i ++) g_hash_table_insert (attr_hash, prop_info[i].ldap_attr, prop_info[i].query_prop); if (oc->oc_at_oids_must) add_to_supported_fields (bl, oc->oc_at_oids_must, attr_hash); if (oc->oc_at_oids_may) add_to_supported_fields (bl, oc->oc_at_oids_may, attr_hash); g_hash_table_destroy (attr_hash); } static void check_schema_support (PASBackendLDAP *bl) { char *attrs[2]; LDAPMessage *resp; LDAP *ldap = bl->priv->ldap; bl->priv->evolutionPersonChecked = TRUE; attrs[0] = "objectClasses"; attrs[1] = NULL; if (ldap_search_ext_s (ldap, "cn=Subschema", LDAP_SCOPE_BASE, "(objectClass=subschema)", attrs, 0, NULL, NULL, NULL, 0, &resp) == LDAP_SUCCESS) { char **values; values = ldap_get_values (ldap, resp, "objectClasses"); if (values) { int i; for (i = 0; values[i]; i ++) { int j; int code; const char *err; LDAPObjectClass *oc = ldap_str2objectclass (values[i], &code, &err, 0); if (!oc) continue; for (j = 0; oc->oc_names[j]; j++) if (!g_strcasecmp (oc->oc_names[j], EVOLUTIONPERSON)) { g_print ("support found on ldap server for objectclass evolutionPerson\n"); bl->priv->evolutionPersonSupported = TRUE; add_oc_attributes_to_supported_fields (bl, oc); } else if (!g_strcasecmp (oc->oc_names[j], INETORGPERSON) || !g_strcasecmp (oc->oc_names[j], ORGANIZATIONALPERSON) || !g_strcasecmp (oc->oc_names[j], PERSON)) { add_oc_attributes_to_supported_fields (bl, oc); } ldap_objectclass_free (oc); } } } } static void pas_backend_ldap_connect (PASBackendLDAP *bl) { PASBackendLDAPPrivate *blpriv = bl->priv; /* close connection first if it's open first */ if (blpriv->ldap) ldap_unbind (blpriv->ldap); blpriv->ldap = ldap_init (blpriv->ldap_host, blpriv->ldap_port); #ifdef DEBUG { int debug_level = ~0; ldap_set_option (blpriv->ldap, LDAP_OPT_DEBUG_LEVEL, &debug_level); } #endif if (NULL != blpriv->ldap) { ldap_simple_bind_s(blpriv->ldap, NULL /*binddn*/, NULL /*passwd*/); blpriv->connected = TRUE; /* check to see if evolutionPerson is supported, if we can (me might not be able to if we can't authenticate. if we can't, try again in auth_user.) */ check_schema_support (bl); } else { g_warning ("pas_backend_ldap_connect failed for " "'ldap://%s:%d/%s'\n", blpriv->ldap_host, blpriv->ldap_port, blpriv->ldap_rootdn ? blpriv->ldap_rootdn : ""); blpriv->connected = FALSE; } } static ECardSimple * search_for_dn_with_objectclasses (PASBackendLDAP *bl, const char *dn, GList **existing_objectclasses) { LDAP *ldap = bl->priv->ldap; LDAPMessage *res, *e; ECardSimple *result = NULL; if (ldap_search_s (ldap, dn, LDAP_SCOPE_BASE, "(objectclass=*)", NULL, 0, &res) == LDAP_SUCCESS) { e = ldap_first_entry (ldap, res); while (NULL != e) { if (!strcmp (ldap_get_dn (ldap, e), dn)) { printf ("found it\n"); result = build_card_from_entry (ldap, e, existing_objectclasses); break; } e = ldap_next_entry (ldap, e); } ldap_msgfree(res); } return result; } static ECardSimple * search_for_dn (PASBackendLDAP *bl, const char *dn) { return search_for_dn_with_objectclasses (bl, dn, NULL); } static void ldap_op_init (LDAPOp *op, PASBackend *backend, PASBook *book, PASBookView *view, LDAPOpHandler handler, LDAPOpDtor dtor) { op->backend = backend; op->book = book; op->view = view; op->handler = handler; op->dtor = dtor; } static void ldap_op_process_current (PASBackend *backend) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); LDAPOp *op = bl->priv->current_op; if (!bl->priv->connected) { if (op->view) pas_book_view_notify_status_message (op->view, _("Connecting to LDAP server...")); pas_backend_ldap_connect(bl); } if (bl->priv->connected) { if (op->handler (backend, op)) ldap_op_finished (op); } else { if (op->view) { pas_book_view_notify_status_message (op->view, _("Unable to connect to LDAP server.")); pas_book_view_notify_complete (op->view); } ldap_op_finished (op); } } static void ldap_op_process (LDAPOp *op) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (op->backend); if (bl->priv->current_op) { /* operation in progress. queue this op for later and return. */ if (op->view) pas_book_view_notify_status_message (op->view, _("Waiting for connection to LDAP server...")); bl->priv->pending_ops = g_list_append (bl->priv->pending_ops, op); } else { /* nothing going on, do this op now */ bl->priv->current_op = op; ldap_op_process_current (op->backend); } } static gboolean ldap_op_process_on_idle (PASBackend *backend) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); bl->priv->op_idle = 0; ldap_op_process_current (backend); return FALSE; } static void ldap_op_restart (LDAPOp *op) { PASBackend *backend = op->backend; PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); g_return_if_fail (op == bl->priv->current_op); bl->priv->op_idle = g_idle_add((GSourceFunc)ldap_op_process_on_idle, backend); } static void ldap_op_finished (LDAPOp *op) { PASBackend *backend = op->backend; PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); g_return_if_fail (op == bl->priv->current_op); op->dtor (backend, op); if (bl->priv->pending_ops) { bl->priv->current_op = bl->priv->pending_ops->data; bl->priv->pending_ops = g_list_remove_link (bl->priv->pending_ops, bl->priv->pending_ops); bl->priv->op_idle = g_idle_add((GSourceFunc)ldap_op_process_on_idle, backend); } else { bl->priv->current_op = NULL; } } static int ldap_error_to_response (int ldap_error) { if (ldap_error == LDAP_SUCCESS) return GNOME_Evolution_Addressbook_BookListener_Success; else if (LDAP_NAME_ERROR (ldap_error)) return GNOME_Evolution_Addressbook_BookListener_CardNotFound; else if (ldap_error == LDAP_INSUFFICIENT_ACCESS) return GNOME_Evolution_Addressbook_BookListener_PermissionDenied; else if (ldap_error == LDAP_SERVER_DOWN) return GNOME_Evolution_Addressbook_BookListener_RepositoryOffline; else if (ldap_error == LDAP_ALREADY_EXISTS) return GNOME_Evolution_Addressbook_BookListener_CardIdAlreadyExists; else return GNOME_Evolution_Addressbook_BookListener_OtherError; } static char * create_dn_from_ecard (ECardSimple *card, const char *root_dn) { char *cn, *cn_part = NULL; char *dn; cn = e_card_simple_get (card, E_CARD_SIMPLE_FIELD_FULL_NAME); if (cn) { if (strchr (cn, ',')) { /* need to escape commas */ char *new_cn = g_malloc0 (strlen (cn) * 3 + 1); int i, j; for (i = 0, j = 0; i < strlen (cn); i ++) { if (cn[i] == ',') { sprintf (new_cn + j, "%%%02X", cn[i]); j += 3; } else { new_cn[j++] = cn[i]; } } cn_part = g_strdup_printf ("cn=%s", new_cn); g_free (new_cn); } else { cn_part = g_strdup_printf ("cn=%s", cn); } } else { cn_part = g_strdup (""); } dn = g_strdup_printf ("%s%s%s", cn_part, (root_dn && strlen(root_dn)) ? "," : "", (root_dn && strlen(root_dn)) ? root_dn: ""); g_free (cn_part); g_print ("generated dn: %s\n", dn); return dn; } static void free_mods (GPtrArray *mods) { int i = 0; LDAPMod *mod; while ((mod = g_ptr_array_index (mods, i++))) { int j; g_free (mod->mod_type); if (mod->mod_op & LDAP_MOD_BVALUES) { for (j = 0; mod->mod_bvalues[j]; j++) { g_free (mod->mod_bvalues[j]->bv_val); g_free (mod->mod_bvalues[j]); } } else { for (j = 0; mod->mod_values[j]; j++) g_free (mod->mod_values[j]); } g_free (mod); } g_ptr_array_free (mods, TRUE); } static GPtrArray* build_mods_from_ecards (PASBackendLDAP *bl, ECardSimple *current, ECardSimple *new, gboolean *new_dn_needed) { gboolean adding = (current == NULL); GPtrArray *result = g_ptr_array_new(); int i; if (new_dn_needed) *new_dn_needed = FALSE; /* we walk down the list of properties we can deal with (that big table at the top of the file) */ for (i = 0; i < num_prop_infos; i ++) { gboolean include; gboolean new_prop_present = FALSE; gboolean current_prop_present = FALSE; struct berval** new_prop_bers = NULL; char *new_prop = NULL; char *current_prop = NULL; /* XXX if it's an evolutionPerson prop and the ldap server doesn't support that objectclass, skip it. */ if (prop_info[i].prop_type & PROP_EVOLVE && !bl->priv->evolutionPersonSupported) continue; /* get the value for the new card, and compare it to the value in the current card to see if we should update it -- if adding is TRUE, short circuit the check. */ if (prop_info[i].prop_type & PROP_TYPE_STRING) { new_prop = e_card_simple_get (new, prop_info[i].field_id); new_prop_present = (new_prop != NULL); } else { new_prop_bers = prop_info[i].ber_func (new); new_prop_present = (new_prop_bers != NULL); } /* need to set INCLUDE to true if the field needs to show up in the ldap modify request */ if (adding) { /* if we're creating a new card, include it if the field is there at all */ if (prop_info[i].prop_type & PROP_TYPE_STRING) include = (new_prop_present && *new_prop); /* empty strings cause problems */ else include = new_prop_present; } else { /* if we're modifying an existing card, include it if the current field value is different than the new one, if it didn't exist previously, or if it's been removed. */ if (prop_info[i].prop_type & PROP_TYPE_STRING) { current_prop = e_card_simple_get (current, prop_info[i].field_id); current_prop_present = (current_prop != NULL); if (new_prop && current_prop) include = *new_prop && strcmp (new_prop, current_prop); else include = (!!new_prop != !!current_prop); } else { int j; struct berval **current_prop_bers = prop_info[i].ber_func (current); current_prop_present = (current_prop_bers != NULL); /* free up the current_prop_bers */ if (current_prop_bers) { for (j = 0; current_prop_bers[j]; j++) { g_free (current_prop_bers[j]->bv_val); g_free (current_prop_bers[j]); } g_free (current_prop_bers); } include = !prop_info[i].compare_func (new, current); } } if (include) { LDAPMod *mod = g_new (LDAPMod, 1); /* the included attribute has changed - we need to update the dn if it's one of the attributes we compute the dn from. */ if (new_dn_needed) *new_dn_needed |= prop_info[i].prop_type & PROP_DN; if (adding) { mod->mod_op = LDAP_MOD_ADD; } else { if (!new_prop_present) mod->mod_op = LDAP_MOD_DELETE; else if (!current_prop_present) mod->mod_op = LDAP_MOD_ADD; else mod->mod_op = LDAP_MOD_REPLACE; } mod->mod_type = g_strdup (prop_info[i].ldap_attr); if (prop_info[i].prop_type & PROP_TYPE_STRING) { mod->mod_values = g_new (char*, 2); mod->mod_values[0] = new_prop; mod->mod_values[1] = NULL; } else { /* PROP_TYPE_COMPLEX */ mod->mod_op |= LDAP_MOD_BVALUES; mod->mod_bvalues = new_prop_bers; } g_ptr_array_add (result, mod); } } /* NULL terminate the list of modifications */ g_ptr_array_add (result, NULL); return result; } static void add_objectclass_mod (PASBackendLDAP *bl, GPtrArray *mod_array, GList *existing_objectclasses) { LDAPMod *objectclass_mod; if (existing_objectclasses) { int i = 0; objectclass_mod = g_new (LDAPMod, 1); objectclass_mod->mod_op = LDAP_MOD_ADD; objectclass_mod->mod_type = g_strdup ("objectClass"); objectclass_mod->mod_values = g_new (char*, bl->priv->evolutionPersonSupported ? 6 : 5); /* yes, this is a linear search for each of our objectclasses, but really, how many objectclasses are there going to be in any sane ldap entry? */ #define FIND_INSERT(oc) \ if (!g_list_find_custom (existing_objectclasses, (oc), (GCompareFunc)g_strcasecmp)) \ objectclass_mod->mod_values[i++] = g_strdup ((oc)); FIND_INSERT (TOP); FIND_INSERT (PERSON); FIND_INSERT (ORGANIZATIONALPERSON); FIND_INSERT (INETORGPERSON); if (bl->priv->evolutionPersonSupported) FIND_INSERT (EVOLUTIONPERSON); objectclass_mod->mod_values[i] = NULL; if (i) { g_ptr_array_add (mod_array, objectclass_mod); } else { g_free (objectclass_mod->mod_type); g_free (objectclass_mod->mod_values); g_free (objectclass_mod); } } else { objectclass_mod = g_new (LDAPMod, 1); objectclass_mod->mod_op = LDAP_MOD_ADD; objectclass_mod->mod_type = g_strdup ("objectClass"); objectclass_mod->mod_values = g_new (char*, bl->priv->evolutionPersonSupported ? 6 : 5); objectclass_mod->mod_values[0] = g_strdup (TOP); objectclass_mod->mod_values[1] = g_strdup (PERSON); objectclass_mod->mod_values[2] = g_strdup (ORGANIZATIONALPERSON); objectclass_mod->mod_values[3] = g_strdup (INETORGPERSON); if (bl->priv->evolutionPersonSupported) { objectclass_mod->mod_values[4] = g_strdup (EVOLUTIONPERSON); objectclass_mod->mod_values[5] = NULL; } else { objectclass_mod->mod_values[4] = NULL; } g_ptr_array_add (mod_array, objectclass_mod); } } typedef struct { LDAPOp op; char *vcard; } LDAPCreateOp; static gboolean create_card_handler (PASBackend *backend, LDAPOp *op) { LDAPCreateOp *create_op = (LDAPCreateOp*)op; PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); ECard *new_ecard; ECardSimple *new_card; char *dn; int response; int ldap_error; GPtrArray *mod_array; LDAPMod **ldap_mods; LDAP *ldap; printf ("vcard = %s\n", create_op->vcard); new_ecard = e_card_new (create_op->vcard); new_card = e_card_simple_new (new_ecard); dn = create_dn_from_ecard (new_card, bl->priv->ldap_rootdn); e_card_simple_set_id (new_card, dn); /* for the notification code below */ ldap = bl->priv->ldap; /* build our mods */ mod_array = build_mods_from_ecards (bl, NULL, new_card, NULL); #if 0 if (!mod_array) { /* there's an illegal field in there. report UnsupportedAttribute back */ g_free (dn); gtk_object_unref (GTK_OBJECT(new_card)); pas_book_respond_create (create_op->op.book, GNOME_Evolution_Addressbook_BookListener_UnsupportedField, dn); return TRUE; } #endif /* remove the NULL at the end */ g_ptr_array_remove (mod_array, NULL); /* add our objectclass(es) */ add_objectclass_mod (bl, mod_array, NULL); /* then put the NULL back */ g_ptr_array_add (mod_array, NULL); #ifdef LDAP_DEBUG_ADD { int i; printf ("Sending the following to the server as ADD\n"); for (i = 0; g_ptr_array_index(mod_array, i); i ++) { LDAPMod *mod = g_ptr_array_index(mod_array, i); if (mod->mod_op & LDAP_MOD_DELETE) printf ("del "); else if (mod->mod_op & LDAP_MOD_REPLACE) printf ("rep "); else printf ("add "); if (mod->mod_op & LDAP_MOD_BVALUES) printf ("ber "); else printf (" "); printf (" %s:\n", mod->mod_type); if (mod->mod_op & LDAP_MOD_BVALUES) { int j; for (j = 0; mod->mod_bvalues[j] && mod->mod_bvalues[j]->bv_val; j++) printf ("\t\t'%s'\n", mod->mod_bvalues[j]->bv_val); } else { int j; for (j = 0; mod->mod_values[j]; j++) printf ("\t\t'%s'\n", mod->mod_values[j]); } } } #endif ldap_mods = (LDAPMod**)mod_array->pdata; if (op->view) pas_book_view_notify_status_message (op->view, _("Adding card to LDAP server...")); /* actually perform the ldap add */ ldap_error = ldap_add_s (ldap, dn, ldap_mods); if (ldap_error == LDAP_SUCCESS) { /* the card was created, let's let the views know about it */ GList *l; for (l = bl->priv->book_views; l; l = l->next) { CORBA_Environment ev; gboolean match; PASBackendLDAPBookView *view = l->data; CORBA_exception_init(&ev); bonobo_object_dup_ref(bonobo_object_corba_objref(BONOBO_OBJECT(view->book_view)), &ev); match = pas_backend_card_sexp_match_vcard (view->card_sexp, e_card_simple_get_vcard_assume_utf8 (new_card)); if (match) pas_book_view_notify_add_1 (view->book_view, e_card_simple_get_vcard_assume_utf8 (new_card)); pas_book_view_notify_complete (view->book_view); bonobo_object_release_unref(bonobo_object_corba_objref(BONOBO_OBJECT(view->book_view)), &ev); } } else { ldap_perror (ldap, "ldap_add_s"); } if (op->view) pas_book_view_notify_complete (op->view); /* and clean up */ free_mods (mod_array); g_free (dn); gtk_object_unref (GTK_OBJECT(new_card)); /* and lastly respond */ response = ldap_error_to_response (ldap_error); pas_book_respond_create (create_op->op.book, response, dn); /* we're synchronous */ return TRUE; } static void create_card_dtor (PASBackend *backend, LDAPOp *op) { LDAPCreateOp *create_op = (LDAPCreateOp*)op; if (op->view) bonobo_object_release_unref(bonobo_object_corba_objref(BONOBO_OBJECT(op->view)), NULL); g_free (create_op->vcard); g_free (create_op); } static void pas_backend_ldap_process_create_card (PASBackend *backend, PASBook *book, PASRequest *req) { LDAPCreateOp *create_op = g_new (LDAPCreateOp, 1); PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); PASBookView *book_view = NULL; if (bl->priv->book_views) { PASBackendLDAPBookView *v = bl->priv->book_views->data; book_view = v->book_view; bonobo_object_dup_ref(bonobo_object_corba_objref(BONOBO_OBJECT(book_view)), NULL); } ldap_op_init ((LDAPOp*)create_op, backend, book, book_view, create_card_handler, create_card_dtor); create_op->vcard = req->vcard; ldap_op_process ((LDAPOp*)create_op); } typedef struct { LDAPOp op; char *id; } LDAPRemoveOp; static gboolean remove_card_handler (PASBackend *backend, LDAPOp *op) { LDAPRemoveOp *remove_op = (LDAPRemoveOp*)op; PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); int response; int ldap_error; ECardSimple *simple; if (op->view) pas_book_view_notify_status_message (op->view, _("Removing card from LDAP server...")); simple = search_for_dn (bl, remove_op->id); if (simple) { ldap_error = ldap_delete_s (bl->priv->ldap, remove_op->id); if (ldap_error == LDAP_SUCCESS) { /* the card was removed, let's let the views know about it */ GList *l; for (l = bl->priv->book_views; l; l = l->next) { CORBA_Environment ev; gboolean match; PASBackendLDAPBookView *view = l->data; CORBA_exception_init(&ev); bonobo_object_dup_ref(bonobo_object_corba_objref(BONOBO_OBJECT(view->book_view)), &ev); match = pas_backend_card_sexp_match_vcard (view->card_sexp, e_card_simple_get_vcard_assume_utf8 (simple)); if (match) pas_book_view_notify_remove (view->book_view, remove_op->id); pas_book_view_notify_complete (view->book_view); bonobo_object_release_unref(bonobo_object_corba_objref(BONOBO_OBJECT(view->book_view)), &ev); } } else { ldap_perror (bl->priv->ldap, "ldap_delete_s"); } response = ldap_error_to_response (ldap_error); } else response = GNOME_Evolution_Addressbook_BookListener_CardNotFound; pas_book_respond_remove (remove_op->op.book, response); if (op->view) pas_book_view_notify_complete (op->view); /* we're synchronous */ return TRUE; } static void remove_card_dtor (PASBackend *backend, LDAPOp *op) { LDAPRemoveOp *remove_op = (LDAPRemoveOp*)op; if (op->view) bonobo_object_release_unref(bonobo_object_corba_objref(BONOBO_OBJECT(op->view)), NULL); g_free (remove_op->id); g_free (remove_op); } static void pas_backend_ldap_process_remove_card (PASBackend *backend, PASBook *book, PASRequest *req) { LDAPRemoveOp *remove_op = g_new (LDAPRemoveOp, 1); PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); PASBookView *book_view = NULL; if (bl->priv->book_views) { PASBackendLDAPBookView *v = bl->priv->book_views->data; book_view = v->book_view; bonobo_object_dup_ref(bonobo_object_corba_objref(BONOBO_OBJECT(book_view)), NULL); } ldap_op_init ((LDAPOp*)remove_op, backend, book, book_view, remove_card_handler, remove_card_dtor); remove_op->id = req->id; ldap_op_process ((LDAPOp*)remove_op); } typedef struct { LDAPOp op; char *vcard; } LDAPModifyOp; static gboolean modify_card_handler (PASBackend *backend, LDAPOp *op) { LDAPModifyOp *modify_op = (LDAPModifyOp*)op; PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); ECard *new_ecard; const char *id; int response; int ldap_error = LDAP_SUCCESS; GPtrArray *mod_array; LDAPMod **ldap_mods; LDAP *ldap; ECardSimple *current_card; GList *existing_objectclasses = NULL; new_ecard = e_card_new (modify_op->vcard); id = e_card_get_id(new_ecard); ldap = bl->priv->ldap; if (op->view) pas_book_view_notify_status_message (op->view, _("Modifying card from LDAP server...")); current_card = search_for_dn_with_objectclasses (bl, id, &existing_objectclasses); if (current_card) { ECardSimple *new_card = e_card_simple_new (new_ecard); gboolean need_new_dn; /* build our mods */ mod_array = build_mods_from_ecards (bl, current_card, new_card, &need_new_dn); if (mod_array->len > 0) { /* remove the NULL at the end */ g_ptr_array_remove (mod_array, NULL); /* add our objectclass(es), making sure evolutionPerson is there if it's supported */ add_objectclass_mod (bl, mod_array, existing_objectclasses); /* then put the NULL back */ g_ptr_array_add (mod_array, NULL); ldap_mods = (LDAPMod**)mod_array->pdata; /* actually perform the ldap modify */ ldap_error = ldap_modify_ext_s (ldap, id, ldap_mods, NULL, NULL); if (ldap_error == LDAP_SUCCESS) { /* the card was modified, let's let the views know about it */ GList *l; for (l = bl->priv->book_views; l; l = l->next) { CORBA_Environment ev; gboolean old_match, new_match; PASBackendLDAPBookView *view = l->data; CORBA_exception_init(&ev); bonobo_object_dup_ref(bonobo_object_corba_objref(BONOBO_OBJECT(view->book_view)), &ev); old_match = pas_backend_card_sexp_match_vcard (view->card_sexp, e_card_simple_get_vcard_assume_utf8 (current_card)); new_match = pas_backend_card_sexp_match_vcard (view->card_sexp, modify_op->vcard); if (old_match && new_match) pas_book_view_notify_change_1 (view->book_view, modify_op->vcard); else if (new_match) pas_book_view_notify_add_1 (view->book_view, modify_op->vcard); else /* if (old_match) */ pas_book_view_notify_remove (view->book_view, e_card_simple_get_id (new_card)); pas_book_view_notify_complete (view->book_view); bonobo_object_release_unref(bonobo_object_corba_objref(BONOBO_OBJECT(view->book_view)), &ev); } } else { ldap_perror (ldap, "ldap_modify_s"); } } else { g_print ("modify list empty. no modification sent\n"); } /* and clean up */ free_mods (mod_array); g_list_foreach (existing_objectclasses, (GFunc)g_free, NULL); g_list_free (existing_objectclasses); gtk_object_unref (GTK_OBJECT(new_card)); gtk_object_unref (GTK_OBJECT(current_card)); } else { g_print ("didn't find original card\n"); } response = ldap_error_to_response (ldap_error); pas_book_respond_modify (modify_op->op.book, response); if (op->view) pas_book_view_notify_complete (op->view); /* we're synchronous */ return TRUE; } static void modify_card_dtor (PASBackend *backend, LDAPOp *op) { LDAPModifyOp *modify_op = (LDAPModifyOp*)op; if (op->view) bonobo_object_release_unref(bonobo_object_corba_objref(BONOBO_OBJECT(op->view)), NULL); g_free (modify_op->vcard); g_free (modify_op); } static void pas_backend_ldap_process_modify_card (PASBackend *backend, PASBook *book, PASRequest *req) { LDAPModifyOp *modify_op = g_new (LDAPModifyOp, 1); PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); PASBookView *book_view = NULL; if (bl->priv->book_views) { PASBackendLDAPBookView *v = bl->priv->book_views->data; book_view = v->book_view; bonobo_object_dup_ref(bonobo_object_corba_objref(BONOBO_OBJECT(book_view)), NULL); } ldap_op_init ((LDAPOp*)modify_op, backend, book, book_view, modify_card_handler, modify_card_dtor); modify_op->vcard = req->vcard; ldap_op_process ((LDAPOp*)modify_op); } static void pas_backend_ldap_process_get_vcard (PASBackend *backend, PASBook *book, PASRequest *req) { PASBackendLDAP *bl; ECardSimple *simple; bl = PAS_BACKEND_LDAP (pas_book_get_backend (book)); simple = search_for_dn (bl, req->id); if (simple) { pas_book_respond_get_vcard (book, GNOME_Evolution_Addressbook_BookListener_Success, e_card_simple_get_vcard_assume_utf8 (simple)); gtk_object_unref (GTK_OBJECT (simple)); } else { pas_book_respond_get_vcard (book, GNOME_Evolution_Addressbook_BookListener_CardNotFound, ""); } } typedef struct { LDAPOp op; PASBook *book; } LDAPGetCursorOp; static long get_length(PASCardCursor *cursor, gpointer data) { PASBackendLDAPCursorPrivate *cursor_data = (PASBackendLDAPCursorPrivate *) data; return cursor_data->num_elements; } static char * get_nth(PASCardCursor *cursor, long n, gpointer data) { PASBackendLDAPCursorPrivate *cursor_data = (PASBackendLDAPCursorPrivate *) data; g_return_val_if_fail (n < cursor_data->num_elements, NULL); return (char*)g_list_nth (cursor_data->elements, n); } static void cursor_destroy(GtkObject *object, gpointer data) { CORBA_Environment ev; GNOME_Evolution_Addressbook_Book corba_book; PASBackendLDAPCursorPrivate *cursor_data = (PASBackendLDAPCursorPrivate *) data; corba_book = bonobo_object_corba_objref(BONOBO_OBJECT(cursor_data->book)); CORBA_exception_init(&ev); GNOME_Evolution_Addressbook_Book_unref(corba_book, &ev); if (ev._major != CORBA_NO_EXCEPTION) { g_warning("cursor_destroy: Exception unreffing " "corba book.\n"); } CORBA_exception_free(&ev); /* free the ldap specific cursor information */ g_list_foreach (cursor_data->elements, (GFunc)g_free, NULL); g_list_free (cursor_data->elements); g_free(cursor_data); } static void pas_backend_ldap_build_all_cards_list(PASBackend *backend, PASBackendLDAPCursorPrivate *cursor_data) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); LDAP *ldap = bl->priv->ldap; int ldap_error; LDAPMessage *res, *e; if ((ldap_error = ldap_search_s (ldap, bl->priv->ldap_rootdn, bl->priv->ldap_scope, "(objectclass=*)", NULL, 0, &res)) != LDAP_SUCCESS) { g_warning ("ldap error '%s' in " "pas_backend_ldap_build_all_cards_list\n", ldap_err2string(ldap_error)); } cursor_data->elements = NULL; cursor_data->num_elements = ldap_count_entries (ldap, res); e = ldap_first_entry(ldap, res); while (NULL != e) { /* for now just make a list of the dn's */ #if 0 for ( a = ldap_first_attribute( ldap, e, &ber ); a != NULL; a = ldap_next_attribute( ldap, e, ber ) ) { } #else cursor_data->elements = g_list_prepend(cursor_data->elements, g_strdup(ldap_get_dn(ldap, e))); #endif e = ldap_next_entry(ldap, e); } ldap_msgfree(res); } static gboolean get_cursor_handler (PASBackend *backend, LDAPOp *op) { LDAPGetCursorOp *cursor_op = (LDAPGetCursorOp*)op; CORBA_Environment ev; int ldap_error = 0; PASCardCursor *cursor; GNOME_Evolution_Addressbook_Book corba_book; PASBackendLDAPCursorPrivate *cursor_data; PASBook *book = cursor_op->book; cursor_data = g_new(PASBackendLDAPCursorPrivate, 1); cursor_data->backend = backend; cursor_data->book = book; pas_backend_ldap_build_all_cards_list(backend, cursor_data); corba_book = bonobo_object_corba_objref(BONOBO_OBJECT(book)); CORBA_exception_init(&ev); GNOME_Evolution_Addressbook_Book_ref(corba_book, &ev); if (ev._major != CORBA_NO_EXCEPTION) { g_warning("pas_backend_file_process_get_cursor: Exception reffing " "corba book.\n"); } CORBA_exception_free(&ev); cursor = pas_card_cursor_new(get_length, get_nth, cursor_data); gtk_signal_connect(GTK_OBJECT(cursor), "destroy", GTK_SIGNAL_FUNC(cursor_destroy), cursor_data); pas_book_respond_get_cursor (book, ldap_error_to_response (ldap_error), cursor); /* we're synchronous */ return TRUE; } static void get_cursor_dtor (PASBackend *backend, LDAPOp *op) { g_free (op); } static void pas_backend_ldap_process_get_cursor (PASBackend *backend, PASBook *book, PASRequest *req) { LDAPGetCursorOp *op = g_new (LDAPGetCursorOp, 1); ldap_op_init ((LDAPOp*)op, backend, book, NULL, get_cursor_handler, get_cursor_dtor); ldap_op_process ((LDAPOp*)op); } /* List property functions */ static void email_populate(ECardSimple *card, char **values) { int i; for (i = 0; values[i] && i < 3; i ++) { e_card_simple_set_email (card, i, values[i]); } } struct berval** email_ber(ECardSimple *card) { struct berval** result; const char *emails[3]; int i, j, num; num = 0; for (i = 0; i < 3; i ++) { emails[i] = e_card_simple_get_email (card, E_CARD_SIMPLE_EMAIL_ID_EMAIL + i); if (emails[i]) num++; } if (num == 0) return NULL; result = g_new (struct berval*, num + 1); for (i = 0; i < num; i ++) result[i] = g_new (struct berval, 1); j = 0; for (i = 0; i < 3; i ++) { if (emails[i]) { result[j]->bv_val = g_strdup (emails[i]); result[j++]->bv_len = strlen (emails[i]); } } result[num] = NULL; return result; } static gboolean email_compare (ECardSimple *ecard1, ECardSimple *ecard2) { const char *email1, *email2; int i; for (i = 0; i < 3; i ++) { gboolean equal; email1 = e_card_simple_get_email (ecard1, E_CARD_SIMPLE_EMAIL_ID_EMAIL + i); email2 = e_card_simple_get_email (ecard2, E_CARD_SIMPLE_EMAIL_ID_EMAIL + i); if (email1 && email2) equal = !strcmp (email1, email2); else equal = (!!email1 == !!email2); if (!equal) return equal; } return TRUE; } static void homephone_populate(ECardSimple *card, char **values) { if (values[0]) e_card_simple_set (card, E_CARD_SIMPLE_FIELD_PHONE_HOME, values[0]); if (values[1]) e_card_simple_set (card, E_CARD_SIMPLE_FIELD_PHONE_HOME_2, values[1]); } struct berval** homephone_ber(ECardSimple *card) { struct berval** result; const char *homephones[3]; int i, j, num; num = 0; if ((homephones[0] = e_card_simple_get (card, E_CARD_SIMPLE_FIELD_PHONE_HOME))) num++; if ((homephones[1] = e_card_simple_get (card, E_CARD_SIMPLE_FIELD_PHONE_HOME_2))) num++; if (num == 0) return NULL; result = g_new (struct berval*, num + 1); for (i = 0; i < num; i ++) result[i] = g_new (struct berval, 1); j = 0; for (i = 0; i < 2; i ++) { if (homephones[i]) { result[j]->bv_val = g_strdup (homephones[i]); result[j++]->bv_len = strlen (homephones[i]); } } result[num] = NULL; return result; } static gboolean homephone_compare (ECardSimple *ecard1, ECardSimple *ecard2) { int phone_ids[2] = { E_CARD_SIMPLE_FIELD_PHONE_HOME, E_CARD_SIMPLE_FIELD_PHONE_HOME_2 }; const char *phone1, *phone2; int i; for (i = 0; i < 2; i ++) { gboolean equal; phone1 = e_card_simple_get (ecard1, phone_ids[i]); phone2 = e_card_simple_get (ecard2, phone_ids[i]); if (phone1 && phone2) equal = !strcmp (phone1, phone2); else equal = (!!phone1 == !!phone2); if (!equal) return equal; } return TRUE; } static void business_populate(ECardSimple *card, char **values) { if (values[0]) e_card_simple_set (card, E_CARD_SIMPLE_FIELD_PHONE_BUSINESS, values[0]); if (values[1]) e_card_simple_set (card, E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_2, values[1]); } struct berval** business_ber(ECardSimple *card) { struct berval** result; const char *business_phones[3]; int i, j, num; num = 0; if ((business_phones[0] = e_card_simple_get (card, E_CARD_SIMPLE_FIELD_PHONE_BUSINESS))) num++; if ((business_phones[1] = e_card_simple_get (card, E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_2))) num++; if (num == 0) return NULL; result = g_new (struct berval*, num + 1); for (i = 0; i < num; i ++) result[i] = g_new (struct berval, 1); j = 0; for (i = 0; i < 2; i ++) { if (business_phones[i]) { result[j]->bv_val = g_strdup (business_phones[i]); result[j++]->bv_len = strlen (business_phones[i]); } } result[num] = NULL; return result; } static gboolean business_compare (ECardSimple *ecard1, ECardSimple *ecard2) { int phone_ids[2] = { E_CARD_SIMPLE_FIELD_PHONE_BUSINESS, E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_2 }; const char *phone1, *phone2; int i; for (i = 0; i < 2; i ++) { gboolean equal; phone1 = e_card_simple_get (ecard1, phone_ids[i]); phone2 = e_card_simple_get (ecard2, phone_ids[i]); if (phone1 && phone2) equal = !strcmp (phone1, phone2); else equal = (!!phone1 == !!phone2); if (!equal) return equal; } return TRUE; } static void anniversary_populate (ECardSimple *card, char **values) { if (values[0]) { ECardDate dt = e_card_date_from_string (values[0]); gtk_object_set (GTK_OBJECT (card->card), "anniversary", &dt, NULL); } } struct berval** anniversary_ber (ECardSimple *card) { ECardDate *dt; struct berval** result = NULL; gtk_object_get (GTK_OBJECT (card->card), "anniversary", &dt, NULL); if (dt) { char *anniversary; anniversary = e_card_date_to_string (dt); result = g_new (struct berval*, 2); result[0] = g_new (struct berval, 1); result[0]->bv_val = anniversary; result[0]->bv_len = strlen (anniversary); result[1] = NULL; } return result; } static gboolean anniversary_compare (ECardSimple *ecard1, ECardSimple *ecard2) { ECardDate *dt; char *date1 = NULL, *date2 = NULL; gboolean equal; gtk_object_get (GTK_OBJECT (ecard1->card), "anniversary", &dt, NULL); if (dt) date1 = e_card_date_to_string (dt); gtk_object_get (GTK_OBJECT (ecard2->card), "anniversary", &dt, NULL); if (dt) date2 = e_card_date_to_string (dt); if (date1 && date2) equal = !strcmp (date1, date2); else equal = (!!date1 == !!date2); g_free (date1); g_free (date2); return equal; } static void birthday_populate (ECardSimple *card, char **values) { if (values[0]) { ECardDate dt = e_card_date_from_string (values[0]); gtk_object_set (GTK_OBJECT (card->card), "birth_date", &dt, NULL); } } struct berval** birthday_ber (ECardSimple *card) { ECardDate *dt; struct berval** result = NULL; gtk_object_get (GTK_OBJECT (card->card), "birth_date", &dt, NULL); if (dt) { char *birthday; birthday = e_card_date_to_string (dt); result = g_new (struct berval*, 2); result[0] = g_new (struct berval, 1); result[0]->bv_val = birthday; result[0]->bv_len = strlen (birthday); result[1] = NULL; } return result; } static gboolean birthday_compare (ECardSimple *ecard1, ECardSimple *ecard2) { ECardDate *dt; char *date1 = NULL, *date2 = NULL; gboolean equal; gtk_object_get (GTK_OBJECT (ecard1->card), "birth_date", &dt, NULL); if (dt) date1 = e_card_date_to_string (dt); gtk_object_get (GTK_OBJECT (ecard2->card), "birth_date", &dt, NULL); if (dt) date2 = e_card_date_to_string (dt); if (date1 && date2) equal = !strcmp (date1, date2); else equal = (!!date1 == !!date2); g_free (date1); g_free (date2); return equal; } typedef struct { GList *list; PASBackendLDAP *bl; } PASBackendLDAPSExpData; #define IS_RFC2254_CHAR(c) ((c) == '*' || (c) =='\\' || (c) == '(' || (c) == ')' || (c) == '\0') static char * rfc2254_escape(char *str) { int i; int len = strlen(str); int newlen = 0; for (i = 0; i < len; i ++) { if (IS_RFC2254_CHAR(str[i])) newlen += 3; else newlen ++; } if (len == newlen) { return g_strdup (str); } else { char *newstr = g_malloc0 (newlen + 1); int j = 0; for (i = 0; i < len; i ++) { if (IS_RFC2254_CHAR(str[i])) { sprintf (newstr + j, "\\%02x", str[i]); j+= 3; } else { newstr[j++] = str[i]; } } return newstr; } } static ESExpResult * func_and(struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { PASBackendLDAPSExpData *ldap_data = data; ESExpResult *r; char ** strings; if (argc > 0) { int i; strings = g_new0(char*, argc+3); strings[0] = g_strdup ("(&"); strings[argc+3 - 2] = g_strdup (")"); for (i = 0; i < argc; i ++) { GList *list_head = ldap_data->list; if (!list_head) break; strings[argc - i] = list_head->data; ldap_data->list = g_list_remove_link(list_head, list_head); g_list_free_1(list_head); } ldap_data->list = g_list_prepend(ldap_data->list, g_strjoinv(" ", strings)); for (i = 0 ; i < argc + 2; i ++) g_free (strings[i]); g_free (strings); } r = e_sexp_result_new(f, ESEXP_RES_BOOL); r->value.bool = FALSE; return r; } static ESExpResult * func_or(struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { PASBackendLDAPSExpData *ldap_data = data; ESExpResult *r; char ** strings; if (argc > 0) { int i; strings = g_new0(char*, argc+3); strings[0] = g_strdup ("(|"); strings[argc+3 - 2] = g_strdup (")"); for (i = 0; i < argc; i ++) { GList *list_head = ldap_data->list; if (!list_head) break; strings[argc - i] = list_head->data; ldap_data->list = g_list_remove_link(list_head, list_head); g_list_free_1(list_head); } ldap_data->list = g_list_prepend(ldap_data->list, g_strjoinv(" ", strings)); for (i = 0 ; i < argc + 2; i ++) g_free (strings[i]); g_free (strings); } r = e_sexp_result_new(f, ESEXP_RES_BOOL); r->value.bool = FALSE; return r; } static ESExpResult * func_not(struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { PASBackendLDAPSExpData *ldap_data = data; ESExpResult *r; /* just replace the head of the list with the NOT of it. */ if (argc > 0) { char *term = ldap_data->list->data; ldap_data->list->data = g_strdup_printf("(!%s)", term); g_free (term); } r = e_sexp_result_new(f, ESEXP_RES_BOOL); r->value.bool = FALSE; return r; } static ESExpResult * func_contains(struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { PASBackendLDAPSExpData *ldap_data = data; ESExpResult *r; if (argc == 2 && argv[0]->type == ESEXP_RES_STRING && argv[1]->type == ESEXP_RES_STRING) { char *propname = argv[0]->value.string; char *str = rfc2254_escape(argv[1]->value.string); gboolean one_star = FALSE; if (strlen(str) == 0) one_star = TRUE; if (!strcmp (propname, "x-evolution-any-field")) { int i; int query_length; char *big_query; char *match_str; match_str = g_strdup_printf("=*%s%s)", str, one_star ? "" : "*"); query_length = 3; /* strlen ("(|") + strlen (")") */ for (i = 0; i < num_prop_infos; i ++) { query_length += 1 /* strlen ("(") */ + strlen(prop_info[i].ldap_attr) + strlen (match_str); } big_query = g_malloc0(query_length + 1); strcat (big_query, "(|"); for (i = 0; i < num_prop_infos; i ++) { strcat (big_query, "("); strcat (big_query, prop_info[i].ldap_attr); strcat (big_query, match_str); } strcat (big_query, ")"); ldap_data->list = g_list_prepend(ldap_data->list, big_query); g_free (match_str); } else { char *ldap_attr = query_prop_to_ldap(propname); if (ldap_attr) ldap_data->list = g_list_prepend(ldap_data->list, g_strdup_printf("(%s=*%s%s)", ldap_attr, str, one_star ? "" : "*")); } g_free (str); } r = e_sexp_result_new(f, ESEXP_RES_BOOL); r->value.bool = FALSE; return r; } static ESExpResult * func_is(struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { PASBackendLDAPSExpData *ldap_data = data; ESExpResult *r; if (argc == 2 && argv[0]->type == ESEXP_RES_STRING && argv[1]->type == ESEXP_RES_STRING) { char *propname = argv[0]->value.string; char *str = rfc2254_escape(argv[1]->value.string); char *ldap_attr = query_prop_to_ldap(propname); if (ldap_attr) ldap_data->list = g_list_prepend(ldap_data->list, g_strdup_printf("(%s=%s)", ldap_attr, str)); else { g_warning ("unknown query property\n"); /* we want something that'll always be false */ ldap_data->list = g_list_prepend(ldap_data->list, g_strdup("objectClass=MyBarnIsBiggerThanYourBarn")); } g_free (str); } r = e_sexp_result_new(f, ESEXP_RES_BOOL); r->value.bool = FALSE; return r; } static ESExpResult * func_beginswith(struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { PASBackendLDAPSExpData *ldap_data = data; ESExpResult *r; if (argc == 2 && argv[0]->type == ESEXP_RES_STRING && argv[1]->type == ESEXP_RES_STRING) { char *propname = argv[0]->value.string; char *str = rfc2254_escape(argv[1]->value.string); char *ldap_attr = query_prop_to_ldap(propname); /* insert hack for fileAs queries, since we need to do the right thing if the server supports them or not, and for entries that have no fileAs attribute. */ if (ldap_attr) { if (!strcmp (ldap_attr, "fileAs")) { ldap_data->list = g_list_prepend(ldap_data->list, g_strdup_printf( "(|(fileAs=%s*)(&(!(fileAs=*))(sn=%s*)))", str, str)); } else { ldap_data->list = g_list_prepend(ldap_data->list, g_strdup_printf("(%s=%s*)", ldap_attr, str)); } } g_free (str); } r = e_sexp_result_new(f, ESEXP_RES_BOOL); r->value.bool = FALSE; return r; } static ESExpResult * func_endswith(struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { PASBackendLDAPSExpData *ldap_data = data; ESExpResult *r; if (argc == 2 && argv[0]->type == ESEXP_RES_STRING && argv[1]->type == ESEXP_RES_STRING) { char *propname = argv[0]->value.string; char *str = rfc2254_escape(argv[1]->value.string); char *ldap_attr = query_prop_to_ldap(propname); if (ldap_attr) ldap_data->list = g_list_prepend(ldap_data->list, g_strdup_printf("(%s=*%s)", ldap_attr, str)); g_free (str); } r = e_sexp_result_new(f, ESEXP_RES_BOOL); r->value.bool = FALSE; return r; } /* 'builtin' functions */ static struct { char *name; ESExpFunc *func; int type; /* set to 1 if a function can perform shortcut evaluation, or doesn't execute everything, 0 otherwise */ } symbols[] = { { "and", func_and, 0 }, { "or", func_or, 0 }, { "not", func_not, 0 }, { "contains", func_contains, 0 }, { "is", func_is, 0 }, { "beginswith", func_beginswith, 0 }, { "endswith", func_endswith, 0 }, }; static gchar * pas_backend_ldap_build_query (PASBackendLDAP *bl, gchar *query) { ESExp *sexp; ESExpResult *r; gchar *retval; PASBackendLDAPSExpData data; int i; data.list = NULL; data.bl = bl; sexp = e_sexp_new(); for(i=0;inext) { g_warning ("conversion to ldap query string failed"); retval = NULL; g_list_foreach (data.list, (GFunc)g_free, NULL); } else { retval = data.list->data; } } else { g_warning ("conversion to ldap query string failed"); retval = NULL; } g_list_free (data.list); return retval; } static gchar * query_prop_to_ldap(gchar *query_prop) { int i; for (i = 0; i < num_prop_infos; i ++) if (!strcmp (query_prop, prop_info[i].query_prop)) return prop_info[i].ldap_attr; return NULL; } typedef struct { LDAPOp op; char *ldap_query; PASBackendLDAP *bl; PASBackendLDAPBookView *view; } LDAPSearchOp; static ECardSimple * build_card_from_entry (LDAP *ldap, LDAPMessage *e, GList **existing_objectclasses) { ECard *ecard = E_CARD(gtk_type_new(e_card_get_type())); ECardSimple *card = e_card_simple_new (ecard); char *dn = ldap_get_dn(ldap, e); char *attr; BerElement *ber = NULL; // g_print ("build_card_from_entry, dn = %s\n", dn); e_card_simple_set_id (card, dn); for (attr = ldap_first_attribute (ldap, e, &ber); attr; attr = ldap_next_attribute (ldap, e, ber)) { int i; struct prop_info *info = NULL; char **values; if (existing_objectclasses && !strcasecmp (attr, "objectclass")) { values = ldap_get_values (ldap, e, attr); for (i = 0; values[i]; i ++) *existing_objectclasses = g_list_append (*existing_objectclasses, g_strdup (values[i])); ldap_value_free (values); } else { for (i = 0; i < num_prop_infos; i ++) if (!g_strcasecmp (attr, prop_info[i].ldap_attr)) info = &prop_info[i]; if (info) { values = ldap_get_values (ldap, e, attr); if (values) { if (info->prop_type & PROP_TYPE_STRING) { /* if it's a normal property just set the string */ e_card_simple_set (card, info->field_id, values[0]); } else if (info->prop_type & PROP_TYPE_COMPLEX) { /* if it's a list call the ecard-populate function, which calls gtk_object_set to set the property */ info->populate_ecard_func(card, values); } ldap_value_free (values); } } } } #ifndef OPENLDAP2 /* if ldap->ld_errno == LDAP_DECODING_ERROR there was an error decoding an attribute, and we shouldn't free ber, since the ldap library already did it. */ if (ldap->ld_errno != LDAP_DECODING_ERROR && ber) ber_free (ber, 0); #endif e_card_simple_sync_card (card); gtk_object_unref (GTK_OBJECT (ecard)); return card; } static void send_pending_adds (PASBackendLDAPBookView *view) { view->num_sent_this_time += view->num_pending_adds; pas_book_view_notify_add (view->book_view, view->pending_adds); g_list_foreach (view->pending_adds, (GFunc)g_free, NULL); view->pending_adds = NULL; view->num_pending_adds = 0; } static gboolean poll_ldap (LDAPSearchOp *op) { PASBackendLDAPBookView *view = op->view; PASBackendLDAP *bl = op->bl; LDAP *ldap = bl->priv->ldap; int rc; LDAPMessage *res, *e; static int received = 0; GTimeVal cur_time; glong cur_millis; struct timeval timeout; timeout.tv_sec = 0; timeout.tv_usec = LDAP_RESULT_TIMEOUT_MILLIS * 1000; if (!view->notified_receiving_results) { view->notified_receiving_results = TRUE; pas_book_view_notify_status_message (view->book_view, _("Receiving LDAP search results...")); } rc = ldap_result (ldap, view->search_msgid, 0, &timeout, &res); if (rc != 0) {/* rc == 0 means timeout exceeded */ if (rc == -1 && received == 0) { pas_book_view_notify_status_message (view->book_view, _("Restarting search.")); /* connection went down and we never got any. */ bl->priv->connected = FALSE; /* this will reopen the connection */ ldap_op_restart ((LDAPOp*)op); return FALSE; } if (rc != LDAP_RES_SEARCH_ENTRY) { view->search_timeout = 0; if (view->num_pending_adds) send_pending_adds (view); pas_book_view_notify_complete (view->book_view); ldap_op_finished ((LDAPOp*)op); received = 0; return FALSE; } received = 1; e = ldap_first_entry(ldap, res); while (NULL != e) { ECardSimple *card = build_card_from_entry (ldap, e, NULL); view->pending_adds = g_list_append (view->pending_adds, e_card_simple_get_vcard_assume_utf8 (card)); view->num_pending_adds ++; gtk_object_unref (GTK_OBJECT(card)); e = ldap_next_entry(ldap, e); } ldap_msgfree(res); } g_get_current_time (&cur_time); cur_millis = TV_TO_MILLIS (cur_time); if (cur_millis - view->grouping_time_start > GROUPING_MINIMUM_WAIT) { if (view->num_pending_adds >= view->target_pending_adds) send_pending_adds (view); if (cur_millis - view->grouping_time_start > GROUPING_MAXIMUM_WAIT) { GTimeVal new_start; if (view->num_pending_adds) send_pending_adds (view); view->target_pending_adds = MIN (GROUPING_MAXIMUM_SIZE, (view->num_sent_this_time + view->num_sent_last_time) / 2); view->target_pending_adds = MAX (view->target_pending_adds, 1); #ifdef PERFORMANCE_SPEW printf ("num sent this time %d, last time %d, target pending adds set to %d\n", view->num_sent_this_time, view->num_sent_last_time, view->target_pending_adds); #endif g_get_current_time (&new_start); view->grouping_time_start = TV_TO_MILLIS (new_start); view->num_sent_last_time = view->num_sent_this_time; view->num_sent_this_time = 0; } } return TRUE; } static gboolean ldap_search_handler (PASBackend *backend, LDAPOp *op) { LDAPSearchOp *search_op = (LDAPSearchOp*) op; if (op->view) pas_book_view_notify_status_message (op->view, _("Searching...")); /* it might not be NULL if we've been restarted */ if (search_op->ldap_query == NULL) search_op->ldap_query = pas_backend_ldap_build_query(PAS_BACKEND_LDAP (backend), search_op->view->search); if (search_op->ldap_query != NULL) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); PASBackendLDAPBookView *view = search_op->view; LDAP *ldap = bl->priv->ldap; int ldap_err; GTimeVal search_start; view->pending_adds = NULL; view->num_pending_adds = 0; view->target_pending_adds = GROUPING_INITIAL_SIZE; g_get_current_time (&search_start); view->grouping_time_start = TV_TO_MILLIS (search_start); view->num_sent_last_time = 0; view->num_sent_this_time = 0; view->notified_receiving_results = FALSE; ldap_err = ldap_search_ext (ldap, bl->priv->ldap_rootdn, bl->priv->ldap_scope, search_op->ldap_query, NULL, 0, NULL, /* XXX */ NULL, /* XXX */ NULL, bl->priv->ldap_limit, &view->search_msgid); if (ldap_err != LDAP_SUCCESS) { pas_book_view_notify_status_message (view->book_view, ldap_err2string(ldap_err)); return TRUE; /* act synchronous in this case */ } if (view->search_msgid == -1) { pas_book_view_notify_status_message (view->book_view, ldap_err2string(ldap_err)); return TRUE; /* act synchronous in this case */ } else { view->search_timeout = g_timeout_add (LDAP_POLL_INTERVAL, (GSourceFunc) poll_ldap, search_op); } /* we're async */ return FALSE; } else { /* error doing the conversion to an ldap query, let's end this now by acting like we're synchronous. */ g_warning ("LDAP problem converting search query %s\n", search_op->view->search); return TRUE; } } static void ldap_search_dtor (PASBackend *backend, LDAPOp *op) { LDAPSearchOp *search_op = (LDAPSearchOp*) op; if (op->view) bonobo_object_release_unref(bonobo_object_corba_objref(BONOBO_OBJECT(op->view)), NULL); g_free (search_op->ldap_query); g_free (search_op); } static void pas_backend_ldap_search (PASBackendLDAP *bl, PASBook *book, PASBackendLDAPBookView *view) { LDAPSearchOp *op = g_new (LDAPSearchOp, 1); bonobo_object_dup_ref(bonobo_object_corba_objref(BONOBO_OBJECT(view->book_view)), NULL); ldap_op_init ((LDAPOp*)op, PAS_BACKEND(bl), book, view->book_view, ldap_search_handler, ldap_search_dtor); op->ldap_query = NULL; op->view = view; op->bl = bl; /* keep track of the search op so we can delete it from the list if the view is destroyed */ view->search_op = (LDAPOp*)op; ldap_op_process ((LDAPOp*)op); } static void pas_backend_ldap_process_get_book_view (PASBackend *backend, PASBook *book, PASRequest *req) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); CORBA_Environment ev; PASBookView *book_view; PASBackendLDAPBookView *view; g_return_if_fail (req->listener != NULL); book_view = pas_book_view_new (req->listener); bonobo_object_ref(BONOBO_OBJECT(book)); gtk_signal_connect(GTK_OBJECT(book_view), "destroy", GTK_SIGNAL_FUNC(view_destroy), book); view = g_new0(PASBackendLDAPBookView, 1); view->book_view = book_view; view->search = g_strdup(req->search); view->card_sexp = pas_backend_card_sexp_new (view->search); view->blpriv = bl->priv; bl->priv->book_views = g_list_prepend(bl->priv->book_views, view); pas_book_respond_get_book_view (book, (book_view != NULL ? GNOME_Evolution_Addressbook_BookListener_Success : GNOME_Evolution_Addressbook_BookListener_CardNotFound /* XXX */), book_view); pas_backend_ldap_search (bl, book, view); g_free (req->search); CORBA_exception_init(&ev); bonobo_object_unref (BONOBO_OBJECT (book_view)); bonobo_object_release_unref(req->listener, &ev); if (ev._major != CORBA_NO_EXCEPTION) { g_warning("pas_backend_file_process_get_book_view: Exception reffing " "corba book.\n"); } CORBA_exception_free(&ev); } static void pas_backend_ldap_process_check_connection (PASBackend *backend, PASBook *book, PASRequest *req) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); pas_book_report_connection (book, bl->priv->connected); } static void pas_backend_ldap_process_authenticate_user (PASBackend *backend, PASBook *book, PASRequest *req) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); int ldap_error; char *dn = NULL; if (!strcmp (req->auth_method, "ldap/simple-email")) { LDAPMessage *res, *e; char *query = g_strdup_printf ("(mail=%s)", req->user); ldap_error = ldap_search_s (bl->priv->ldap, bl->priv->ldap_rootdn, bl->priv->ldap_scope, query, NULL, 0, &res); g_free (query); if (ldap_error == LDAP_SUCCESS) { e = ldap_first_entry (bl->priv->ldap, res); dn = g_strdup(ldap_get_dn (bl->priv->ldap, e)); ldap_msgfree (res); } else { pas_book_respond_authenticate_user (book, GNOME_Evolution_Addressbook_BookListener_PermissionDenied); return; } } else { dn = g_strdup (req->user); } /* now authenticate against the DN we were either supplied or queried for */ printf ("authenticating as %s\n", dn); ldap_error = ldap_simple_bind_s(bl->priv->ldap, dn, req->passwd); g_free (dn); pas_book_respond_authenticate_user (book, ldap_error_to_response (ldap_error)); bl->priv->writable = (ldap_error == LDAP_SUCCESS); if (!bl->priv->evolutionPersonChecked) check_schema_support (bl); pas_book_report_writable (book, bl->priv->writable); } static void pas_backend_ldap_process_get_supported_fields (PASBackend *backend, PASBook *book, PASRequest *req) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); pas_book_respond_get_supported_fields (book, GNOME_Evolution_Addressbook_BookListener_Success, bl->priv->supported_fields); } static void pas_backend_ldap_process_client_requests (PASBook *book) { PASBackend *backend; PASRequest *req; backend = pas_book_get_backend (book); req = pas_book_pop_request (book); if (req == NULL) return; switch (req->op) { case CreateCard: pas_backend_ldap_process_create_card (backend, book, req); break; case RemoveCard: pas_backend_ldap_process_remove_card (backend, book, req); break; case ModifyCard: pas_backend_ldap_process_modify_card (backend, book, req); break; case CheckConnection: pas_backend_ldap_process_check_connection (backend, book, req); break; case GetVCard: pas_backend_ldap_process_get_vcard (backend, book, req); break; case GetCursor: pas_backend_ldap_process_get_cursor (backend, book, req); break; case GetBookView: pas_backend_ldap_process_get_book_view (backend, book, req); break; case GetChanges: /* FIXME: Code this. */ break; case AuthenticateUser: pas_backend_ldap_process_authenticate_user (backend, book, req); break; case GetSupportedFields: pas_backend_ldap_process_get_supported_fields (backend, book, req); break; } g_free (req); } static void pas_backend_ldap_book_destroy_cb (PASBook *book, gpointer data) { PASBackendLDAP *backend; backend = PAS_BACKEND_LDAP (data); pas_backend_remove_client (PAS_BACKEND (backend), book); } static GNOME_Evolution_Addressbook_BookListener_CallStatus pas_backend_ldap_load_uri (PASBackend *backend, const char *uri) { PASBackendLDAP *bl = PAS_BACKEND_LDAP (backend); LDAPURLDesc *lud; int ldap_error; char **attributes; int i; int limit = 100; g_assert (bl->priv->connected == FALSE); attributes = g_strsplit (uri, ";", 0); if (attributes[0] == NULL) return FALSE; for (i = 1; attributes[i]; i++) { char *equals; char *value; int key_length; equals = strchr (attributes[i], '='); if (equals) { key_length = equals - attributes[i]; value = equals + 1; } else { key_length = strlen (attributes[i]); value = NULL; } if (key_length == strlen("limit") && !strncmp (attributes[i], "limit", key_length)) { if (value) limit = atoi(value); } } ldap_error = ldap_url_parse ((char*)attributes[0], &lud); g_strfreev (attributes); if (ldap_error == LDAP_SUCCESS) { g_free(bl->priv->uri); bl->priv->uri = g_strdup (uri); bl->priv->ldap_host = g_strdup(lud->lud_host); bl->priv->ldap_port = lud->lud_port; /* if a port wasn't specified, default to LDAP_PORT */ if (bl->priv->ldap_port == 0) bl->priv->ldap_port = LDAP_PORT; bl->priv->ldap_rootdn = g_strdup(lud->lud_dn); bl->priv->ldap_limit = limit; bl->priv->ldap_scope = lud->lud_scope; ldap_free_urldesc(lud); pas_backend_ldap_connect (bl); if (bl->priv->ldap == NULL) return GNOME_Evolution_Addressbook_BookListener_RepositoryOffline; else return GNOME_Evolution_Addressbook_BookListener_Success; } else return GNOME_Evolution_Addressbook_BookListener_OtherError; } /* Get_uri handler for the addressbook LDAP backend */ static const char * pas_backend_ldap_get_uri (PASBackend *backend) { PASBackendLDAP *bl; bl = PAS_BACKEND_LDAP (backend); return bl->priv->uri; } static gboolean pas_backend_ldap_add_client (PASBackend *backend, GNOME_Evolution_Addressbook_BookListener listener) { PASBackendLDAP *bl; PASBook *book; g_assert (backend != NULL); g_assert (PAS_IS_BACKEND_LDAP (backend)); bl = PAS_BACKEND_LDAP (backend); book = pas_book_new (backend, listener); if (!book) { if (!bl->priv->clients) pas_backend_last_client_gone (backend); return FALSE; } gtk_signal_connect (GTK_OBJECT (book), "destroy", pas_backend_ldap_book_destroy_cb, backend); gtk_signal_connect (GTK_OBJECT (book), "requests_queued", pas_backend_ldap_process_client_requests, NULL); bl->priv->clients = g_list_prepend ( bl->priv->clients, book); if (bl->priv->connected) { pas_book_respond_open ( book, GNOME_Evolution_Addressbook_BookListener_Success); } else { pas_book_respond_open ( book, GNOME_Evolution_Addressbook_BookListener_OtherError); } pas_book_report_writable (book, bl->priv->writable); bonobo_object_unref (BONOBO_OBJECT (book)); return TRUE; } static void pas_backend_ldap_remove_client (PASBackend *backend, PASBook *book) { PASBackendLDAP *bl; GList *l; PASBook *lbook; g_return_if_fail (backend != NULL); g_return_if_fail (PAS_IS_BACKEND_LDAP (backend)); g_return_if_fail (book != NULL); g_return_if_fail (PAS_IS_BOOK (book)); bl = PAS_BACKEND_LDAP (backend); /* Find the book in the list of clients */ for (l = bl->priv->clients; l; l = l->next) { lbook = PAS_BOOK (l->data); if (lbook == book) break; } g_assert (l != NULL); /* Disconnect */ bl->priv->clients = g_list_remove_link (bl->priv->clients, l); g_list_free_1 (l); /* When all clients go away, notify the parent factory about it so that * it may decide whether to kill the backend or not. */ if (!bl->priv->clients) pas_backend_last_client_gone (backend); } static char * pas_backend_ldap_get_static_capabilites (PASBackend *backend) { return g_strdup("net"); } static gboolean pas_backend_ldap_construct (PASBackendLDAP *backend) { g_assert (backend != NULL); g_assert (PAS_IS_BACKEND_LDAP (backend)); if (! pas_backend_construct (PAS_BACKEND (backend))) return FALSE; return TRUE; } /** * pas_backend_ldap_new: */ PASBackend * pas_backend_ldap_new (void) { PASBackendLDAP *backend; backend = gtk_type_new (pas_backend_ldap_get_type ()); if (! pas_backend_ldap_construct (backend)) { gtk_object_unref (GTK_OBJECT (backend)); return NULL; } return PAS_BACKEND (backend); } static void call_dtor (LDAPOp *op, gpointer data) { op->dtor (op->backend, op); } static void pas_backend_ldap_destroy (GtkObject *object) { PASBackendLDAP *bl; bl = PAS_BACKEND_LDAP (object); g_list_foreach (bl->priv->pending_ops, (GFunc)call_dtor, NULL); g_list_free (bl->priv->pending_ops); if (bl->priv->supported_fields) gtk_object_unref (GTK_OBJECT (bl->priv->supported_fields)); g_free (bl->priv->uri); GTK_OBJECT_CLASS (pas_backend_ldap_parent_class)->destroy (object); } static void pas_backend_ldap_class_init (PASBackendLDAPClass *klass) { GtkObjectClass *object_class = (GtkObjectClass *) klass; PASBackendClass *parent_class; pas_backend_ldap_parent_class = gtk_type_class (pas_backend_get_type ()); parent_class = PAS_BACKEND_CLASS (klass); /* Set the virtual methods. */ parent_class->load_uri = pas_backend_ldap_load_uri; parent_class->get_uri = pas_backend_ldap_get_uri; parent_class->add_client = pas_backend_ldap_add_client; parent_class->remove_client = pas_backend_ldap_remove_client; parent_class->get_static_capabilities = pas_backend_ldap_get_static_capabilites; object_class->destroy = pas_backend_ldap_destroy; } static void pas_backend_ldap_init (PASBackendLDAP *backend) { PASBackendLDAPPrivate *priv; priv = g_new0 (PASBackendLDAPPrivate, 1); priv->supported_fields = e_list_new ((EListCopyFunc)g_strdup, (EListFreeFunc)g_free, NULL); priv->ldap_limit = 100; backend->priv = priv; } /** * pas_backend_ldap_get_type: */ GtkType pas_backend_ldap_get_type (void) { static GtkType type = 0; if (! type) { GtkTypeInfo info = { "PASBackendLDAP", sizeof (PASBackendLDAP), sizeof (PASBackendLDAPClass), (GtkClassInitFunc) pas_backend_ldap_class_init, (GtkObjectInitFunc) pas_backend_ldap_init, NULL, /* reserved 1 */ NULL, /* reserved 2 */ (GtkClassInitFunc) NULL }; type = gtk_type_unique (pas_backend_get_type (), &info); } return type; }