mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-08-07 17:49:17 +00:00
Merged V3.2 to HEAD
16662: LDAP sync: improved group association filtering, referential integrity checking, deletion strategy and performance tuning of batch sizes 16648: ETHREEOH-2752: Improved ticket validation fix - Invalidate user's tickets during person deletion rather than validation or it can mess up chained validation 16647: ETHREEOH-2534: Fixed Sharepoint NTLM authentication - user details were never getting cached in the session 16579: Small improvement to LDAP error reporting - Committed errors counted before successes in a logging interval 16515: LDAP sync performance - Improved full sync strategy - run differential queries to work out required updates/additions and full queries to work out required deletions. Saves updating unchanged nodes. - Use a TreeSet rather than a HashSet to gather group associations in an attempt to avoid blowing the heap size 16498: More LDAP performance improvements - Uses thread pool with 4 worker threads and blocking queue to process returned results. The number of worker threads can be controlled by the synchronization.workerThreads property. - Switched LDAP connection pooling back on again - Group Associations processsed individually so that errors are collated and we get a better idea of their throughput - Fixed potential bug. Group membership resolution done with isolated LDAP context to avoid cookies from paging creeping in. 16424: Try switching off LDAP connection pooling to see if it works better with our flaky server. 16414: Further LDAP fault tolerance - Log causes of group member resolution failures where possible 16413: More fault tolerance for LDAP sync - Always commit last sync times before overall sync is complete to avoid the 'forgetting' of differential sync information - DN comparisons should be case insensitive to avoid issues resolving DNs to user and group IDs 16398: Improved monitoring and fault tolerance for LDAP sync - When the batch is complete a summary of the number of errors and the last error stack trace will be logged at ERROR level - Each individual error is logged at WARN level and progress information (including % complete) is collated and logged at INFO level after a configurable interval - In the Enterprise Edition all metrics can be monitored in real time through JMX - Sanity testing to be performed by Mike! 16319: Merged HEAD to V3.2 16316: ALFCOM-3397: JBoss 5 compatibility fix - Relative paths used by LDAP subsystem configuration weren't being resolved correctly - See also https://jira.jboss.org/jira/browse/JBAS-6548 and https://jira.springsource.org/browse/SPR-5120 16272: ETHREEOH-2752: Once more with feeling! 16261: ETHREEOH-2752: Correct exception propagation. 16260: ETHREEOH-2752: Fix ticket validation - Current ticket was getting forgotten by previous fix - Person validation in CHECK mode now done AFTER the current user is set, so that the current ticket is remembered 16243: ETHREEOH-2752: Improve ticket validation used by all authentication filters - Now takes into account whether person actually exists or not - Tickets for non-nonexistent persons are now considered invalid and cached session information is invalidated - New BaseAuthenticationFilter superclass for all authentication filters - Improved fix to ETHREEOH-2839: WebDAV user is cached consistently using a different session attribute from the Web Client 16233: ETHREEOH-2754: Correction to previous checkin. - relogin for SSO authentication, logout for normal login page - logout is default 16232: ETHREEOH-2754: Log Out Action outcome passed as a parameter - relogin for SSO authentication, login for normal login page - Means the log out link always leads to the correct place, even when the session has expired - Also lowered ticket validation error logging to DEBUG level to avoid unnecessary noise in the logs from expired sessions 16220: ETHREEOH-2839: Fixed potential ClassCastExceptions when Alfresco accessed via WebDAV and Web Client links in same browser - WebDAV side no longer directly casts session user to a WebDAVUser - ContextListener no longer casts session user to web client user - Web client side will 'promote' session user to a web client User if necessary via AuthenticationHelper - All authentication filters made to use appropriate AuthenticationHelper methods 16211: ETHREEOH-2835: LDAP sync batches user and group deletions as well as creations - Also improved logging of sync failures 16197: ETHREEOH-2782: LDAP subsystems now support search-based user DN resolution - When ldap.authentication.userNameFormat isn't set (now the default) converts a user ID to a DN by running ldap.synchronization.personQuery with an extra condition tacked on the end to find the user by ID - Structured directories and authentication by attributes not in the DN such as email address now supported 16189: ALFCOM-3283: Prevent errors when user accepts an invite when not logged in - new isGuest attribute propagated to user object - header component (used by accept-invite page) needs to avoid calling prefs and site webscripts for guest user - Conditional stuff in header template changed to use user.isGuest git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@16896 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
@@ -24,6 +24,8 @@
|
||||
*/
|
||||
package org.alfresco.repo.security.sync;
|
||||
|
||||
import java.util.AbstractCollection;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.Collection;
|
||||
import java.util.Collections;
|
||||
@@ -31,6 +33,7 @@ import java.util.Date;
|
||||
import java.util.HashSet;
|
||||
import java.util.Iterator;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
|
||||
@@ -46,6 +49,7 @@ import org.alfresco.service.cmr.repository.NodeService;
|
||||
import org.alfresco.service.cmr.security.AuthorityService;
|
||||
import org.alfresco.service.cmr.security.AuthorityType;
|
||||
import org.alfresco.service.cmr.security.PersonService;
|
||||
import org.alfresco.util.GUID;
|
||||
import org.alfresco.util.PropertyMap;
|
||||
import org.springframework.context.ApplicationContext;
|
||||
import org.springframework.context.support.ClassPathXmlApplicationContext;
|
||||
@@ -65,7 +69,7 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
"classpath:alfresco/application-context.xml", "classpath:sync-test-context.xml"
|
||||
};
|
||||
|
||||
/** The Spring application context */
|
||||
/** The Spring application context. */
|
||||
private static ApplicationContext context = new ClassPathXmlApplicationContext(
|
||||
ChainingUserRegistrySynchronizerTest.CONFIG_LOCATIONS);
|
||||
|
||||
@@ -90,6 +94,10 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
/** The retrying transaction helper. */
|
||||
private RetryingTransactionHelper retryingTransactionHelper;
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see junit.framework.TestCase#setUp()
|
||||
*/
|
||||
@Override
|
||||
protected void setUp() throws Exception
|
||||
{
|
||||
@@ -110,6 +118,10 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
.getBean("retryingTransactionHelper");
|
||||
}
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see junit.framework.TestCase#tearDown()
|
||||
*/
|
||||
@Override
|
||||
protected void tearDown() throws Exception
|
||||
{
|
||||
@@ -191,6 +203,12 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Tear down test users and groups.
|
||||
*
|
||||
* @throws Exception
|
||||
* the exception
|
||||
*/
|
||||
private void tearDownTestUsersAndGroups() throws Exception
|
||||
{
|
||||
// Wipe out everything that was in Z1 and Z2
|
||||
@@ -301,7 +319,7 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
* <pre>
|
||||
* Z1
|
||||
* G1 - U6
|
||||
* G2 -
|
||||
* G2 -
|
||||
* G3 - U2, G5 - U6
|
||||
* G6 - u3
|
||||
*
|
||||
@@ -365,7 +383,30 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
}
|
||||
|
||||
/**
|
||||
* Constructs a description of a test group
|
||||
* Tests synchronization with a zone with a larger volume of authorities.
|
||||
*
|
||||
* @throws Exception
|
||||
* the exception
|
||||
*/
|
||||
public void testVolume() throws Exception
|
||||
{
|
||||
List<NodeDescription> persons = new ArrayList<NodeDescription>(new RandomPersonCollection(100));
|
||||
List<NodeDescription> groups = new ArrayList<NodeDescription>(new RandomGroupCollection(100, persons));
|
||||
this.applicationContextManager.setUserRegistries(new MockUserRegistry("Z0", persons, groups));
|
||||
this.retryingTransactionHelper.doInTransaction(new RetryingTransactionCallback<Object>()
|
||||
{
|
||||
|
||||
public Object execute() throws Throwable
|
||||
{
|
||||
ChainingUserRegistrySynchronizerTest.this.synchronizer.synchronize(true, true);
|
||||
return null;
|
||||
}
|
||||
});
|
||||
tearDownTestUsersAndGroups();
|
||||
}
|
||||
|
||||
/**
|
||||
* Constructs a description of a test group.
|
||||
*
|
||||
* @param name
|
||||
* the name
|
||||
@@ -375,9 +416,10 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
*/
|
||||
private NodeDescription newGroup(String name, String... members)
|
||||
{
|
||||
NodeDescription group = new NodeDescription();
|
||||
String longName = longName(name);
|
||||
NodeDescription group = new NodeDescription(longName);
|
||||
PropertyMap properties = group.getProperties();
|
||||
properties.put(ContentModel.PROP_AUTHORITY_NAME, longName(name));
|
||||
properties.put(ContentModel.PROP_AUTHORITY_NAME, longName);
|
||||
if (members.length > 0)
|
||||
{
|
||||
Set<String> assocs = group.getChildAssociations();
|
||||
@@ -413,7 +455,7 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
*/
|
||||
private NodeDescription newPerson(String userName, String email)
|
||||
{
|
||||
NodeDescription person = new NodeDescription();
|
||||
NodeDescription person = new NodeDescription(userName);
|
||||
PropertyMap properties = person.getProperties();
|
||||
properties.put(ContentModel.PROP_USERNAME, userName);
|
||||
properties.put(ContentModel.PROP_FIRSTNAME, userName + "F");
|
||||
@@ -481,7 +523,7 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
}
|
||||
|
||||
/**
|
||||
* Asserts that a person's email has the expected value
|
||||
* Asserts that a person's email has the expected value.
|
||||
*
|
||||
* @param personName
|
||||
* the person name
|
||||
@@ -518,10 +560,27 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
private String zoneId;
|
||||
|
||||
/** The persons. */
|
||||
private NodeDescription[] persons;
|
||||
private Collection<NodeDescription> persons;
|
||||
|
||||
/** The groups. */
|
||||
private NodeDescription[] groups;
|
||||
private Collection<NodeDescription> groups;
|
||||
|
||||
/**
|
||||
* Instantiates a new mock user registry.
|
||||
*
|
||||
* @param zoneId
|
||||
* the zone id
|
||||
* @param persons
|
||||
* the persons
|
||||
* @param groups
|
||||
* the groups
|
||||
*/
|
||||
public MockUserRegistry(String zoneId, Collection<NodeDescription> persons, Collection<NodeDescription> groups)
|
||||
{
|
||||
this.zoneId = zoneId;
|
||||
this.persons = persons;
|
||||
this.groups = groups;
|
||||
}
|
||||
|
||||
/**
|
||||
* Instantiates a new mock user registry.
|
||||
@@ -535,9 +594,7 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
*/
|
||||
public MockUserRegistry(String zoneId, NodeDescription[] persons, NodeDescription[] groups)
|
||||
{
|
||||
this.zoneId = zoneId;
|
||||
this.persons = persons;
|
||||
this.groups = groups;
|
||||
this(zoneId, Arrays.asList(persons), Arrays.asList(groups));
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -552,20 +609,32 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see org.alfresco.repo.security.sync.UserRegistry#getGroups(java.util.Date)
|
||||
* @see org.alfresco.repo.security.sync.UserRegistry#getGroups(java.util.Date, java.util.Set, boolean)
|
||||
*/
|
||||
public Iterator<NodeDescription> getGroups(Date modifiedSince)
|
||||
public Collection<NodeDescription> getGroups(Date modifiedSince, Set<String> candidateAuthoritiesForDeletion,
|
||||
boolean prune)
|
||||
{
|
||||
return Arrays.asList(this.groups).iterator();
|
||||
if (prune)
|
||||
{
|
||||
for (NodeDescription person : this.persons)
|
||||
{
|
||||
candidateAuthoritiesForDeletion.remove(person.getProperties().get(ContentModel.PROP_USERNAME));
|
||||
}
|
||||
for (NodeDescription group : this.groups)
|
||||
{
|
||||
candidateAuthoritiesForDeletion.remove(group.getProperties().get(ContentModel.PROP_AUTHORITY_NAME));
|
||||
}
|
||||
}
|
||||
return this.groups;
|
||||
}
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see org.alfresco.repo.security.sync.UserRegistry#getPersons(java.util.Date)
|
||||
*/
|
||||
public Iterator<NodeDescription> getPersons(Date modifiedSince)
|
||||
public Collection<NodeDescription> getPersons(Date modifiedSince)
|
||||
{
|
||||
return Arrays.asList(this.persons).iterator();
|
||||
return this.persons;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -615,4 +684,144 @@ public class ChainingUserRegistrySynchronizerTest extends TestCase
|
||||
return this.contexts.keySet();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* A collection whose iterator returns randomly generated persons.
|
||||
*/
|
||||
public class RandomPersonCollection extends AbstractCollection<NodeDescription>
|
||||
{
|
||||
|
||||
/** The collection size. */
|
||||
private final int size;
|
||||
|
||||
/**
|
||||
* The Constructor.
|
||||
*
|
||||
* @param size
|
||||
* the collection size
|
||||
*/
|
||||
public RandomPersonCollection(int size)
|
||||
{
|
||||
this.size = size;
|
||||
}
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see java.util.AbstractCollection#iterator()
|
||||
*/
|
||||
@Override
|
||||
public Iterator<NodeDescription> iterator()
|
||||
{
|
||||
return new Iterator<NodeDescription>()
|
||||
{
|
||||
|
||||
private int pos;
|
||||
|
||||
public boolean hasNext()
|
||||
{
|
||||
return pos < size;
|
||||
}
|
||||
|
||||
public NodeDescription next()
|
||||
{
|
||||
pos++;
|
||||
return newPerson("U" + GUID.generate());
|
||||
}
|
||||
|
||||
public void remove()
|
||||
{
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see java.util.AbstractCollection#size()
|
||||
*/
|
||||
@Override
|
||||
public int size()
|
||||
{
|
||||
return size;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* A collection whose iterator returns randomly generated groups with random associations to a given list of
|
||||
* persons.
|
||||
*/
|
||||
public class RandomGroupCollection extends AbstractCollection<NodeDescription>
|
||||
{
|
||||
|
||||
/** The collection size. */
|
||||
private final int size;
|
||||
|
||||
/** The persons. */
|
||||
private final List<NodeDescription> persons;
|
||||
|
||||
/**
|
||||
* The Constructor.
|
||||
*
|
||||
* @param size
|
||||
* the collection size
|
||||
* @param persons
|
||||
* the persons
|
||||
*/
|
||||
public RandomGroupCollection(int size, List<NodeDescription> persons)
|
||||
{
|
||||
this.size = size;
|
||||
this.persons = persons;
|
||||
}
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see java.util.AbstractCollection#iterator()
|
||||
*/
|
||||
@Override
|
||||
public Iterator<NodeDescription> iterator()
|
||||
{
|
||||
return new Iterator<NodeDescription>()
|
||||
{
|
||||
|
||||
private int pos;
|
||||
|
||||
public boolean hasNext()
|
||||
{
|
||||
return pos < size;
|
||||
}
|
||||
|
||||
public NodeDescription next()
|
||||
{
|
||||
pos++;
|
||||
String[] personNames = new String[10];
|
||||
for (int i = 0; i < personNames.length; i++)
|
||||
{
|
||||
personNames[i] = (String) persons.get((int) (Math.random() * (double) (persons.size() - 1)))
|
||||
.getProperties().get(ContentModel.PROP_USERNAME);
|
||||
}
|
||||
return newGroup("G" + GUID.generate(), personNames);
|
||||
}
|
||||
|
||||
public void remove()
|
||||
{
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
/*
|
||||
* (non-Javadoc)
|
||||
* @see java.util.AbstractCollection#size()
|
||||
*/
|
||||
@Override
|
||||
public int size()
|
||||
{
|
||||
return size;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
Reference in New Issue
Block a user