/*
* Copyright (C) 2005-2011 Alfresco Software Limited.
*
* This file is part of Alfresco
*
* Alfresco is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Alfresco is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with Alfresco. If not, see .
*/
package org.alfresco.repo.domain.solr;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import junit.framework.TestCase;
import org.alfresco.model.ContentModel;
import org.alfresco.repo.domain.node.Node;
import org.alfresco.repo.domain.permissions.AclDAO;
import org.alfresco.repo.security.authentication.AuthenticationComponent;
import org.alfresco.repo.security.permissions.ACLType;
import org.alfresco.repo.security.permissions.AccessControlListProperties;
import org.alfresco.repo.solr.Acl;
import org.alfresco.repo.solr.AclChangeSet;
import org.alfresco.repo.solr.NodeParameters;
import org.alfresco.repo.solr.Transaction;
import org.alfresco.repo.transaction.RetryingTransactionHelper.RetryingTransactionCallback;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.service.cmr.repository.StoreRef;
import org.alfresco.service.cmr.security.MutableAuthenticationService;
import org.alfresco.service.cmr.security.PersonService;
import org.alfresco.service.transaction.TransactionService;
import org.alfresco.test_category.OwnJVMTestsCategory;
import org.alfresco.util.ApplicationContextHelper;
import org.alfresco.util.PropertyMap;
import org.junit.experimental.categories.Category;
import org.springframework.context.ConfigurableApplicationContext;
/**
* Tests for the SOLR DAO
*
* @since 4.0
*/
@Category(OwnJVMTestsCategory.class)
public class SOLRDAOTest extends TestCase
{
private ConfigurableApplicationContext ctx = (ConfigurableApplicationContext) ApplicationContextHelper.getApplicationContext();
private AuthenticationComponent authenticationComponent;
private MutableAuthenticationService authenticationService;
private PersonService personService;
private TransactionService transactionService;
private NodeService nodeService;
private AclDAO aclDaoComponent;
private SOLRDAO solrDAO;
@Override
public void setUp() throws Exception
{
solrDAO = (SOLRDAO)ctx.getBean("solrDAO");
authenticationComponent = (AuthenticationComponent)ctx.getBean("authenticationComponent");
authenticationService = (MutableAuthenticationService)ctx.getBean("authenticationService");
personService = (PersonService)ctx.getBean("PersonService");
transactionService = (TransactionService)ctx.getBean("transactionComponent");
nodeService = (NodeService) ctx.getBean("NodeService");
aclDaoComponent = (AclDAO) ctx.getBean("aclDAO");
authenticationComponent.setSystemUserAsCurrentUser();
}
private List getNodes(final NodeParameters nodeParameters)
{
return transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback>()
{
@Override
public List execute() throws Throwable
{
return solrDAO.getNodes(nodeParameters);
}
}, true);
}
private List getAcls(final List aclChangeSetIds, final Long minAclId, final int maxResults)
{
return transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback>()
{
@Override
public List execute() throws Throwable
{
return solrDAO.getAcls(aclChangeSetIds, minAclId, maxResults);
}
}, true);
}
private List getTransactions(
final Long minTxnId, final Long fromCommitTime,
final Long maxTxnId, final Long toCommitTime,
final int maxResults)
{
return transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback>()
{
@Override
public List execute() throws Throwable
{
return solrDAO.getTransactions(minTxnId, fromCommitTime, maxTxnId, toCommitTime, maxResults);
}
}, true);
}
private List getAclChangeSets(
final Long minAclChangeSetId, final Long fromCommitTime,
final Long maxAclChangeSetId, final Long toCommitTime,
final int maxResults)
{
return transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback>()
{
@Override
public List execute() throws Throwable
{
return solrDAO.getAclChangeSets(minAclChangeSetId, fromCommitTime, maxAclChangeSetId, toCommitTime, maxResults);
}
}, true);
}
public void testQueryChangeSets_NoLimit()
{
long startTime = System.currentTimeMillis() - (5 * 60000L);
try
{
getAclChangeSets(null, startTime, null, null, 0);
fail("Must have result limit");
}
catch (IllegalArgumentException e)
{
// Expected
}
}
public void testQueryChangeSets_Time()
{
long startTime = System.currentTimeMillis() + (5 * 60000L); // The future
List results = getAclChangeSets(null, startTime, null, null, 50);
assertTrue("ChangeSet count not limited", results.size() == 0);
}
public void testQueryChangeSets_Limit()
{
List results = getAclChangeSets(null, 0L, null, null, 50);
assertTrue("Transaction count not limited", results.size() <= 50);
}
/**
* Argument checks.
*/
public void testQueryAcls_Arguments()
{
try
{
// No IDs
getAcls(Collections.emptyList(), null, 50);
fail("Expected IllegalArgumentException");
}
catch (IllegalArgumentException e)
{
// Expected
}
}
public void testQueryAcls_All()
{
// Do a query for some changesets
List aclChangeSets = getAclChangeSets(null, 0L, null, null, 50);
// Choose some changesets with changes
int aclTotal = 0;
Iterator aclChangeSetsIterator = aclChangeSets.iterator();
while (aclChangeSetsIterator.hasNext())
{
AclChangeSet aclChangeSet = aclChangeSetsIterator.next();
if (aclChangeSet.getAclCount() == 0)
{
aclChangeSetsIterator.remove();
}
else
{
aclTotal += aclChangeSet.getAclCount();
}
}
// Stop if we don't have ACLs
if (aclTotal == 0)
{
return;
}
List aclChangeSetIds = toIds(aclChangeSets);
// Now use those to query for details
List acls = getAcls(aclChangeSetIds, null, 1000);
// Check that the ACL ChangeSet IDs are correct
Set aclChangeSetIdsSet = new HashSet(aclChangeSetIds);
for (Acl acl : acls)
{
Long aclChangeSetId = acl.getAclChangeSetId();
assertTrue("ACL ChangeSet ID not in original list", aclChangeSetIdsSet.contains(aclChangeSetId));
}
}
public void testQueryAcls_Single()
{
List aclChangeSets = getAclChangeSets(null, 0L, null, null, 1000);
// Find one with multiple ALCs
AclChangeSet aclChangeSet = null;
for (AclChangeSet aclChangeSetLoop : aclChangeSets)
{
if (aclChangeSetLoop.getAclCount() > 1)
{
aclChangeSet = aclChangeSetLoop;
break;
}
}
if (aclChangeSet == null)
{
// Nothing to test: Very unlikely
return;
}
// Loop a few times and check that the count is correct
Long aclChangeSetId = aclChangeSet.getId();
List aclChangeSetIds = Collections.singletonList(aclChangeSetId);
int aclCount = aclChangeSet.getAclCount();
int totalAclCount = 0;
Long minAclId = null;
while (true)
{
List acls = getAcls(aclChangeSetIds, minAclId, 1);
if (acls.size() == 0)
{
break;
}
if(acls.size() == 1)
{
// OK single acl
}
else if(acls.size() == 2)
{
// definning has unlinkfd shared acl
assertEquals("Not a defining and shared pair", acls.get(0).getInheritedId(), acls.get(1).getId());
}
else
{
fail("More then two acls");
}
totalAclCount++;;
minAclId = acls.get(0).getId() + 1;
}
// This may not be true - it depands on lazy/eager shared acl creation.
//assertEquals("Expected to page to exact number of results", aclCount, totalAclCount);
}
private List toIds(List aclChangeSets)
{
List ids = new ArrayList(aclChangeSets.size());
for (AclChangeSet aclChangeSet : aclChangeSets)
{
ids.add(aclChangeSet.getId());
}
return ids;
}
public void testQueryTransactions_NoLimit()
{
long startTime = System.currentTimeMillis() - (5 * 60000L);
try
{
getTransactions(null, startTime, null, null, 0);
fail("Must have result limit");
}
catch (IllegalArgumentException e)
{
// Expected
}
}
public void testQueryTransactions_Time()
{
long startTime = System.currentTimeMillis() + (5 * 60000L); // The future
List results = getTransactions(null, startTime, null, null, 50);
assertTrue("Transaction count not limited", results.size() == 0);
}
public void testQueryTransactions_Limit()
{
List results = getTransactions(null, 0L, null, null, 50);
assertTrue("Transaction count not limited", results.size() <= 50);
}
public void testGetNodesSimple()
{
long startTime = 0L;
List txns = getTransactions(null, startTime, null, null, 500);
List txnIds = toTxnIds(txns);
NodeParameters nodeParameters = new NodeParameters();
nodeParameters.setTransactionIds(txnIds);
nodeParameters.setStoreProtocol(StoreRef.PROTOCOL_WORKSPACE);
nodeParameters.setStoreIdentifier(StoreRef.STORE_REF_WORKSPACE_SPACESSTORE.getIdentifier());
List nodes = getNodes(nodeParameters);
assertTrue("Expect 'some' nodes associated with txns", nodes.size() > 0);
}
public void testGetNodesForStore()
{
List txns = getTransactions(null, null, null, null, 500);
List txnIds = toTxnIds(txns);
NodeParameters nodeParameters = new NodeParameters();
nodeParameters.setTransactionIds(txnIds);
List nodes = getNodes(nodeParameters);
assertTrue("Expect 'some' nodes associated with txns", nodes.size() > 0);
}
public void testGetNodesForTxnRange()
{
List txns = getTransactions(null, null, null, null, 500);
List txnIds = toTxnIds(txns);
// Only works if there are transactions
if (txnIds.size() < 2)
{
// Nothing to test
return;
}
NodeParameters nodeParameters = new NodeParameters();
nodeParameters.setFromTxnId(txnIds.get(0));
nodeParameters.setToTxnId(txnIds.get(1));
List nodes = getNodes(nodeParameters);
assertTrue("Expect 'some' nodes associated with txns", nodes.size() > 0);
}
private List toTxnIds(List txns)
{
List txnIds = new ArrayList(txns.size());
for(Transaction txn : txns)
{
txnIds.add(txn.getId());
}
return txnIds;
}
private boolean containsAclId(List acls, Long id)
{
for (Acl acl : acls)
{
if (acl.getId().equals(id))
{
return true;
}
}
return false;
}
/**
* MNT-11107: during User Home creation Shared Acl is created that is inherited from Acl
* which is assigned to User Home folder node. This Shared Acl is not assigned to any node.
* However, solrDAO should be able to find it so that it can be indexed.
*/
public void testInheritedAclIndexing() throws Exception
{
final String USER_MNT11107 = "TestUserMNT11107";
Long sharedAclId = transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback()
{
@Override
public Long execute() throws Throwable
{
// Create a user
if (authenticationService.authenticationExists(USER_MNT11107))
authenticationService.deleteAuthentication(USER_MNT11107);
if (personService.personExists(USER_MNT11107))
personService.deletePerson(USER_MNT11107);
authenticationService.createAuthentication(USER_MNT11107, "PWD".toCharArray());
PropertyMap personProperties = new PropertyMap();
personProperties.put(ContentModel.PROP_USERNAME, USER_MNT11107);
personProperties.put(ContentModel.PROP_AUTHORITY_DISPLAY_NAME, "title" + USER_MNT11107);
personProperties.put(ContentModel.PROP_FIRSTNAME, "firstName");
personProperties.put(ContentModel.PROP_LASTNAME, "lastName");
personProperties.put(ContentModel.PROP_EMAIL, USER_MNT11107 + "@example.com");
personProperties.put(ContentModel.PROP_JOBTITLE, "jobTitle");
NodeRef person = personService.createPerson(personProperties);
NodeRef testUserHomeFolder = (NodeRef) nodeService.getProperty(person, ContentModel.PROP_HOMEFOLDER);
assertNotNull("testUserHomeFolder is null", testUserHomeFolder);
Long aclIdForUserHomeFolder = nodeService.getNodeAclId(testUserHomeFolder);
Long inheritedAclId = aclDaoComponent.getInheritedAccessControlList(aclIdForUserHomeFolder);
return inheritedAclId;
}
});
try
{
assertNotNull("Acl for User Home folder should have inherited Acl", sharedAclId);
AccessControlListProperties aclProps = aclDaoComponent.getAccessControlListProperties(sharedAclId);
assertEquals("Inherited Acl should be of SHARED type", aclProps.getAclType(), ACLType.SHARED);
assertTrue("Acl should inherit", aclProps.getInherits());
assertNotNull("AclChangeSet for inherited Acl should not be NULL", aclProps.getAclChangeSetId());
List aclChangeSetIds = new ArrayList();
aclChangeSetIds.add(aclProps.getAclChangeSetId());
List acls = solrDAO.getAcls(aclChangeSetIds, null, 1000);
assertTrue("Shared Acl should be found by solrDAO so that it can be indexed", containsAclId(acls, sharedAclId));
}
finally
{
transactionService.getRetryingTransactionHelper().doInTransaction(new RetryingTransactionCallback()
{
@Override
public Void execute() throws Throwable
{
// Tidy up
authenticationComponent.setSystemUserAsCurrentUser();
authenticationService.deleteAuthentication(USER_MNT11107);
personService.deletePerson(USER_MNT11107);
return null;
}
});
}
}
}