/* * 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; } }); } } }