mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-08 12:22:34 +00:00
2655 lines
71 KiB
C
2655 lines
71 KiB
C
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public
|
|
* License Version 1.1 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
|
* implied. See the License for the specific language governing
|
|
* rights and limitations under the License.
|
|
*
|
|
* The Original Code is mozilla.org code.
|
|
*
|
|
* The Initial Developer of the Original Code is Netscape
|
|
* Communications Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All
|
|
* Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*/
|
|
|
|
/* dirprefs.c -- directory server preferences */
|
|
#include "rosetta.h"
|
|
#include "xp.h"
|
|
|
|
#include "xp_mcom.h"
|
|
#include "xpgetstr.h"
|
|
#include "dirprefs.h"
|
|
#include "prefapi.h"
|
|
#include "libi18n.h"
|
|
#ifdef MOZ_LDAP
|
|
#include "ldap.h"
|
|
#else
|
|
#define LDAP_PORT 389
|
|
#define LDAPS_PORT 636
|
|
#endif
|
|
|
|
#ifndef MOZ_MAIL_NEWS
|
|
|
|
int DIR_GetLdapServers(XP_List *wholeList, XP_List *subList)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
const char **DIR_GetAttributeStrings (DIR_Server *server, DIR_AttributeId id)
|
|
{
|
|
return 0 ;
|
|
}
|
|
|
|
const char *DIR_GetFirstAttributeString (DIR_Server *server, DIR_AttributeId id)
|
|
{
|
|
return 0 ;
|
|
}
|
|
|
|
#else
|
|
|
|
#ifndef MOZADDRSTANDALONE
|
|
|
|
extern int MK_OUT_OF_MEMORY;
|
|
extern int MK_ADDR_PAB;
|
|
|
|
extern int MK_LDAP_COMMON_NAME;
|
|
extern int MK_LDAP_GIVEN_NAME;
|
|
extern int MK_LDAP_SURNAME;
|
|
extern int MK_LDAP_EMAIL_ADDRESS;
|
|
extern int MK_LDAP_PHONE_NUMBER;
|
|
extern int MK_LDAP_ORGANIZATION;
|
|
extern int MK_LDAP_ORG_UNIT;
|
|
extern int MK_LDAP_LOCALITY;
|
|
extern int MK_LDAP_STREET;
|
|
extern int MK_LDAP_CUSTOM1;
|
|
extern int MK_LDAP_CUSTOM2;
|
|
extern int MK_LDAP_CUSTOM3;
|
|
extern int MK_LDAP_CUSTOM4;
|
|
extern int MK_LDAP_CUSTOM5;
|
|
extern int MK_LDAP_DESCRIPTION;
|
|
extern int MK_LDAP_EMPLOYEE_TYPE;
|
|
extern int MK_LDAP_FAX_NUMBER;
|
|
extern int MK_LDAP_MANAGER;
|
|
extern int MK_LDAP_OBJECT_CLASS;
|
|
extern int MK_LDAP_POSTAL_ADDRESS;
|
|
extern int MK_LDAP_POSTAL_CODE;
|
|
extern int MK_LDAP_SECRETARY;
|
|
extern int MK_LDAP_TITLE;
|
|
extern int MK_LDAP_CAR_LICENSE;
|
|
extern int MK_LDAP_BUSINESS_CAT;
|
|
extern int MK_LDAP_DEPT_NUMBER;
|
|
extern int MK_LDAP_REPL_CANT_SYNC_REPLICA;
|
|
|
|
#else
|
|
|
|
#define MK_OUT_OF_MEMORY -1;
|
|
|
|
#endif /* !MOZADDRSTANDALONE */
|
|
|
|
/*****************************************************************************
|
|
* Private structs and stuff
|
|
*/
|
|
|
|
/* DIR_Server.flags */
|
|
#define DIR_AUTO_COMPLETE_ENABLED 0x00000001
|
|
#define DIR_ENABLE_AUTH 0x00000002
|
|
#define DIR_SAVE_PASSWORD 0x00000004
|
|
#define DIR_DEFEAT_UTF8 0x00000008
|
|
#define DIR_IS_SECURE 0x00000010
|
|
#define DIR_SAVE_RESULTS 0x00000020
|
|
#define DIR_EFFICIENT_WILDCARDS 0x00000040
|
|
|
|
/* DIR_Server.customAttributes is a list of DIR_Attribute structures */
|
|
typedef struct DIR_Attribute
|
|
{
|
|
DIR_AttributeId id;
|
|
char *prettyName;
|
|
char **attrNames;
|
|
} DIR_Attribute;
|
|
|
|
/* Our internal view of a default attribute is a resourceId for the pretty
|
|
* name and the real attribute name. These are catenated to create a string
|
|
* of the form "Pretty Name:attrname"
|
|
*/
|
|
typedef struct DIR_DefaultAttribute
|
|
{
|
|
DIR_AttributeId id;
|
|
int resourceId;
|
|
char *name;
|
|
} DIR_DefaultAttribute;
|
|
|
|
/* DIR_Filter.flags */
|
|
#define DIR_F_SUBST_STARS_FOR_SPACES 0x00000001
|
|
#define DIR_F_REPEAT_FILTER_FOR_TOKENS 0x00000002
|
|
|
|
/* DIR_Server.filters is a list of DIR_Filter structures */
|
|
typedef struct DIR_Filter
|
|
{
|
|
char *string;
|
|
uint32 flags;
|
|
} DIR_Filter;
|
|
|
|
/* Default settings for site-configurable prefs */
|
|
#define kDefaultTokenSeps " ,."
|
|
#define kDefaultSubstStarsForSpaces TRUE
|
|
#define kDefaultRepeatFilterForTokens TRUE
|
|
#define kDefaultEfficientWildcards TRUE
|
|
#define kDefaultFilter "(cn=*%s*)"
|
|
#define kDefaultEfficientFilter "(|(givenname=%s)(sn=%s))"
|
|
#define kDefaultStopOnHit TRUE
|
|
#define kDefaultMaxHits 100
|
|
#define kDefaultIsOffline TRUE
|
|
#define kDefaultEnableAuth FALSE
|
|
#define kDefaultSavePassword FALSE
|
|
#define kDefaultUtf8Disabled FALSE
|
|
#define kDefaultLdapPublicDirectory FALSE
|
|
|
|
#define kDefaultAutoCompleteEnabled FALSE
|
|
#define kDefaultAutoCompleteStyle acsGivenAndSurname
|
|
|
|
#define kDefaultReplicaEnabled FALSE
|
|
#define kDefaultReplicaFileName NULL
|
|
#define kDefaultReplicaDataVersion NULL
|
|
#define kDefaultReplicaDescription NULL
|
|
#define kDefaultReplicaChangeNumber -1
|
|
#define kDefaultReplicaFilter "(objectclass=*)"
|
|
#define kDefaultReplicaExcludedAttributes NULL
|
|
|
|
#define kLdapServersPrefName "network.hosts.ldap_servers"
|
|
|
|
const char *DIR_GetAttributeName (DIR_Server *server, DIR_AttributeId id);
|
|
static DIR_DefaultAttribute *DIR_GetDefaultAttribute (DIR_AttributeId id);
|
|
void DIR_SetFileName(char** filename, const char* leafName);
|
|
|
|
|
|
/*****************************************************************************
|
|
* Functions for creating the new back end managed DIR_Server list.
|
|
*/
|
|
|
|
static XP_List * dir_ServerList = NULL;
|
|
|
|
XP_List * DIR_GetDirServers()
|
|
{
|
|
const char * msg = "abook.nab";
|
|
if (!dir_ServerList)
|
|
{
|
|
/* we need to build the DIR_Server list */
|
|
dir_ServerList = XP_ListNew();
|
|
|
|
PREF_SetDefaultCharPref("browser.addressbook_location",msg);
|
|
|
|
DIR_GetServerPreferences (&dir_ServerList, msg);
|
|
/* char * ldapPref = PR_smprintf("ldap_%d.end_of_directories", kCurrentListVersion);
|
|
if (ldapPref)
|
|
PREF_RegisterCallback(ldapPref, DirServerListChanged, NULL);
|
|
XP_FREEIF (ldapPref); */
|
|
}
|
|
return dir_ServerList;
|
|
}
|
|
|
|
int DIR_ShutDown() /* FEs should call this when the app is shutting down. It frees all DIR_Servers regardless of ref count values! */
|
|
{
|
|
int i = 1;
|
|
if (dir_ServerList)
|
|
{
|
|
for (i = 1; i <= XP_ListCount(dir_ServerList); i++)
|
|
DIR_DeleteServer(XP_ListGetObjectNum (dir_ServerList, i));
|
|
XP_ListDestroy (dir_ServerList);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DIR_DecrementServerRefCount (DIR_Server *server)
|
|
{
|
|
XP_ASSERT(server);
|
|
if (server && --server->refCount <= 0)
|
|
return DIR_DeleteServer(server);
|
|
return 0;
|
|
}
|
|
|
|
int DIR_IncrementServerRefCount (DIR_Server *server)
|
|
{
|
|
XP_ASSERT(server);
|
|
if (server)
|
|
server->refCount++;
|
|
return 0;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Functions for creating DIR_Servers
|
|
*/
|
|
|
|
int DIR_InitServer (DIR_Server *server)
|
|
{
|
|
XP_ASSERT(server);
|
|
if (server)
|
|
{
|
|
XP_BZERO(server, sizeof(DIR_Server));
|
|
server->saveResults = TRUE;
|
|
server->efficientWildcards = kDefaultEfficientWildcards;
|
|
server->port = LDAP_PORT;
|
|
server->maxHits = kDefaultMaxHits;
|
|
server->isOffline = kDefaultIsOffline;
|
|
server->refCount = 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* Functions for cloning DIR_Servers
|
|
*/
|
|
|
|
static DIR_Attribute *DIR_CopyAttribute (DIR_Attribute *inAttribute)
|
|
{
|
|
DIR_Attribute *outAttribute = (DIR_Attribute*) XP_ALLOC(sizeof(DIR_Attribute));
|
|
if (outAttribute)
|
|
{
|
|
int count = 0;
|
|
outAttribute->id = inAttribute->id;
|
|
outAttribute->prettyName = XP_STRDUP(inAttribute->prettyName);
|
|
while (inAttribute->attrNames[count])
|
|
count++;
|
|
outAttribute->attrNames = (char**) XP_ALLOC((count + 1) * sizeof(char*));
|
|
if (outAttribute->attrNames)
|
|
{
|
|
int i;
|
|
for (i = 0; i < count; i++)
|
|
outAttribute->attrNames[i] = XP_STRDUP(inAttribute->attrNames[i]);
|
|
outAttribute->attrNames[i] = NULL;
|
|
}
|
|
}
|
|
return outAttribute;
|
|
}
|
|
|
|
|
|
static DIR_Filter *DIR_CopyFilter (DIR_Filter *inFilter)
|
|
{
|
|
DIR_Filter *outFilter = (DIR_Filter*) XP_ALLOC(sizeof(DIR_Filter));
|
|
if (outFilter)
|
|
{
|
|
outFilter->flags = inFilter->flags;
|
|
outFilter->string = XP_STRDUP(inFilter->string);
|
|
}
|
|
return outFilter;
|
|
}
|
|
|
|
|
|
static int dir_CopyTokenList (char **inList, int inCount, char ***outList, int *outCount)
|
|
{
|
|
int status = 0;
|
|
if (0 != inCount && NULL != inList)
|
|
{
|
|
*outList = (char**) XP_ALLOC(inCount * sizeof(char*));
|
|
if (*outList)
|
|
{
|
|
int i;
|
|
for (i = 0; i < inCount; i++)
|
|
(*outList)[i] = XP_STRDUP (inList[i]);
|
|
*outCount = inCount;
|
|
}
|
|
else
|
|
status = MK_OUT_OF_MEMORY;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
|
|
static DIR_ReplicationInfo *dir_CopyReplicationInfo (DIR_ReplicationInfo *inInfo)
|
|
{
|
|
DIR_ReplicationInfo *outInfo = (DIR_ReplicationInfo*) XP_CALLOC (1, sizeof(DIR_ReplicationInfo));
|
|
if (outInfo)
|
|
{
|
|
outInfo->lastChangeNumber = inInfo->lastChangeNumber;
|
|
if (inInfo->description)
|
|
outInfo->description = XP_STRDUP (inInfo->description);
|
|
if (inInfo->fileName)
|
|
outInfo->fileName = XP_STRDUP (inInfo->fileName);
|
|
if (inInfo->dataVersion)
|
|
outInfo->dataVersion = XP_STRDUP (inInfo->dataVersion);
|
|
if (inInfo->filter)
|
|
outInfo->filter = XP_STRDUP (inInfo->filter);
|
|
dir_CopyTokenList (inInfo->excludedAttributes, inInfo->excludedAttributesCount,
|
|
&outInfo->excludedAttributes, &outInfo->excludedAttributesCount);
|
|
}
|
|
return outInfo;
|
|
}
|
|
|
|
int DIR_CopyServer (DIR_Server *in, DIR_Server **out)
|
|
{
|
|
int err = 0;
|
|
if (in) {
|
|
*out = XP_ALLOC(sizeof(DIR_Server));
|
|
if (*out)
|
|
{
|
|
XP_BZERO (*out, sizeof(DIR_Server));
|
|
|
|
if (in->description)
|
|
{
|
|
(*out)->description = XP_STRDUP(in->description);
|
|
if (!(*out)->description)
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (in->serverName)
|
|
{
|
|
(*out)->serverName = XP_STRDUP(in->serverName);
|
|
if (!(*out)->serverName)
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (in->searchBase)
|
|
{
|
|
(*out)->searchBase = XP_STRDUP(in->searchBase);
|
|
if (!(*out)->searchBase)
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (in->fileName)
|
|
{
|
|
(*out)->fileName = XP_STRDUP(in->fileName);
|
|
if (!(*out)->fileName)
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (in->prefId)
|
|
{
|
|
(*out)->prefId = XP_STRDUP(in->prefId);
|
|
if (!(*out)->prefId)
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
(*out)->port = in->port;
|
|
(*out)->maxHits = in->maxHits;
|
|
(*out)->isSecure = in->isSecure;
|
|
(*out)->saveResults = in->saveResults;
|
|
(*out)->isOffline = in->isOffline;
|
|
(*out)->efficientWildcards = in->efficientWildcards;
|
|
(*out)->dirType = in->dirType;
|
|
|
|
(*out)->flags = in->flags;
|
|
|
|
(*out)->enableAuth = in->enableAuth;
|
|
(*out)->savePassword = in->savePassword;
|
|
if (in->authDn)
|
|
{
|
|
(*out)->authDn = XP_STRDUP (in->authDn);
|
|
if (!(*out)->authDn)
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
if (in->password)
|
|
{
|
|
(*out)->password = XP_STRDUP (in->password);
|
|
if (!(*out)->password)
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (in->customAttributes)
|
|
{
|
|
(*out)->customAttributes = XP_ListNew();
|
|
if ((*out)->customAttributes)
|
|
{
|
|
XP_List *list = in->customAttributes;
|
|
DIR_Attribute *attribute = NULL;
|
|
|
|
while ((attribute = XP_ListNextObject (list)) != NULL)
|
|
{
|
|
DIR_Attribute *outAttr = DIR_CopyAttribute (attribute);
|
|
if (outAttr)
|
|
XP_ListAddObject ((*out)->customAttributes, outAttr);
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (in->customFilters)
|
|
{
|
|
(*out)->customFilters = XP_ListNew();
|
|
if ((*out)->customFilters)
|
|
{
|
|
XP_List *list = in->customFilters;
|
|
DIR_Filter *filter = NULL;
|
|
|
|
while ((filter = XP_ListNextObject (list)) != NULL)
|
|
{
|
|
DIR_Filter *outFilter = DIR_CopyFilter (filter);
|
|
if (outFilter)
|
|
XP_ListAddObject ((*out)->customFilters, outFilter);
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (in->replInfo)
|
|
(*out)->replInfo = dir_CopyReplicationInfo (in->replInfo);
|
|
|
|
if (in->basicSearchAttributesCount > 0)
|
|
{
|
|
int bsaLength = in->basicSearchAttributesCount * sizeof(DIR_AttributeId);
|
|
(*out)->basicSearchAttributes = (DIR_AttributeId*) XP_ALLOC(bsaLength);
|
|
if ((*out)->basicSearchAttributes)
|
|
{
|
|
XP_MEMCPY ((*out)->basicSearchAttributes, in->basicSearchAttributes, bsaLength);
|
|
(*out)->basicSearchAttributesCount = in->basicSearchAttributesCount;
|
|
}
|
|
}
|
|
|
|
dir_CopyTokenList (in->dnAttributes, in->dnAttributesCount,
|
|
&(*out)->dnAttributes, &(*out)->dnAttributesCount);
|
|
|
|
dir_CopyTokenList (in->suppressedAttributes, in->suppressedAttributesCount,
|
|
&(*out)->suppressedAttributes, &(*out)->suppressedAttributesCount);
|
|
|
|
if (in->customDisplayUrl)
|
|
(*out)->customDisplayUrl = XP_STRDUP (in->customDisplayUrl);
|
|
if (in->searchPairList)
|
|
(*out)->searchPairList = XP_STRDUP (in->searchPairList);
|
|
|
|
(*out)->autoCompleteStyle = in->autoCompleteStyle;
|
|
|
|
(*out)->refCount = 1;
|
|
}
|
|
else {
|
|
err = MK_OUT_OF_MEMORY;
|
|
(*out) = NULL;
|
|
}
|
|
}
|
|
else {
|
|
XP_ASSERT (0);
|
|
err = -1;
|
|
(*out) = NULL;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Function for comparing DIR_Servers
|
|
*/
|
|
|
|
XP_Bool DIR_AreServersSame (DIR_Server *first, DIR_Server *second)
|
|
{
|
|
/* This function used to be written to assume that we only had one PAB so it
|
|
only checked the server type for PABs. If both were PABDirectories, then
|
|
it returned TRUE. Now that we support multiple address books, we need to
|
|
check type & file name for address books to test if they are the same */
|
|
|
|
XP_Bool result = FALSE;
|
|
|
|
if (first && second)
|
|
{
|
|
/* assume for right now one personal address book type where offline is false */
|
|
if ((first->dirType == PABDirectory) && (second->dirType == PABDirectory)) {
|
|
if ((first->isOffline == FALSE) && (second->isOffline == FALSE)) /* are they both really address books? */
|
|
{
|
|
XP_ASSERT(first->fileName && second->fileName);
|
|
if (first->fileName && second->fileName)
|
|
if (XP_STRCASECMP(first->fileName, second->fileName) == 0)
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
else {
|
|
XP_ASSERT (first->serverName && second->serverName);
|
|
if (first->serverName && second->serverName)
|
|
{
|
|
if (XP_STRCASECMP (first->serverName, second->serverName) == 0)
|
|
{
|
|
if (first->port == second->port)
|
|
{
|
|
/* allow for a null search base */
|
|
if (first->searchBase == NULL && second->searchBase == NULL)
|
|
return TRUE;
|
|
/* otherwise check the strings */
|
|
if (first->searchBase && second->searchBase)
|
|
{
|
|
if (XP_STRCASECMP (first->searchBase, second->searchBase) == 0)
|
|
{
|
|
return TRUE;
|
|
}
|
|
else
|
|
return FALSE;
|
|
}
|
|
else
|
|
return FALSE;
|
|
}
|
|
}
|
|
else
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (first->dirType != second->dirType)
|
|
return FALSE;
|
|
|
|
/* otherwise check for ldap servers */
|
|
XP_ASSERT (first->serverName && second->serverName);
|
|
if (first->serverName && second->serverName)
|
|
{
|
|
if (!XP_STRCASECMP (first->serverName, second->serverName))
|
|
{
|
|
if (first->port == second->port)
|
|
{
|
|
/* allow for a null search base */
|
|
if (first->searchBase == NULL && second->searchBase == NULL)
|
|
result = TRUE;
|
|
/* otherwise check the strings */
|
|
if (first->searchBase && second->searchBase)
|
|
{
|
|
if (!XP_STRCASECMP (first->searchBase, second->searchBase))
|
|
{
|
|
result = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* Functions for destroying DIR_Servers
|
|
*/
|
|
|
|
static void dir_DeleteTokenList (char **tokenList, int tokenListCount)
|
|
{
|
|
int tokenIdx;
|
|
for (tokenIdx = 0; tokenIdx < tokenListCount; tokenIdx++)
|
|
XP_FREE(tokenList[tokenIdx]);
|
|
XP_FREE(tokenList);
|
|
}
|
|
|
|
|
|
static int DIR_DeleteFilter (DIR_Filter *filter)
|
|
{
|
|
if (filter->string)
|
|
XP_FREE(filter->string);
|
|
XP_FREE(filter);
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int DIR_DeleteAttribute (DIR_Attribute *attribute)
|
|
{
|
|
int i = 0;
|
|
if (attribute->prettyName)
|
|
XP_FREE(attribute->prettyName);
|
|
if (attribute->attrNames)
|
|
{
|
|
while (attribute->attrNames[i])
|
|
XP_FREE((char**)attribute->attrNames[i++]);
|
|
XP_FREE(attribute->attrNames);
|
|
}
|
|
XP_FREE(attribute);
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void dir_DeleteReplicationInfo (DIR_Server *server)
|
|
{
|
|
DIR_ReplicationInfo *info = NULL;
|
|
if (server && (info = server->replInfo) != NULL)
|
|
{
|
|
dir_DeleteTokenList (info->excludedAttributes, info->excludedAttributesCount);
|
|
|
|
XP_FREEIF(info->description);
|
|
XP_FREEIF(info->fileName);
|
|
XP_FREEIF(info->dataVersion);
|
|
XP_FREEIF(info->filter);
|
|
XP_FREE(info);
|
|
}
|
|
}
|
|
|
|
/* When the back end manages the server list, deleting a server just decrements
|
|
* its ref count, in the old world, we actually delete the server
|
|
*/
|
|
int DIR_DeleteServer (DIR_Server *server)
|
|
{
|
|
if (server)
|
|
{
|
|
int i;
|
|
|
|
XP_FREEIF (server->description);
|
|
XP_FREEIF (server->serverName);
|
|
XP_FREEIF (server->searchBase);
|
|
XP_FREEIF (server->fileName);
|
|
XP_FREEIF (server->lastSearchString);
|
|
XP_FREEIF (server->tokenSeps);
|
|
XP_FREEIF (server->authDn);
|
|
XP_FREEIF (server->password);
|
|
XP_FREEIF (server->prefId);
|
|
|
|
if (server->customFilters)
|
|
{
|
|
for (i = 1; i <= XP_ListCount(server->customFilters); i++)
|
|
DIR_DeleteFilter ((DIR_Filter*) XP_ListGetObjectNum (server->customFilters, i));
|
|
XP_ListDestroy (server->customFilters);
|
|
}
|
|
|
|
if (server->customAttributes)
|
|
{
|
|
XP_List *list = server->customAttributes;
|
|
DIR_Attribute *walkAttrStruct = NULL;
|
|
while ((walkAttrStruct = XP_ListNextObject(list)) != NULL)
|
|
DIR_DeleteAttribute (walkAttrStruct);
|
|
XP_ListDestroy (server->customAttributes);
|
|
}
|
|
|
|
if (server->suppressedAttributes)
|
|
dir_DeleteTokenList (server->suppressedAttributes, server->suppressedAttributesCount);
|
|
if (server->dnAttributes)
|
|
dir_DeleteTokenList (server->dnAttributes, server->dnAttributesCount);
|
|
XP_FREEIF (server->basicSearchAttributes);
|
|
if (server->replInfo)
|
|
dir_DeleteReplicationInfo (server);
|
|
|
|
XP_FREEIF (server->customDisplayUrl);
|
|
XP_FREEIF (server->searchPairList);
|
|
|
|
XP_FREE (server);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int DIR_DeleteServerList(XP_List *wholeList)
|
|
{
|
|
int i;
|
|
if (wholeList)
|
|
{
|
|
for (i = 1; i <= XP_ListCount(wholeList); i++)
|
|
DIR_DeleteServer(XP_ListGetObjectNum (wholeList, i));
|
|
XP_ListDestroy (wholeList);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int DIR_CleanUpServerPreferences(XP_List *deletedList)
|
|
{
|
|
/* In the new world order of DIR_Servers it has been decreed that to clean
|
|
* up a server you should set its DIR_CLEAR_SERVER flag. Then, release
|
|
* your ref count on the list or servers (if you have one)
|
|
*/
|
|
XP_ASSERT(FALSE);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* Functions for retrieving subsets of the DIR_Server list
|
|
*/
|
|
|
|
static int DIR_GetHtmlServers(XP_List *wholeList, XP_List *subList)
|
|
{
|
|
int i;
|
|
if (wholeList && subList)
|
|
{
|
|
for (i = 1; i <= XP_ListCount(wholeList); i++)
|
|
{
|
|
DIR_Server *s = (DIR_Server*) XP_ListGetObjectNum (wholeList, i);
|
|
if (HTMLDirectory == s->dirType)
|
|
XP_ListAddObjectToEnd (subList, s);
|
|
}
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int DIR_GetPersonalAddressBooks (XP_List *wholeList, XP_List * subList)
|
|
{
|
|
int i;
|
|
if (wholeList && subList)
|
|
{
|
|
for (i = 1; i <= XP_ListCount(wholeList); i++)
|
|
{
|
|
DIR_Server *s = (DIR_Server*) XP_ListGetObjectNum (wholeList, i);
|
|
if (PABDirectory == s->dirType)
|
|
XP_ListAddObjectToEnd (subList, s);
|
|
}
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int DIR_GetLdapServers(XP_List *wholeList, XP_List *subList)
|
|
{
|
|
int i;
|
|
if (wholeList && subList)
|
|
{
|
|
for (i = 1; i <= XP_ListCount(wholeList); i++)
|
|
{
|
|
DIR_Server *s = (DIR_Server*) XP_ListGetObjectNum (wholeList, i);
|
|
if (LDAPDirectory == s->dirType)
|
|
XP_ListAddObjectToEnd (subList, s);
|
|
}
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int DIR_ReorderLdapServers(XP_List *wholeList)
|
|
{
|
|
int status = 0;
|
|
int length = 0;
|
|
DIR_Server *s = NULL;
|
|
char *prefValue = NULL;
|
|
|
|
XP_List *walkList = wholeList;
|
|
while (s = (DIR_Server*) XP_ListNextObject (walkList))
|
|
{
|
|
length += XP_STRLEN (s->prefId) + 2; /* +2 for ", " */
|
|
}
|
|
|
|
prefValue = (char*) XP_ALLOC (length + 1); /* +1 for null term */
|
|
if (prefValue)
|
|
{
|
|
*prefValue = '\0';
|
|
walkList = wholeList;
|
|
while (s = (DIR_Server*) XP_ListNextObject (walkList))
|
|
{
|
|
XP_STRCAT (prefValue, s->prefId);
|
|
XP_STRCAT (prefValue, ", ");
|
|
}
|
|
if (PREF_SetCharPref (kLdapServersPrefName, prefValue) < 0)
|
|
status = -1; /* not sure how to make a pref error into an mk error */
|
|
XP_FREE(prefValue);
|
|
}
|
|
else
|
|
status = MK_OUT_OF_MEMORY;
|
|
|
|
return status;
|
|
}
|
|
|
|
int DIR_GetPersonalAddressBook(XP_List *wholeList, DIR_Server **pab)
|
|
{
|
|
int i;
|
|
if (wholeList && pab)
|
|
{
|
|
*pab = NULL;
|
|
for (i = 1; i <= XP_ListCount(wholeList); i++)
|
|
{
|
|
DIR_Server *s = (DIR_Server*) XP_ListGetObjectNum (wholeList, i);
|
|
if ((PABDirectory == s->dirType) && (FALSE == s->isOffline))
|
|
{
|
|
if (s->serverName) {
|
|
if (XP_STRLEN (s->serverName) == 0) {
|
|
*pab = s;
|
|
return 0;
|
|
}
|
|
}
|
|
if (s->serverName == NULL) {
|
|
*pab = s;
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int DIR_GetComposeNameCompletionAddressBook(XP_List *wholeList, DIR_Server **cab)
|
|
{
|
|
int i;
|
|
if (wholeList && cab)
|
|
{
|
|
*cab = NULL;
|
|
for (i = 1; i <= XP_ListCount(wholeList); i++)
|
|
{
|
|
DIR_Server *s = (DIR_Server*) XP_ListGetObjectNum (wholeList, i);
|
|
if (PABDirectory == s->dirType)
|
|
{
|
|
*cab = s;
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
* Functions for managing JavaScript prefs for the DIR_Servers
|
|
*/
|
|
|
|
#if !defined(MOZADDRSTANDALONE)
|
|
|
|
static char *DIR_GetStringPref(const char *prefRoot, const char *prefLeaf, char *scratch, const char *defaultValue)
|
|
{
|
|
int valueLength = 0;
|
|
char *value = NULL;
|
|
XP_STRCPY(scratch, prefRoot);
|
|
XP_STRCAT (scratch, prefLeaf);
|
|
|
|
if (PREF_NOERROR == PREF_CopyCharPref(scratch, &value))
|
|
{
|
|
/* unfortunately, there may be some prefs out there which look like this */
|
|
if (!XP_STRCMP(value, "(null)"))
|
|
value = defaultValue ? XP_STRDUP(defaultValue) : NULL;
|
|
}
|
|
else
|
|
value = defaultValue ? XP_STRDUP(defaultValue) : NULL;
|
|
return value;
|
|
}
|
|
|
|
|
|
static int32 DIR_GetIntPref(const char *prefRoot, const char *prefLeaf, char *scratch, int32 defaultValue)
|
|
{
|
|
int32 value;
|
|
XP_STRCPY(scratch, prefRoot);
|
|
if (PREF_NOERROR != PREF_GetIntPref(XP_STRCAT (scratch, prefLeaf), &value))
|
|
value = defaultValue;
|
|
return value;
|
|
}
|
|
|
|
|
|
static XP_Bool DIR_GetBoolPref(const char *prefRoot, const char *prefLeaf, char *scratch, XP_Bool defaultValue)
|
|
{
|
|
XP_Bool value;
|
|
XP_STRCPY(scratch, prefRoot);
|
|
if (PREF_NOERROR != PREF_GetBoolPref(XP_STRCAT (scratch, prefLeaf), &value))
|
|
value = defaultValue;
|
|
return value;
|
|
}
|
|
|
|
|
|
int DIR_AttributeNameToId(const char *attrName, DIR_AttributeId *id)
|
|
{
|
|
int status = 0;
|
|
|
|
switch (attrName[0])
|
|
{
|
|
case 'a':
|
|
if (!XP_STRCASECMP(attrName, "auth"))
|
|
*id = auth;
|
|
else
|
|
status = -1;
|
|
break;
|
|
|
|
case 'c' :
|
|
if (!XP_STRCASECMP(attrName, "cn"))
|
|
*id = cn;
|
|
else if (!XP_STRNCASECMP(attrName, "custom", 6))
|
|
{
|
|
switch (attrName[6])
|
|
{
|
|
case '1': *id = custom1; break;
|
|
case '2': *id = custom2; break;
|
|
case '3': *id = custom3; break;
|
|
case '4': *id = custom4; break;
|
|
case '5': *id = custom5; break;
|
|
default: status = -1;
|
|
}
|
|
}
|
|
else
|
|
status = -1;
|
|
break;
|
|
case 'g':
|
|
if (!XP_STRCASECMP(attrName, "givenname"))
|
|
*id = givenname;
|
|
else
|
|
status = -1;
|
|
break;
|
|
case 'l':
|
|
if (!XP_STRCASECMP(attrName, "l"))
|
|
*id = l;
|
|
else
|
|
status = -1;
|
|
break;
|
|
case 'm':
|
|
if (!XP_STRCASECMP(attrName, "mail"))
|
|
*id = mail;
|
|
else
|
|
status = -1;
|
|
break;
|
|
case 'o':
|
|
if (!XP_STRCASECMP(attrName, "o"))
|
|
*id = o;
|
|
else if (!XP_STRCASECMP(attrName, "ou"))
|
|
*id = ou;
|
|
else
|
|
status = -1;
|
|
break;
|
|
case 's':
|
|
if (!XP_STRCASECMP(attrName, "street"))
|
|
*id = street;
|
|
else if (!XP_STRCASECMP(attrName, "sn"))
|
|
*id = sn;
|
|
else
|
|
status = -1;
|
|
break;
|
|
case 't':
|
|
if (!XP_STRCASECMP(attrName, "telephonenumber"))
|
|
*id = telephonenumber;
|
|
else
|
|
status = -1;
|
|
break;
|
|
default:
|
|
status = -1;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
static int DIR_AddCustomAttribute(DIR_Server *server, const char *attrName, char *jsAttr)
|
|
{
|
|
int status = 0;
|
|
char *jsCompleteAttr = NULL;
|
|
char *jsAttrForTokenizing = jsAttr;
|
|
|
|
DIR_AttributeId id;
|
|
status = DIR_AttributeNameToId (attrName, &id);
|
|
|
|
/* If the string they gave us doesn't have a ':' in it, assume it's one or more
|
|
* attributes without a pretty name. So find the default pretty name, and generate
|
|
* a "complete" string to use for tokenizing.
|
|
*/
|
|
if (status == 0 && !XP_STRCHR(jsAttr, ':'))
|
|
{
|
|
const char *defaultPrettyName = DIR_GetAttributeName (server, id);
|
|
if (defaultPrettyName)
|
|
{
|
|
jsCompleteAttr = PR_smprintf ("%s:%s", defaultPrettyName, jsAttr);
|
|
if (jsCompleteAttr)
|
|
jsAttrForTokenizing = jsCompleteAttr;
|
|
else
|
|
status = MK_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
|
|
if (status == 0)
|
|
{
|
|
char *scratchAttr = XP_STRDUP(jsAttrForTokenizing);
|
|
DIR_Attribute *attrStruct = (DIR_Attribute*) XP_ALLOC(sizeof(DIR_Attribute));
|
|
if (!server->customAttributes)
|
|
server->customAttributes = XP_ListNew();
|
|
|
|
if (attrStruct && server->customAttributes && scratchAttr)
|
|
{
|
|
char *attrToken = NULL;
|
|
int attrCount = 0;
|
|
|
|
XP_BZERO(attrStruct, sizeof(DIR_Attribute));
|
|
|
|
/* Try to pull out the pretty name into the struct */
|
|
attrStruct->id = id;
|
|
attrStruct->prettyName = XP_STRDUP(XP_STRTOK(scratchAttr, ":"));
|
|
|
|
/* Count up the attribute names */
|
|
while ((attrToken = XP_STRTOK(NULL, ", ")) != NULL)
|
|
attrCount++;
|
|
|
|
/* Pull the attribute names into the struct */
|
|
XP_STRCPY(scratchAttr, jsAttrForTokenizing);
|
|
XP_STRTOK(scratchAttr, ":");
|
|
attrStruct->attrNames = (char**) XP_ALLOC((attrCount + 1) * sizeof(char*));
|
|
if (attrStruct->attrNames)
|
|
{
|
|
int i = 0;
|
|
while ((attrToken = XP_STRTOK(NULL, ", ")) != NULL)
|
|
attrStruct->attrNames[i++] = XP_STRDUP(attrToken);
|
|
attrStruct->attrNames[i] = NULL; /* null-terminate the array */
|
|
}
|
|
|
|
if (status == 0)
|
|
XP_ListAddObject (server->customAttributes, attrStruct);
|
|
else
|
|
DIR_DeleteAttribute (attrStruct);
|
|
|
|
XP_FREE(scratchAttr);
|
|
}
|
|
else
|
|
status = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (jsCompleteAttr)
|
|
XP_FREE(jsCompleteAttr);
|
|
|
|
return status;
|
|
}
|
|
|
|
static int dir_CreateTokenListFromWholePref(const char *pref, char ***outList, int *outCount)
|
|
{
|
|
int result = 0;
|
|
char *commaSeparatedList = NULL;
|
|
|
|
if (PREF_NOERROR == PREF_CopyCharPref(pref, &commaSeparatedList) && commaSeparatedList)
|
|
{
|
|
char *tmpList = commaSeparatedList;
|
|
*outCount = 1;
|
|
while (*tmpList)
|
|
if (*tmpList++ == ',')
|
|
(*outCount)++;
|
|
|
|
*outList = (char**) XP_ALLOC(*outCount * sizeof(char*));
|
|
if (*outList)
|
|
{
|
|
int i;
|
|
char *token = XP_STRTOK(commaSeparatedList, ", ");
|
|
for (i = 0; i < *outCount; i++)
|
|
{
|
|
(*outList)[i] = XP_STRDUP(token);
|
|
token = XP_STRTOK(NULL, ", ");
|
|
}
|
|
}
|
|
else
|
|
result = MK_OUT_OF_MEMORY;
|
|
|
|
XP_FREE (commaSeparatedList);
|
|
}
|
|
else
|
|
result = -1;
|
|
return result;
|
|
}
|
|
|
|
|
|
static int dir_CreateTokenListFromPref(const char *prefBase, const char *prefLeaf, char *scratch, char ***outList, int *outCount)
|
|
{
|
|
XP_STRCPY (scratch, prefBase);
|
|
XP_STRCAT (scratch, prefLeaf);
|
|
|
|
return dir_CreateTokenListFromWholePref(scratch, outList, outCount);
|
|
}
|
|
|
|
|
|
static int dir_ConvertTokenListToIdList(char **tokenList, int tokenCount, DIR_AttributeId **outList)
|
|
{
|
|
*outList = (DIR_AttributeId*) XP_ALLOC(sizeof(DIR_AttributeId) * tokenCount);
|
|
if (*outList)
|
|
{
|
|
int i;
|
|
for (i = 0; i < tokenCount; i++)
|
|
DIR_AttributeNameToId (tokenList[i], &(*outList)[i]);
|
|
}
|
|
else
|
|
return MK_OUT_OF_MEMORY;
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void dir_GetReplicationInfo(const char *prefName, DIR_Server *server, char *scratch)
|
|
{
|
|
char *replPrefName;
|
|
XP_Bool arePrefsValid = FALSE;
|
|
|
|
XP_ASSERT (server->replInfo == NULL);
|
|
|
|
replPrefName = (char *) XP_ALLOC(128);
|
|
server->replInfo = (DIR_ReplicationInfo *) XP_CALLOC (1, sizeof (DIR_ReplicationInfo));
|
|
if (server->replInfo && replPrefName)
|
|
{
|
|
XP_STRCPY(replPrefName, prefName);
|
|
XP_STRCAT(replPrefName, "replication.");
|
|
|
|
if (DIR_GetBoolPref (replPrefName, "enabled", scratch, kDefaultReplicaEnabled))
|
|
DIR_SetFlag (server, DIR_REPLICATION_ENABLED);
|
|
|
|
server->replInfo->fileName = DIR_GetStringPref (replPrefName, "fileName", scratch, kDefaultReplicaFileName);
|
|
server->replInfo->dataVersion = DIR_GetStringPref (replPrefName, "dataVersion", scratch, kDefaultReplicaDataVersion);
|
|
|
|
/* The file name and data version must be set or we ignore all of the
|
|
* replication prefs.
|
|
*/
|
|
if (server->replInfo->fileName && server->replInfo->dataVersion)
|
|
{
|
|
dir_CreateTokenListFromPref (replPrefName, "excludedAttributes", scratch, &server->replInfo->excludedAttributes,
|
|
&server->replInfo->excludedAttributesCount);
|
|
|
|
server->replInfo->description = DIR_GetStringPref (replPrefName, "description", scratch, kDefaultReplicaDescription);
|
|
server->replInfo->filter = DIR_GetStringPref (replPrefName, "filter", scratch, kDefaultReplicaFilter);
|
|
server->replInfo->lastChangeNumber = DIR_GetIntPref (replPrefName, "lastChangeNumber", scratch, kDefaultReplicaChangeNumber);
|
|
|
|
arePrefsValid = TRUE;
|
|
}
|
|
}
|
|
|
|
if (!arePrefsValid)
|
|
{
|
|
if (server->replInfo)
|
|
{
|
|
XP_FREEIF(server->replInfo->fileName);
|
|
XP_FREEIF(server->replInfo->dataVersion);
|
|
}
|
|
XP_FREEIF(server->replInfo);
|
|
XP_FREEIF(replPrefName);
|
|
}
|
|
}
|
|
|
|
|
|
/* Called at startup-time to read whatever overrides the LDAP site administrator has
|
|
* done to the attribute names
|
|
*/
|
|
static int DIR_GetCustomAttributePrefs(const char *prefName, DIR_Server *server, char *scratch)
|
|
{
|
|
char **tokenList = NULL;
|
|
char *childList = NULL;
|
|
XP_STRCPY(scratch, prefName);
|
|
|
|
if (PREF_NOERROR == PREF_CreateChildList(XP_STRCAT(scratch, "attributes"), &childList))
|
|
{
|
|
if (childList && childList[0])
|
|
{
|
|
char *child = NULL;
|
|
int index = 0;
|
|
while ((child = PREF_NextChild (childList, &index)) != NULL)
|
|
{
|
|
char *jsValue = NULL;
|
|
if (PREF_NOERROR == PREF_CopyCharPref (child, &jsValue))
|
|
{
|
|
if (jsValue && jsValue[0])
|
|
{
|
|
char *attrName = child + XP_STRLEN(scratch) + 1;
|
|
DIR_AddCustomAttribute (server, attrName, jsValue);
|
|
}
|
|
XP_FREEIF(jsValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
XP_FREEIF(childList);
|
|
}
|
|
|
|
/* The replicated attributes and basic search attributes can only be
|
|
* attributes which are in our predefined set (DIR_AttributeId) so
|
|
* store those in an array of IDs for more convenient access
|
|
*/
|
|
dir_GetReplicationInfo (prefName, server, scratch);
|
|
|
|
if (0 == dir_CreateTokenListFromPref (prefName, "basicSearchAttributes", scratch,
|
|
&tokenList, &server->basicSearchAttributesCount))
|
|
{
|
|
dir_ConvertTokenListToIdList (tokenList, server->basicSearchAttributesCount,
|
|
&server->basicSearchAttributes);
|
|
dir_DeleteTokenList (tokenList, server->basicSearchAttributesCount);
|
|
}
|
|
|
|
/* The DN attributes and suppressed attributes can be attributes that
|
|
* we've never heard of, so they're stored by name, so we can match 'em
|
|
* as we get 'em from the server
|
|
*/
|
|
dir_CreateTokenListFromPref (prefName, "html.dnAttributes", scratch,
|
|
&server->dnAttributes, &server->dnAttributesCount);
|
|
dir_CreateTokenListFromPref (prefName, "html.excludedAttributes", scratch,
|
|
&server->suppressedAttributes, &server->suppressedAttributesCount);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Called at startup-time to read whatever overrides the LDAP site administrator has
|
|
* done to the filtering logic
|
|
*/
|
|
static int DIR_GetCustomFilterPrefs(const char *prefName, DIR_Server *server, char *scratch)
|
|
{
|
|
int status = 0;
|
|
XP_Bool keepGoing = TRUE;
|
|
int filterNum = 1;
|
|
char *localScratch = XP_ALLOC(128);
|
|
if (!localScratch)
|
|
return MK_OUT_OF_MEMORY;
|
|
|
|
while (keepGoing && !status)
|
|
{
|
|
char *childList = NULL;
|
|
|
|
PR_snprintf (scratch, 128, "%sfilter%d", prefName, filterNum);
|
|
if (PREF_NOERROR == PREF_CreateChildList(scratch, &childList))
|
|
{
|
|
if ('\0' != childList[0])
|
|
{
|
|
DIR_Filter *filter = (DIR_Filter*) XP_ALLOC (sizeof(DIR_Filter));
|
|
if (filter)
|
|
{
|
|
XP_Bool tempBool;
|
|
XP_BZERO(filter, sizeof(DIR_Filter));
|
|
|
|
/* Pull per-directory filter preferences out of JS values */
|
|
if (1 == filterNum)
|
|
{
|
|
server->tokenSeps = DIR_GetStringPref (prefName, "wordSeparators", localScratch, kDefaultTokenSeps);
|
|
XP_STRCAT(scratch, ".");
|
|
}
|
|
|
|
/* Pull per-filter preferences out of JS values */
|
|
filter->string = DIR_GetStringPref (scratch, "string", localScratch,
|
|
server->efficientWildcards ? kDefaultFilter : kDefaultEfficientFilter);
|
|
tempBool = DIR_GetBoolPref (scratch, "repeatFilterForWords", localScratch, kDefaultRepeatFilterForTokens);
|
|
if (tempBool)
|
|
filter->flags |= DIR_F_REPEAT_FILTER_FOR_TOKENS;
|
|
tempBool = DIR_GetBoolPref (scratch, "substituteStarsForSpaces", localScratch, kDefaultSubstStarsForSpaces);
|
|
if (tempBool)
|
|
filter->flags |= DIR_F_SUBST_STARS_FOR_SPACES;
|
|
|
|
/* Add resulting DIR_Filter to the list */
|
|
if (!server->customFilters)
|
|
server->customFilters = XP_ListNew();
|
|
if (server->customFilters)
|
|
XP_ListAddObject (server->customFilters, filter);
|
|
else
|
|
status = MK_OUT_OF_MEMORY;
|
|
}
|
|
else
|
|
status = MK_OUT_OF_MEMORY;
|
|
filterNum++;
|
|
}
|
|
else
|
|
keepGoing = FALSE;
|
|
XP_FREE(childList);
|
|
}
|
|
else
|
|
keepGoing = FALSE;
|
|
}
|
|
|
|
XP_FREE(localScratch);
|
|
return status;
|
|
}
|
|
|
|
/* This will convert from the old preference that was a path and filename */
|
|
/* to a just a filename */
|
|
static void DIR_ConvertServerFileName(DIR_Server* pServer)
|
|
{
|
|
char* leafName = pServer->fileName;
|
|
char* newLeafName = NULL;
|
|
#if defined(XP_WIN) || defined(XP_OS2)
|
|
/* This is to allow users to share the same address book.
|
|
* It only works if the user specify a full path filename.
|
|
*/
|
|
if (! XP_FileIsFullPath(leafName))
|
|
newLeafName = XP_STRRCHR (leafName, '\\');
|
|
#else
|
|
newLeafName = XP_STRRCHR (leafName, '/');
|
|
#endif
|
|
pServer->fileName = newLeafName ? XP_STRDUP(newLeafName + 1) : XP_STRDUP(leafName);
|
|
if (leafName) XP_FREE(leafName);
|
|
}
|
|
|
|
/* This will generate a correct filename and then remove the path */
|
|
void DIR_SetFileName(char** fileName, const char* leafName)
|
|
{
|
|
char* tempName = WH_TempName(xpAddrBook, leafName);
|
|
char* nativeName = WH_FileName(tempName, xpAddrBook);
|
|
char* urlName = XP_PlatformFileToURL(nativeName);
|
|
#if defined(XP_WIN) || defined(XP_UNIX) || defined(XP_MAC) || defined(XP_OS2)
|
|
char* newLeafName = XP_STRRCHR (urlName + XP_STRLEN("file://"), '/');
|
|
(*fileName) = newLeafName ? XP_STRDUP(newLeafName + 1) : XP_STRDUP(urlName + XP_STRLEN("file://"));
|
|
#else
|
|
(*fileName) = XP_STRDUP(urlName + XP_STRLEN("file://"));
|
|
#endif
|
|
if (urlName) XP_FREE(urlName);
|
|
if (nativeName) XP_FREE(nativeName);
|
|
if (tempName) XP_FREE(tempName);
|
|
}
|
|
|
|
void DIR_SetServerFileName(DIR_Server *server, const char* leafName)
|
|
{
|
|
DIR_SetFileName(&(server->fileName), leafName);
|
|
}
|
|
|
|
/* This will reconstruct a correct filename including the path */
|
|
void DIR_GetServerFileName(char** filename, const char* leafName)
|
|
{
|
|
#ifdef XP_MAC
|
|
char* realLeafName;
|
|
char* nativeName;
|
|
char* urlName;
|
|
if (XP_STRCHR(leafName, ':') != NULL)
|
|
realLeafName = XP_STRRCHR(leafName, ':') + 1; /* makes sure that leafName is not a fullpath */
|
|
else
|
|
realLeafName = leafName;
|
|
|
|
nativeName = WH_FileName(realLeafName, xpAddrBookNew);
|
|
urlName = XP_PlatformFileToURL(nativeName);
|
|
(*filename) = XP_STRDUP(urlName + XP_STRLEN("file://"));
|
|
if (urlName) XP_FREE(urlName);
|
|
#elif defined(XP_WIN)
|
|
/* jefft -- Bug 73349. To allow users share same address book.
|
|
* This only works if user sepcified a full path name in his
|
|
* prefs.js
|
|
*/
|
|
char *nativeName = WH_FilePlatformName (leafName);
|
|
char *fullnativeName;
|
|
if (XP_FileIsFullPath(nativeName)) {
|
|
fullnativeName = nativeName;
|
|
nativeName = NULL;
|
|
}
|
|
else {
|
|
fullnativeName = WH_FileName(nativeName, xpAddrBookNew);
|
|
}
|
|
(*filename) = fullnativeName;
|
|
#else
|
|
char* nativeName = WH_FilePlatformName (leafName);
|
|
char* fullnativeName = WH_FileName(nativeName, xpAddrBookNew);
|
|
(*filename) = fullnativeName;
|
|
#endif
|
|
if (nativeName) XP_FREE(nativeName);
|
|
}
|
|
|
|
static int DIR_GetPrefsFromBranch(XP_List **list, const char *pabFile, const char *branch)
|
|
{
|
|
int32 numDirectories = 0;
|
|
int i = 0;
|
|
char *prefstring = NULL;
|
|
char *tempString = NULL;
|
|
int result = 0;
|
|
XP_Bool hasPAB = FALSE;
|
|
DIR_Server *pNewServer = NULL;
|
|
char *newfilename = NULL;
|
|
|
|
if (*list)
|
|
DIR_DeleteServerList(*list);
|
|
|
|
prefstring = (char *) XP_ALLOC(128);
|
|
tempString = (char *) XP_ALLOC(256);
|
|
|
|
(*list) = XP_ListNew();
|
|
/* get the preference for how many directories */
|
|
if (prefstring && tempString && (*list) && branch && *branch)
|
|
{
|
|
char *numberOfDirs = PR_smprintf ("%s.number_of_directories", branch);
|
|
if (numberOfDirs)
|
|
PREF_GetIntPref(numberOfDirs, &numDirectories);
|
|
|
|
for (i = 1; i <= numDirectories; i++)
|
|
{
|
|
pNewServer = (DIR_Server *) XP_ALLOC(sizeof(DIR_Server));
|
|
if (pNewServer)
|
|
{
|
|
XP_Bool prefBool;
|
|
int prefInt;
|
|
|
|
DIR_InitServer(pNewServer);
|
|
XP_SPRINTF(prefstring, "%s.directory%i.", branch, i);
|
|
|
|
pNewServer->isSecure = DIR_GetBoolPref (prefstring, "isSecure", tempString, FALSE);
|
|
pNewServer->saveResults = DIR_GetBoolPref (prefstring, "saveResults", tempString, TRUE);
|
|
pNewServer->efficientWildcards = DIR_GetBoolPref (prefstring, "efficientWildcards", tempString, TRUE);
|
|
pNewServer->port = DIR_GetIntPref (prefstring, "port", tempString, pNewServer->isSecure ? LDAPS_PORT : LDAP_PORT);
|
|
if (pNewServer->port == 0)
|
|
pNewServer->port = pNewServer->isSecure ? LDAPS_PORT : LDAP_PORT;
|
|
pNewServer->maxHits = DIR_GetIntPref (prefstring, "maxHits", tempString, kDefaultMaxHits);
|
|
pNewServer->description = DIR_GetStringPref (prefstring, "description", tempString, "");
|
|
pNewServer->serverName = DIR_GetStringPref (prefstring, "serverName", tempString, "");
|
|
pNewServer->searchBase = DIR_GetStringPref (prefstring, "searchBase", tempString, "");
|
|
if (!XP_STRCASECMP(pNewServer->serverName, "ldap.infospace.com") && !*pNewServer->searchBase)
|
|
{
|
|
/* 4.0 unfortunately shipped with the wrong searchbase for Infospace, so
|
|
* if we find Infospace, and their searchBase hasn't been set, make it right
|
|
* It's not legal to do this in all.js unless we rename the LDAP prefs tree (ugh)
|
|
*/
|
|
pNewServer->searchBase = XP_STRDUP ("c=US");
|
|
PREF_SetCharPref (tempString, "c=US");
|
|
}
|
|
|
|
pNewServer->dirType = (DirectoryType)DIR_GetIntPref (prefstring, "dirType", tempString, (int) LDAPDirectory);
|
|
if (pNewServer->dirType == PABDirectory)
|
|
{
|
|
hasPAB = TRUE;
|
|
/* make sure there is a true PAB */
|
|
if (XP_STRLEN (pNewServer->serverName) == 0)
|
|
pNewServer->isOffline = FALSE;
|
|
pNewServer->saveResults = TRUE; /* never let someone delete their PAB this way */
|
|
}
|
|
|
|
pNewServer->fileName = DIR_GetStringPref (prefstring, "filename", tempString, "");
|
|
if (!pNewServer->fileName || !*(pNewServer->fileName))
|
|
{
|
|
if (pNewServer->dirType != PABDirectory)
|
|
DIR_SetServerFileName (pNewServer, pNewServer->serverName);
|
|
else
|
|
{
|
|
XP_FREEIF(pNewServer->fileName);
|
|
pNewServer->fileName = XP_STRDUP (pabFile);
|
|
}
|
|
}
|
|
#if defined(XP_WIN) || defined(XP_UNIX) || defined(XP_MAC) || defined(XP_OS2)
|
|
DIR_ConvertServerFileName(pNewServer);
|
|
#endif
|
|
pNewServer->lastSearchString = DIR_GetStringPref (prefstring, "searchString", tempString, "");
|
|
pNewServer->isOffline = DIR_GetBoolPref (prefstring, "isOffline", tempString, kDefaultIsOffline);
|
|
|
|
/* This is where site-configurable attributes and filters are read from JavaScript */
|
|
DIR_GetCustomAttributePrefs (prefstring, pNewServer, tempString);
|
|
DIR_GetCustomFilterPrefs (prefstring, pNewServer, tempString);
|
|
|
|
/* Get authentication prefs */
|
|
pNewServer->enableAuth = DIR_GetBoolPref (prefstring, "enableAuth", tempString, kDefaultEnableAuth);
|
|
pNewServer->authDn = DIR_GetStringPref (prefstring, "authDn", tempString, NULL);
|
|
pNewServer->savePassword = DIR_GetBoolPref (prefstring, "savePassword", tempString, kDefaultSavePassword);
|
|
if (pNewServer->savePassword)
|
|
pNewServer->password = DIR_GetStringPref (prefstring, "password", tempString, "");
|
|
|
|
prefBool = DIR_GetBoolPref (prefstring, "autoComplete.enabled", tempString, kDefaultAutoCompleteEnabled);
|
|
DIR_ForceFlag (pNewServer, DIR_AUTO_COMPLETE_ENABLED, prefBool);
|
|
prefInt = DIR_GetIntPref (prefstring, "autoComplete.style", tempString, kDefaultAutoCompleteStyle);
|
|
pNewServer->autoCompleteStyle = (DIR_AutoCompleteStyle) prefInt;
|
|
|
|
prefBool = DIR_GetBoolPref (prefstring, "utf8Disabled", tempString, kDefaultUtf8Disabled);
|
|
DIR_ForceFlag (pNewServer, DIR_UTF8_DISABLED, prefBool);
|
|
|
|
prefBool = DIR_GetBoolPref (prefstring, "ldapPublicDirectory", tempString, kDefaultLdapPublicDirectory);
|
|
DIR_ForceFlag (pNewServer, DIR_LDAP_PUBLIC_DIRECTORY, prefBool);
|
|
DIR_ForceFlag (pNewServer, DIR_LDAP_ROOTDSE_PARSED, prefBool);
|
|
|
|
pNewServer->customDisplayUrl = DIR_GetStringPref (prefstring, "customDisplayUrl", tempString, "");
|
|
|
|
XP_ListAddObjectToEnd((*list), pNewServer);
|
|
}
|
|
}
|
|
|
|
XP_FREEIF(numberOfDirs);
|
|
|
|
/* all.js should have filled this stuff in */
|
|
XP_ASSERT(hasPAB);
|
|
XP_ASSERT(numDirectories != 0);
|
|
}
|
|
else
|
|
result = -1;
|
|
|
|
XP_FREEIF (prefstring);
|
|
XP_FREEIF (tempString);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
int DIR_GetServerPreferences(XP_List **list, const char* pabFile)
|
|
{
|
|
int err = 0;
|
|
XP_List *oldList = NULL;
|
|
XP_List *newList = NULL;
|
|
char *oldChildren = NULL;
|
|
|
|
int32 listVersion = -1;
|
|
XP_Bool userHasOldPrefs = FALSE;
|
|
|
|
if (PREF_NOERROR == PREF_GetIntPref ("ldapList.version", &listVersion))
|
|
userHasOldPrefs = (kCurrentListVersion > listVersion);
|
|
else
|
|
userHasOldPrefs = TRUE;
|
|
|
|
/* Look to see if there's an old-style "directories" tree in prefs */
|
|
if (PREF_NOERROR == PREF_CreateChildList ("directories", &oldChildren))
|
|
{
|
|
if (oldChildren)
|
|
{
|
|
if (XP_STRLEN(oldChildren))
|
|
{
|
|
if (userHasOldPrefs)
|
|
err = DIR_GetPrefsFromBranch (&oldList, pabFile, "directories");
|
|
PREF_DeleteBranch ("directories");
|
|
}
|
|
XP_FREEIF(oldChildren);
|
|
}
|
|
}
|
|
/* Look to see if there's an old-style "ldap" tree in prefs */
|
|
else if (PREF_NOERROR == PREF_CreateChildList ("ldap", &oldChildren))
|
|
{
|
|
if (oldChildren)
|
|
{
|
|
if (XP_STRLEN(oldChildren))
|
|
{
|
|
if (userHasOldPrefs)
|
|
err = DIR_GetPrefsFromBranch (&oldList, pabFile, "ldap");
|
|
PREF_DeleteBranch ("ldap");
|
|
}
|
|
XP_FREEIF(oldChildren);
|
|
}
|
|
}
|
|
|
|
/* Find the new-style "ldap_1" tree in prefs */
|
|
DIR_GetPrefsFromBranch(&newList, pabFile, "ldap_1");
|
|
|
|
if (oldList && newList)
|
|
{
|
|
/* Merge the new tree onto the old tree, new on top, unique old at bottom */
|
|
DIR_Server *oldServer;
|
|
XP_List *walkOldList = oldList;
|
|
|
|
while (NULL != (oldServer = XP_ListNextObject(walkOldList)))
|
|
{
|
|
XP_Bool addOldServer = TRUE;
|
|
DIR_Server *newServer;
|
|
XP_List *walkNewList = newList;
|
|
|
|
while (NULL != (newServer = XP_ListNextObject(walkNewList)) && addOldServer)
|
|
{
|
|
if (DIR_AreServersSame(oldServer, newServer))
|
|
addOldServer = FALSE; /* don't add servers which are in the new list */
|
|
else if (PABDirectory == oldServer->dirType)
|
|
addOldServer = FALSE; /* don't need the old PAB; there's already one in ALL.JS */
|
|
else if (!XP_STRCMP(oldServer->serverName, "ldap-trace.fedex.com"))
|
|
addOldServer = FALSE;
|
|
}
|
|
|
|
if (addOldServer)
|
|
{
|
|
DIR_Server *copyOfOldServer;
|
|
DIR_CopyServer(oldServer, ©OfOldServer);
|
|
XP_ListAddObjectToEnd(newList, copyOfOldServer);
|
|
}
|
|
}
|
|
|
|
/* Delete the list of old-style prefs */
|
|
DIR_DeleteServerList (oldList);
|
|
|
|
/* Write the new/merged list so we get it next time we ask */
|
|
DIR_SaveServerPreferences (newList);
|
|
}
|
|
|
|
PREF_SetIntPref ("ldapList.version", kCurrentListVersion);
|
|
|
|
*list = newList;
|
|
return err;
|
|
}
|
|
|
|
|
|
static void DIR_ClearPrefBranch(const char *branch)
|
|
{
|
|
/* This little function provides a way to delete a prefs object but still
|
|
* allow reassignment of that object later.
|
|
*/
|
|
char *recreateBranch = NULL;
|
|
|
|
PREF_DeleteBranch (branch);
|
|
recreateBranch = PR_smprintf("pref_inittree(\"%s\")", branch);
|
|
if (recreateBranch)
|
|
{
|
|
PREF_QuietEvaluateJSBuffer (recreateBranch, XP_STRLEN(recreateBranch));
|
|
XP_FREE(recreateBranch);
|
|
}
|
|
}
|
|
|
|
|
|
static void DIR_ClearIntPref (const char *pref)
|
|
{
|
|
int32 oldDefault;
|
|
int prefErr = PREF_GetDefaultIntPref (pref, &oldDefault);
|
|
DIR_ClearPrefBranch (pref);
|
|
if (prefErr >= 0)
|
|
PREF_SetDefaultIntPref (pref, oldDefault);
|
|
}
|
|
|
|
|
|
static void DIR_ClearStringPref (const char *pref)
|
|
{
|
|
char *oldDefault = NULL;
|
|
int prefErr = PREF_CopyDefaultCharPref (pref, &oldDefault);
|
|
DIR_ClearPrefBranch (pref);
|
|
if (prefErr >= 0)
|
|
PREF_SetDefaultCharPref (pref, oldDefault);
|
|
XP_FREEIF(oldDefault);
|
|
}
|
|
|
|
|
|
static void DIR_ClearBoolPref (const char *pref)
|
|
{
|
|
XP_Bool oldDefault;
|
|
int prefErr = PREF_GetDefaultBoolPref (pref, &oldDefault);
|
|
DIR_ClearPrefBranch (pref);
|
|
if (prefErr >= 0)
|
|
PREF_SetDefaultBoolPref (pref, oldDefault);
|
|
}
|
|
|
|
|
|
static void DIR_SetStringPref (const char *prefRoot, const char *prefLeaf, char *scratch, const char *value, const char *defaultValue)
|
|
{
|
|
char *defaultPref = NULL;
|
|
int prefErr = PREF_NOERROR;
|
|
|
|
XP_STRCPY(scratch, prefRoot);
|
|
XP_STRCAT (scratch, prefLeaf);
|
|
|
|
if (PREF_NOERROR == PREF_CopyDefaultCharPref (scratch, &defaultPref))
|
|
{
|
|
/* If there's a default pref, just set ours in and let libpref worry
|
|
* about potential defaults in all.js
|
|
*/
|
|
if (value) /* added this check to make sure we have a value before we try to set it..*/
|
|
prefErr = PREF_SetCharPref (scratch, value);
|
|
else
|
|
DIR_ClearStringPref(scratch);
|
|
|
|
XP_FREE(defaultPref);
|
|
}
|
|
else
|
|
{
|
|
/* If there's no default pref, look for a user pref, and only set our value in
|
|
* if the user pref is different than one of them.
|
|
*/
|
|
char *userPref = NULL;
|
|
if (PREF_NOERROR == PREF_CopyCharPref (scratch, &userPref))
|
|
{
|
|
if (value && (defaultValue ? XP_STRCASECMP(value, defaultValue) : value != defaultValue))
|
|
prefErr = PREF_SetCharPref (scratch, value);
|
|
else
|
|
DIR_ClearStringPref (scratch);
|
|
}
|
|
else
|
|
{
|
|
if (value && (defaultValue ? XP_STRCASECMP(value, defaultValue) : value != defaultValue))
|
|
prefErr = PREF_SetCharPref (scratch, value);
|
|
}
|
|
}
|
|
|
|
XP_ASSERT(prefErr >= 0);
|
|
}
|
|
|
|
|
|
static void DIR_SetIntPref (const char *prefRoot, const char *prefLeaf, char *scratch, int32 value, int32 defaultValue)
|
|
{
|
|
int32 defaultPref;
|
|
int prefErr = 0;
|
|
|
|
XP_STRCPY(scratch, prefRoot);
|
|
XP_STRCAT (scratch, prefLeaf);
|
|
|
|
if (PREF_NOERROR == PREF_GetDefaultIntPref (scratch, &defaultPref))
|
|
{
|
|
/* solve the problem where reordering user prefs must override default prefs */
|
|
PREF_SetIntPref (scratch, value);
|
|
}
|
|
else
|
|
{
|
|
int32 userPref;
|
|
if (PREF_NOERROR == PREF_GetIntPref (scratch, &userPref))
|
|
{
|
|
if (value != defaultValue)
|
|
prefErr = PREF_SetIntPref(scratch, value);
|
|
else
|
|
DIR_ClearIntPref (scratch);
|
|
}
|
|
else
|
|
{
|
|
if (value != defaultValue)
|
|
prefErr = PREF_SetIntPref (scratch, value);
|
|
}
|
|
}
|
|
|
|
XP_ASSERT(prefErr >= 0);
|
|
}
|
|
|
|
|
|
static void DIR_SetBoolPref (const char *prefRoot, const char *prefLeaf, char *scratch, XP_Bool value, XP_Bool defaultValue)
|
|
{
|
|
XP_Bool defaultPref;
|
|
int prefErr = PREF_NOERROR;
|
|
|
|
XP_STRCPY(scratch, prefRoot);
|
|
XP_STRCAT (scratch, prefLeaf);
|
|
|
|
if (PREF_NOERROR == PREF_GetDefaultBoolPref (scratch, &defaultPref))
|
|
{
|
|
/* solve the problem where reordering user prefs must override default prefs */
|
|
prefErr = PREF_SetBoolPref (scratch, value);
|
|
}
|
|
else
|
|
{
|
|
XP_Bool userPref;
|
|
if (PREF_NOERROR == PREF_GetBoolPref (scratch, &userPref))
|
|
{
|
|
if (value != defaultValue)
|
|
prefErr = PREF_SetBoolPref(scratch, value);
|
|
else
|
|
DIR_ClearBoolPref (scratch);
|
|
}
|
|
else
|
|
{
|
|
if (value != defaultValue)
|
|
prefErr = PREF_SetBoolPref (scratch, value);
|
|
}
|
|
|
|
}
|
|
|
|
XP_ASSERT(prefErr >= 0);
|
|
}
|
|
|
|
|
|
static int DIR_ConvertAttributeToPrefsString (DIR_Attribute *attrib, char **ppPrefsString)
|
|
{
|
|
int err = 0;
|
|
|
|
/* Compute size in bytes req'd for prefs string */
|
|
int length = XP_STRLEN(attrib->prettyName);
|
|
int i = 0;
|
|
while (attrib->attrNames[i])
|
|
{
|
|
length += XP_STRLEN(attrib->attrNames[i]) + 1; /* +1 for comma separator */
|
|
i++;
|
|
}
|
|
length += 1; /* +1 for colon */
|
|
|
|
/* Allocate prefs string */
|
|
*ppPrefsString = (char*) XP_ALLOC(length + 1); /* +1 for null term */
|
|
|
|
/* Unravel attrib struct back out into prefs */
|
|
if (*ppPrefsString)
|
|
{
|
|
int j = 0;
|
|
XP_STRCPY (*ppPrefsString, attrib->prettyName);
|
|
XP_STRCAT (*ppPrefsString, ":");
|
|
while (attrib->attrNames[j])
|
|
{
|
|
XP_STRCAT (*ppPrefsString, attrib->attrNames[j]);
|
|
if (j + 1 < i)
|
|
XP_STRCAT (*ppPrefsString, ",");
|
|
j++;
|
|
}
|
|
}
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
static int DIR_SaveOneCustomAttribute (const char *prefRoot, char *scratch, DIR_Server *server, DIR_AttributeId id)
|
|
{
|
|
int err;
|
|
|
|
const char *name = DIR_GetDefaultAttribute (id)->name;
|
|
err = 0;
|
|
|
|
if (server->customAttributes)
|
|
{
|
|
DIR_Attribute *attrib = NULL;
|
|
XP_List *walkList = server->customAttributes;
|
|
while ((attrib = XP_ListNextObject(walkList)) != NULL)
|
|
{
|
|
if (attrib->id == id)
|
|
{
|
|
char *jsString = NULL;
|
|
if (0 == DIR_ConvertAttributeToPrefsString(attrib, &jsString))
|
|
{
|
|
DIR_SetStringPref (prefRoot, name, scratch, jsString, "");
|
|
XP_FREE(jsString);
|
|
return err;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/* This server doesn't have a custom attribute for the requested ID
|
|
* so set it to the null string just in case there's an ALL.JS setting
|
|
* or had a previous user value
|
|
*/
|
|
DIR_SetStringPref (prefRoot, name, scratch, "", "");
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
static int DIR_SaveCustomAttributes (const char *prefRoot, char *scratch, DIR_Server *server)
|
|
{
|
|
int err;
|
|
char *localScratch = (char*) XP_ALLOC(256);
|
|
err = 0;
|
|
|
|
XP_STRCPY (scratch, prefRoot);
|
|
XP_STRCAT (scratch, "attributes.");
|
|
if (localScratch)
|
|
{
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, cn);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, givenname);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, sn);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, mail);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, telephonenumber);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, o);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, ou);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, l);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, street);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, custom1);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, custom2);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, custom3);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, custom4);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, custom5);
|
|
DIR_SaveOneCustomAttribute (scratch, localScratch, server, auth);
|
|
XP_FREE(localScratch);
|
|
}
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
static int DIR_SaveCustomFilters (const char *prefRoot, char *scratch, DIR_Server *server)
|
|
{
|
|
int err;
|
|
char *localScratch = (char*) XP_ALLOC(256);
|
|
err = 0;
|
|
|
|
XP_STRCPY (scratch, prefRoot);
|
|
XP_STRCAT (scratch, "filter1.");
|
|
if (server->customFilters)
|
|
{
|
|
/* Save the custom filters into the JS prefs */
|
|
DIR_Filter *filter = NULL;
|
|
XP_List *walkList = server->customFilters;
|
|
if (localScratch)
|
|
{
|
|
while ((filter = XP_ListNextObject(walkList)) != NULL)
|
|
{
|
|
DIR_SetBoolPref (scratch, "repeatFilterForWords", localScratch,
|
|
(filter->flags & DIR_F_REPEAT_FILTER_FOR_TOKENS) != 0, kDefaultRepeatFilterForTokens);
|
|
DIR_SetStringPref (scratch, "string", localScratch, filter->string, kDefaultFilter);
|
|
}
|
|
XP_FREE(localScratch);
|
|
localScratch = NULL;
|
|
}
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
else
|
|
{
|
|
/* The DIR_Server object doesn't think it has any custom filters,
|
|
* so make sure the prefs settings are empty too
|
|
*/
|
|
DIR_SetBoolPref (scratch, "repeatFilterForWords", localScratch,
|
|
kDefaultRepeatFilterForTokens, kDefaultRepeatFilterForTokens);
|
|
DIR_SetStringPref (scratch, "string", localScratch, kDefaultFilter, kDefaultFilter);
|
|
}
|
|
|
|
if (localScratch) /* memory leak! I'm adding this to patch up the leak. */
|
|
XP_FREE(localScratch);
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
static int dir_SaveReplicationInfo (const char *prefRoot, char *scratch, DIR_Server *server)
|
|
{
|
|
int err = 0;
|
|
char *localScratch = (char*) XP_ALLOC(256);
|
|
if (!localScratch)
|
|
return MK_OUT_OF_MEMORY;
|
|
|
|
XP_STRCPY (scratch, prefRoot);
|
|
XP_STRCAT (scratch, "replication.");
|
|
|
|
DIR_SetBoolPref (scratch, "enabled", localScratch, DIR_TestFlag (server, DIR_REPLICATION_ENABLED), kDefaultReplicaEnabled);
|
|
|
|
if (server->replInfo)
|
|
{
|
|
char *excludedList = NULL;
|
|
int i;
|
|
int excludedLength = 0;
|
|
for (i = 0; i < server->replInfo->excludedAttributesCount; i++)
|
|
excludedLength += XP_STRLEN (server->replInfo->excludedAttributes[i]) + 2; /* +2 for ", " */
|
|
if (excludedLength)
|
|
{
|
|
excludedList = (char*) XP_ALLOC (excludedLength + 1);
|
|
if (excludedList)
|
|
{
|
|
excludedList[0] = '\0';
|
|
for (i = 0; i < server->replInfo->excludedAttributesCount; i++)
|
|
{
|
|
XP_STRCAT (excludedList, server->replInfo->excludedAttributes[i]);
|
|
XP_STRCAT (excludedList, ", ");
|
|
}
|
|
}
|
|
else
|
|
err = MK_OUT_OF_MEMORY;
|
|
}
|
|
|
|
DIR_SetStringPref (scratch, "excludedAttributes", localScratch, excludedList, kDefaultReplicaExcludedAttributes);
|
|
|
|
DIR_SetStringPref (scratch, "description", localScratch, server->replInfo->description, kDefaultReplicaDescription);
|
|
DIR_SetStringPref (scratch, "fileName", localScratch, server->replInfo->fileName, kDefaultReplicaFileName);
|
|
DIR_SetStringPref (scratch, "filter", localScratch, server->replInfo->filter, kDefaultReplicaFilter);
|
|
DIR_SetIntPref (scratch, "lastChangeNumber", localScratch, server->replInfo->lastChangeNumber, kDefaultReplicaChangeNumber);
|
|
DIR_SetStringPref (scratch, "dataVersion", localScratch, server->replInfo->dataVersion, kDefaultReplicaDataVersion);
|
|
}
|
|
|
|
XP_FREE(localScratch);
|
|
return err;
|
|
}
|
|
|
|
|
|
int DIR_SaveServerPreferences (XP_List *wholeList)
|
|
{
|
|
int i;
|
|
DIR_Server *s;
|
|
char * prefstring = NULL;
|
|
char * tempString = NULL;
|
|
|
|
if (wholeList)
|
|
{
|
|
prefstring = (char *) XP_ALLOC(128);
|
|
tempString = (char *) XP_ALLOC(256);
|
|
if (prefstring && tempString)
|
|
{
|
|
PREF_SetIntPref("ldap_1.number_of_directories", XP_ListCount (wholeList));
|
|
|
|
for (i = 1; i <= XP_ListCount(wholeList); i++)
|
|
{
|
|
s = (DIR_Server*) XP_ListGetObjectNum (wholeList, i);
|
|
if (s)
|
|
{
|
|
XP_SPRINTF(prefstring, "ldap_1.directory%i.", i);
|
|
|
|
DIR_SetStringPref (prefstring, "description", tempString, s->description, "");
|
|
DIR_SetStringPref (prefstring, "serverName", tempString, s->serverName, "");
|
|
DIR_SetStringPref (prefstring, "searchBase", tempString, s->searchBase, "");
|
|
DIR_SetStringPref (prefstring, "filename", tempString, s->fileName, "");
|
|
if (s->port == 0)
|
|
s->port = s->isSecure ? LDAPS_PORT : LDAP_PORT;
|
|
DIR_SetIntPref (prefstring, "port", tempString, s->port, s->isSecure ? LDAPS_PORT : LDAP_PORT);
|
|
DIR_SetIntPref (prefstring, "maxHits", tempString, s->maxHits, kDefaultMaxHits);
|
|
DIR_SetBoolPref (prefstring, "isSecure", tempString, s->isSecure, FALSE);
|
|
DIR_SetBoolPref (prefstring, "saveResults", tempString, s->saveResults, TRUE);
|
|
DIR_SetBoolPref (prefstring, "efficientWildcards", tempString, s->efficientWildcards, TRUE);
|
|
DIR_SetStringPref (prefstring, "searchString", tempString, s->lastSearchString, "");
|
|
DIR_SetIntPref (prefstring, "dirType", tempString, s->dirType, (int) LDAPDirectory);
|
|
DIR_SetBoolPref (prefstring, "isOffline", tempString, s->isOffline, kDefaultIsOffline);
|
|
|
|
DIR_SetBoolPref (prefstring, "autoComplete.enabled", tempString, DIR_TestFlag(s, DIR_AUTO_COMPLETE_ENABLED), kDefaultAutoCompleteEnabled);
|
|
DIR_SetIntPref (prefstring, "autoComplete.style", tempString, (int32) s->autoCompleteStyle, kDefaultAutoCompleteStyle);
|
|
|
|
DIR_SetBoolPref (prefstring, "utf8Disabled", tempString, DIR_TestFlag(s, DIR_UTF8_DISABLED), kDefaultUtf8Disabled);
|
|
|
|
DIR_SetBoolPref (prefstring, "enableAuth", tempString, s->enableAuth, kDefaultEnableAuth);
|
|
DIR_SetStringPref (prefstring, "authDn", tempString, s->authDn, NULL);
|
|
DIR_SetBoolPref (prefstring, "savePassword", tempString, s->savePassword, kDefaultSavePassword);
|
|
DIR_SetStringPref (prefstring, "password", tempString, s->savePassword ? s->password : "", "");
|
|
|
|
DIR_SetBoolPref (prefstring, "publicDirectory", tempString, DIR_TestFlag(s, DIR_LDAP_PUBLIC_DIRECTORY), kDefaultLdapPublicDirectory);
|
|
|
|
DIR_SaveCustomAttributes (prefstring, tempString, s);
|
|
DIR_SaveCustomFilters (prefstring, tempString, s);
|
|
dir_SaveReplicationInfo (prefstring, tempString, s);
|
|
|
|
DIR_SetStringPref (prefstring, "customDisplayUrl", tempString, s->customDisplayUrl, "");
|
|
}
|
|
}
|
|
XP_SPRINTF(tempString, "%i", tempString);
|
|
PREF_SetCharPref("ldap_1.end_of_directories", tempString);
|
|
}
|
|
|
|
XP_FREEIF (prefstring);
|
|
XP_FREEIF (tempString);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Functions for getting site-configurable preferences, from JavaScript if
|
|
* the site admin has provided them, else out of thin air.
|
|
*/
|
|
|
|
static DIR_DefaultAttribute *DIR_GetDefaultAttribute (DIR_AttributeId id)
|
|
{
|
|
int i = 0;
|
|
|
|
static DIR_DefaultAttribute defaults[16];
|
|
|
|
if (defaults[0].name == NULL)
|
|
{
|
|
defaults[0].id = cn;
|
|
defaults[0].resourceId = MK_LDAP_COMMON_NAME;
|
|
defaults[0].name = "cn";
|
|
|
|
defaults[1].id = givenname;
|
|
defaults[1].resourceId = MK_LDAP_GIVEN_NAME;
|
|
defaults[1].name = "givenName";
|
|
|
|
defaults[2].id = sn;
|
|
defaults[2].resourceId = MK_LDAP_SURNAME;
|
|
defaults[2].name = "sn";
|
|
|
|
defaults[3].id = mail;
|
|
defaults[3].resourceId = MK_LDAP_EMAIL_ADDRESS;
|
|
defaults[3].name = "mail";
|
|
|
|
defaults[4].id = telephonenumber;
|
|
defaults[4].resourceId = MK_LDAP_PHONE_NUMBER;
|
|
defaults[4].name = "telephoneNumber";
|
|
|
|
defaults[5].id = o;
|
|
defaults[5].resourceId = MK_LDAP_ORGANIZATION;
|
|
defaults[5].name = "o";
|
|
|
|
defaults[6].id = ou;
|
|
defaults[6].resourceId = MK_LDAP_ORG_UNIT;
|
|
defaults[6].name = "ou";
|
|
|
|
defaults[7].id = l;
|
|
defaults[7].resourceId = MK_LDAP_LOCALITY;
|
|
defaults[7].name = "l";
|
|
|
|
defaults[8].id = street;
|
|
defaults[8].resourceId = MK_LDAP_STREET;
|
|
defaults[8].name = "street";
|
|
|
|
defaults[9].id = custom1;
|
|
defaults[9].resourceId = MK_LDAP_CUSTOM1;
|
|
defaults[9].name = "custom1";
|
|
|
|
defaults[10].id = custom2;
|
|
defaults[10].resourceId = MK_LDAP_CUSTOM2;
|
|
defaults[10].name = "custom2";
|
|
|
|
defaults[11].id = custom3;
|
|
defaults[11].resourceId = MK_LDAP_CUSTOM3;
|
|
defaults[11].name = "custom3";
|
|
|
|
defaults[12].id = custom4;
|
|
defaults[12].resourceId = MK_LDAP_CUSTOM4;
|
|
defaults[12].name = "custom4";
|
|
|
|
defaults[13].id = custom5;
|
|
defaults[13].resourceId = MK_LDAP_CUSTOM5;
|
|
defaults[13].name = "custom5";
|
|
|
|
defaults[14].id = auth;
|
|
defaults[14].resourceId = MK_LDAP_EMAIL_ADDRESS;
|
|
defaults[14].name = "mail";
|
|
|
|
defaults[15].id = cn;
|
|
defaults[15].resourceId = 0;
|
|
defaults[15].name = NULL;
|
|
}
|
|
|
|
while (defaults[i].name)
|
|
{
|
|
if (defaults[i].id == id)
|
|
return &defaults[i];
|
|
i++;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
const char *DIR_GetAttributeName (DIR_Server *server, DIR_AttributeId id)
|
|
{
|
|
char *result = NULL;
|
|
|
|
/* First look in the custom attributes in case the attribute is overridden */
|
|
XP_List *list = server->customAttributes;
|
|
DIR_Attribute *walkList = NULL;
|
|
|
|
while ((walkList = XP_ListNextObject(list)) != NULL)
|
|
{
|
|
if (walkList->id == id)
|
|
result = walkList->prettyName;
|
|
}
|
|
|
|
/* If we didn't find it, look in our own static list of attributes */
|
|
if (!result)
|
|
{
|
|
DIR_DefaultAttribute *def;
|
|
if ((def = DIR_GetDefaultAttribute(id)) != NULL)
|
|
result = XP_GetString(def->resourceId);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
const char **DIR_GetAttributeStrings (DIR_Server *server, DIR_AttributeId id)
|
|
{
|
|
const char **result = NULL;
|
|
|
|
if (server && server->customAttributes)
|
|
{
|
|
/* First look in the custom attributes in case the attribute is overridden */
|
|
XP_List *list = server->customAttributes;
|
|
DIR_Attribute *walkList = NULL;
|
|
|
|
while ((walkList = XP_ListNextObject(list)) != NULL)
|
|
{
|
|
if (walkList->id == id)
|
|
result = (const char**)walkList->attrNames;
|
|
}
|
|
}
|
|
|
|
/* If we didn't find it, look in our own static list of attributes */
|
|
if (!result)
|
|
{
|
|
static const char *array[2];
|
|
array[0] = DIR_GetDefaultAttribute(id)->name;
|
|
array[1] = NULL;
|
|
result = (const char**)array;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
const char *DIR_GetFirstAttributeString (DIR_Server *server, DIR_AttributeId id)
|
|
{
|
|
const char **array = DIR_GetAttributeStrings (server, id);
|
|
return array[0];
|
|
}
|
|
|
|
const char *DIR_GetReplicationFilter (DIR_Server *server)
|
|
{
|
|
if (server && server->replInfo)
|
|
return server->replInfo->filter;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
const char *DIR_GetFilterString (DIR_Server *server)
|
|
{
|
|
DIR_Filter *filter = XP_ListTopObject (server->customFilters);
|
|
if (filter)
|
|
return filter->string;
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static DIR_Filter *DIR_LookupFilter (DIR_Server *server, const char *filter)
|
|
{
|
|
XP_List *list = server->customFilters;
|
|
DIR_Filter *walkFilter = NULL;
|
|
|
|
while ((walkFilter = XP_ListNextObject(list)) != NULL)
|
|
if (!XP_STRCASECMP(filter, walkFilter->string))
|
|
return walkFilter;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
XP_Bool DIR_RepeatFilterForTokens (DIR_Server *server, const char *filter)
|
|
{
|
|
const DIR_Filter *filterStruct = DIR_LookupFilter (server, filter);
|
|
if (filterStruct)
|
|
return (filterStruct->flags & DIR_F_REPEAT_FILTER_FOR_TOKENS) != 0;
|
|
|
|
return kDefaultRepeatFilterForTokens;
|
|
}
|
|
|
|
|
|
XP_Bool DIR_SubstStarsForSpaces (DIR_Server *server, const char *filter)
|
|
{
|
|
const DIR_Filter *filterStruct = DIR_LookupFilter (server, filter);
|
|
if (filterStruct)
|
|
return (filterStruct->flags & DIR_F_SUBST_STARS_FOR_SPACES) != 0;
|
|
|
|
return kDefaultSubstStarsForSpaces;
|
|
}
|
|
|
|
|
|
const char *DIR_GetTokenSeparators (DIR_Server *server)
|
|
{
|
|
return server->tokenSeps ? server->tokenSeps : kDefaultTokenSeps;
|
|
}
|
|
|
|
XP_Bool DIR_UseCustomAttribute (DIR_Server *server, DIR_AttributeId id)
|
|
{
|
|
XP_List *list = server->customAttributes;
|
|
DIR_Attribute *walkList = NULL;
|
|
|
|
while ((walkList = XP_ListNextObject(list)) != NULL)
|
|
{
|
|
if (walkList->id == id)
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
XP_Bool DIR_IsDnAttribute (DIR_Server *s, const char *attrib)
|
|
{
|
|
if (s && s->dnAttributes)
|
|
{
|
|
/* Look in the server object to see if there are prefs to tell
|
|
* us which attributes contain DNs
|
|
*/
|
|
int i;
|
|
for (i = 0; i < s->dnAttributesCount; i++)
|
|
{
|
|
if (!XP_STRCASECMP(attrib, s->dnAttributes[i]))
|
|
return TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* We have some default guesses about what attributes
|
|
* are likely to contain DNs
|
|
*/
|
|
switch (XP_TO_LOWER(attrib[0]))
|
|
{
|
|
case 'm':
|
|
if (!XP_STRCASECMP(attrib, "manager") ||
|
|
!XP_STRCASECMP(attrib, "member"))
|
|
return TRUE;
|
|
break;
|
|
case o:
|
|
if (!XP_STRCASECMP(attrib, "owner"))
|
|
return TRUE;
|
|
break;
|
|
case 'u':
|
|
if (!XP_STRCASECMP(attrib, "uniquemember"))
|
|
return TRUE;
|
|
break;
|
|
}
|
|
}
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
XP_Bool DIR_IsAttributeExcludedFromHtml (DIR_Server *s, const char *attrib)
|
|
{
|
|
if (s && s->suppressedAttributes)
|
|
{
|
|
/* Look in the server object to see if there are prefs to tell
|
|
* us which attributes shouldn't be shown in HTML
|
|
*/
|
|
int i;
|
|
for (i = 0; i < s->suppressedAttributesCount; i++)
|
|
{
|
|
if (!XP_STRCASECMP(attrib, s->suppressedAttributes[i]))
|
|
return TRUE;
|
|
}
|
|
}
|
|
/* else don't exclude it. By default we show everything */
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void dir_PushStringToPrefs (DIR_Server *s, char **curVal, const char *newVal, const char *name)
|
|
{
|
|
XP_List *servers = NULL;
|
|
XP_List *walkList = NULL;
|
|
DIR_Server *walkServer = NULL;
|
|
XP_Bool found = FALSE;
|
|
int idx = 0;
|
|
|
|
/* Don't do anything if we already have a value and it's the same as the new one */
|
|
if (curVal && *curVal)
|
|
{
|
|
if (!XP_STRCMP(*curVal, newVal))
|
|
return;
|
|
}
|
|
|
|
/* Find the server in the list so we can get its index.
|
|
* Can't use XP_GetNumFromObject because this DIR_Server isn't guaranteed
|
|
* to be in the FE's list -- it may be from a copy of the FE's list
|
|
*/
|
|
walkList = FE_GetDirServers ();
|
|
while (!found)
|
|
{
|
|
walkServer = (DIR_Server*) XP_ListNextObject (walkList);
|
|
if (!walkServer)
|
|
break;
|
|
found = DIR_AreServersSame (walkServer, s);
|
|
idx++;
|
|
}
|
|
|
|
if (found)
|
|
{
|
|
/* Build the name of the prefs string */
|
|
char buf[512];
|
|
char *prefRoot = PR_smprintf ("ldap_1.directory%d.", idx);
|
|
|
|
/* Set into the prefs */
|
|
if (prefRoot)
|
|
{
|
|
*curVal = XP_STRDUP(newVal);
|
|
DIR_SetStringPref (prefRoot, name, buf, *curVal, "");
|
|
|
|
XP_FREE(prefRoot);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
void DIR_SetAuthDN (DIR_Server *s, const char *dn)
|
|
{
|
|
dir_PushStringToPrefs (s, &(s->authDn), dn, "authDn");
|
|
}
|
|
|
|
|
|
void DIR_SetPassword (DIR_Server *s, const char *password)
|
|
{
|
|
dir_PushStringToPrefs (s, &(s->password), password, "password");
|
|
}
|
|
|
|
|
|
XP_Bool DIR_IsEscapedAttribute (DIR_Server *s, const char *attrib)
|
|
{
|
|
/* We're not exposing this setting in JS prefs right now, but in case we
|
|
* might want to in the future, leave the DIR_Server* in the prototype.
|
|
*/
|
|
|
|
switch (XP_TO_LOWER(attrib[0]))
|
|
{
|
|
case 'p':
|
|
if (!XP_STRCASECMP(attrib, "postaladdress"))
|
|
return TRUE;
|
|
break;
|
|
case 'f':
|
|
if (!XP_STRCASECMP(attrib, "facsimiletelephonenumber"))
|
|
return TRUE;
|
|
break;
|
|
case 'o':
|
|
if (!XP_STRCASECMP(attrib, "othermail"))
|
|
return TRUE;
|
|
break;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
char *DIR_Unescape (const char *src, XP_Bool makeHtml)
|
|
{
|
|
/* Borrowed from libnet\mkparse.c */
|
|
#define UNHEX(C) \
|
|
((C >= '0' && C <= '9') ? C - '0' : \
|
|
((C >= 'A' && C <= 'F') ? C - 'A' + 10 : \
|
|
((C >= 'a' && C <= 'f') ? C - 'a' + 10 : 0)))
|
|
|
|
char *dest = NULL;
|
|
int destLength = 0;
|
|
|
|
int dollarCount = 0;
|
|
int convertedLengthOfDollar = makeHtml ? 4 : 1;
|
|
|
|
const char *tmpSrc = src;
|
|
|
|
while (*tmpSrc)
|
|
if (*tmpSrc++ == '$')
|
|
dollarCount++;
|
|
|
|
destLength = XP_STRLEN(src) + (dollarCount * convertedLengthOfDollar);
|
|
dest = (char*) XP_ALLOC (destLength + 1);
|
|
if (dest)
|
|
{
|
|
char *tmpDst = dest;
|
|
*dest = '\0';
|
|
tmpSrc = src;
|
|
|
|
while (*tmpSrc)
|
|
{
|
|
switch (*tmpSrc)
|
|
{
|
|
case '$':
|
|
/* A dollar sign is a linebreak. This is easy for HTML, but if we're converting
|
|
* for the Address Book or something without multiple lines, just put in a space
|
|
*/
|
|
if (makeHtml)
|
|
{
|
|
*tmpDst++ = '<';
|
|
*tmpDst++ = 'B';
|
|
*tmpDst++ = 'R';
|
|
*tmpDst++ = '>';
|
|
}
|
|
else
|
|
*tmpDst++ = ' ';
|
|
break;
|
|
case '\\': {
|
|
/* A backslash indicates that two hex digits follow, which we're supposed to
|
|
* convert. The spec sez that '$', '#' and '\'' (single quote) must be encoded
|
|
* this way.
|
|
*/
|
|
XP_Bool didEscape = FALSE;
|
|
char c1 = *(tmpSrc + 1);
|
|
if (c1 && (XP_IS_DIGIT(c1) || XP_IS_ALPHA(c1)))
|
|
{
|
|
char c2 = *(tmpSrc + 2);
|
|
if (c2 && (XP_IS_DIGIT(c2) || XP_IS_ALPHA(c2)))
|
|
{
|
|
*tmpDst++ = (UNHEX(c1) << 4) | UNHEX(c2);
|
|
tmpSrc +=2;
|
|
didEscape = TRUE;
|
|
}
|
|
}
|
|
if (!didEscape)
|
|
*tmpDst++ = *tmpSrc;
|
|
}
|
|
break;
|
|
default:
|
|
/* Just a plain old char -- copy it over */
|
|
*tmpDst++ = *tmpSrc;
|
|
}
|
|
tmpSrc++;
|
|
}
|
|
*tmpDst = '\0';
|
|
}
|
|
|
|
return dest;
|
|
}
|
|
|
|
|
|
int DIR_ValidateRootDSE (DIR_Server *server, char *version, int32 first, int32 last)
|
|
{
|
|
/* Here we validate the replication info that the server has against the
|
|
* state of the local replica as stored in JS prefs.
|
|
*/
|
|
|
|
if (!server || !version)
|
|
return -1;
|
|
|
|
/* If the replication info file name in the server is NULL, that means the
|
|
* server has not been replicated. Reinitialize the change number and the
|
|
* data version and generate a file name for the replica database.
|
|
*/
|
|
if (!server->replInfo)
|
|
server->replInfo = (DIR_ReplicationInfo *) XP_CALLOC (1, sizeof (DIR_ReplicationInfo));
|
|
if (!server->replInfo)
|
|
return MK_OUT_OF_MEMORY;
|
|
|
|
if (!server->replInfo->fileName)
|
|
{
|
|
server->replInfo->lastChangeNumber = kDefaultReplicaChangeNumber;
|
|
XP_FREEIF(server->replInfo->filter);
|
|
server->replInfo->filter = XP_STRDUP(kDefaultReplicaFilter);
|
|
XP_FREEIF(server->replInfo->dataVersion);
|
|
server->replInfo->dataVersion = XP_STRDUP(version);
|
|
DIR_SetFileName (&(server->replInfo->fileName), server->serverName);
|
|
return 0;
|
|
}
|
|
|
|
/* There are three cases in which we should reinitialize the replica:
|
|
* 1) The data version of the server's DB is different than when we last
|
|
* saw it, which means that the first and last change number we know
|
|
* are totally meaningless.
|
|
* 2) Some changes have come and gone on the server since we last
|
|
* replicated. Since we have no way to know what those changes were,
|
|
* we have no way to get sync'd up with the current server state
|
|
* 3) We have already replicated changes that the server hasn't made yet.
|
|
* Not likely.
|
|
*/
|
|
if ( !server->replInfo->dataVersion || XP_STRCASECMP(version, server->replInfo->dataVersion)
|
|
|| first > server->replInfo->lastChangeNumber + 1
|
|
|| last < server->replInfo->lastChangeNumber)
|
|
{
|
|
server->replInfo->lastChangeNumber = kDefaultReplicaChangeNumber;
|
|
XP_FREEIF(server->replInfo->dataVersion);
|
|
server->replInfo->dataVersion = XP_STRDUP(version);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef MOZ_LDAP
|
|
int DIR_ParseRootDSE (DIR_Server *server, LDAP *ld, LDAPMessage *message)
|
|
{
|
|
char **values = NULL;
|
|
|
|
server->flags |= DIR_LDAP_ROOTDSE_PARSED;
|
|
server->flags &= ~(DIR_LDAP_VERSION3 | DIR_LDAP_VIRTUALLISTVIEW);
|
|
|
|
values = ldap_get_values (ld, message, "supportedLDAPVersion");
|
|
if (values && values[0])
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; values[i]; i++)
|
|
{
|
|
if (XP_ATOI (values[i]) == LDAP_VERSION3)
|
|
{
|
|
server->flags |= DIR_LDAP_VERSION3;
|
|
break;
|
|
}
|
|
}
|
|
ldap_value_free (values);
|
|
}
|
|
|
|
values = ldap_get_values (ld, message, "supportedControl");
|
|
if (values)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; values[i]; i++)
|
|
{
|
|
if (XP_STRCMP (values[i], LDAP_CONTROL_VLVREQUEST) == 0)
|
|
{
|
|
server->flags |= DIR_LDAP_VIRTUALLISTVIEW;
|
|
break;
|
|
}
|
|
}
|
|
ldap_value_free (values);
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
|
|
void DIR_SetAutoCompleteEnabled (XP_List *list, DIR_Server *server, XP_Bool enabled)
|
|
{
|
|
XP_ASSERT(server); /*list can be null*/
|
|
if (server)
|
|
{
|
|
DIR_Server *tmp;
|
|
|
|
if (enabled)
|
|
{
|
|
while (NULL != (tmp = XP_ListNextObject(list)))
|
|
DIR_ClearFlag (tmp, DIR_AUTO_COMPLETE_ENABLED);
|
|
DIR_SetFlag (server, DIR_AUTO_COMPLETE_ENABLED);
|
|
}
|
|
else
|
|
DIR_ClearFlag (server, DIR_AUTO_COMPLETE_ENABLED);
|
|
}
|
|
}
|
|
|
|
XP_Bool DIR_TestFlag (DIR_Server *server, uint32 flag)
|
|
{
|
|
if (server)
|
|
return 0 != (server->flags & flag);
|
|
return FALSE;
|
|
}
|
|
|
|
void DIR_SetFlag (DIR_Server *server, uint32 flag)
|
|
{
|
|
XP_ASSERT(server);
|
|
if (server)
|
|
server->flags |= flag;
|
|
}
|
|
|
|
void DIR_ClearFlag (DIR_Server *server, uint32 flag)
|
|
{
|
|
XP_ASSERT(server);
|
|
if (server)
|
|
server->flags &= ~flag;
|
|
}
|
|
|
|
|
|
void DIR_ForceFlag (DIR_Server *server, uint32 flag, XP_Bool setIt)
|
|
{
|
|
XP_ASSERT(server);
|
|
if (server)
|
|
{
|
|
if (setIt)
|
|
server->flags |= flag;
|
|
else
|
|
server->flags &= ~flag;
|
|
}
|
|
}
|
|
|
|
|
|
/* Centralize this charset conversion so everyone can do the UTF8 conversion
|
|
* in the same way. Also, if someone ever makes us do T.61 or some other silly
|
|
* thing, we can use these bottlenecks
|
|
*/
|
|
|
|
char *DIR_ConvertToServerCharSet (DIR_Server *server, char *src, int16 srcCsid)
|
|
{
|
|
if (server && (server->flags & DIR_UTF8_DISABLED))
|
|
return XP_STRDUP (src);
|
|
else
|
|
return (char*) INTL_ConvertLineWithoutAutoDetect (srcCsid, CS_UTF8, (unsigned char*) src, XP_STRLEN(src));
|
|
}
|
|
|
|
char *DIR_ConvertFromServerCharSet (DIR_Server *server, char *src, int16 destCsid)
|
|
{
|
|
if (server && (server->flags & DIR_UTF8_DISABLED))
|
|
return XP_STRDUP (src);
|
|
else
|
|
return (char*) INTL_ConvertLineWithoutAutoDetect (CS_UTF8, destCsid, (unsigned char*) src, XP_STRLEN(src));
|
|
}
|
|
|
|
|
|
char *DIR_BuildUrl (DIR_Server *server, const char *dn, XP_Bool forAddToAB)
|
|
{
|
|
char *url = NULL;
|
|
char *escapedDn = NET_Escape (dn, URL_XALPHAS);
|
|
if (escapedDn)
|
|
{
|
|
if (!forAddToAB && server->customDisplayUrl && server->customDisplayUrl[0])
|
|
{
|
|
/* Allow users to customize the URL we run when we open an LDAP entry. This
|
|
* is intended to appease the people who want extensive customization of the
|
|
* HTML we generate for LDAP. We're sidestepping the issue by allowing them
|
|
* to easily plug in DSGW, the LDAP-to-HTTP gateway, which already has
|
|
* very extensive template support.
|
|
*/
|
|
url = PR_smprintf (server->customDisplayUrl, escapedDn);
|
|
}
|
|
else
|
|
{
|
|
/* The default case, where we run an LDAP URL in a browser window
|
|
*/
|
|
char *urlTemplate = "%s//%s/%s";
|
|
char *urlPortTemplate = "%s//%s:%d/%s";
|
|
char *urlScheme = NULL;
|
|
int port = server->port;
|
|
int standardPort = server->isSecure ? LDAPS_PORT : LDAP_PORT;
|
|
|
|
if (server->isSecure)
|
|
urlScheme = forAddToAB ? "addbook-ldaps:" : "ldaps:";
|
|
else
|
|
urlScheme = forAddToAB ? "addbook-ldap:" : "ldap:";
|
|
|
|
if (port == standardPort)
|
|
url = PR_smprintf (urlTemplate, urlScheme, server->serverName, escapedDn);
|
|
else
|
|
url = PR_smprintf (urlPortTemplate, urlScheme, server->serverName, port, escapedDn);
|
|
|
|
}
|
|
XP_FREE (escapedDn);
|
|
}
|
|
return url;
|
|
}
|
|
#endif /* !MOZADDRSTANDALONE */
|
|
|
|
#endif /* !MOZ_MAIL_NEWS */
|
|
|
|
|
|
|