mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-07-31 17:39:05 +00:00
Merged HEAD-QA to HEAD (4.2) (including moving test classes into separate folders)
51903 to 54309 git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@54310 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
@@ -1,196 +0,0 @@
|
||||
/*
|
||||
* Copyright (C) 2005-2010 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 <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
package org.alfresco.repo.content.replication;
|
||||
|
||||
import java.io.File;
|
||||
import java.util.HashSet;
|
||||
import java.util.Set;
|
||||
|
||||
import junit.framework.TestCase;
|
||||
|
||||
import org.alfresco.repo.content.ContentContext;
|
||||
import org.alfresco.repo.content.ContentStore;
|
||||
import org.alfresco.repo.content.ContentStore.ContentUrlHandler;
|
||||
import org.alfresco.repo.content.filestore.FileContentStore;
|
||||
import org.alfresco.service.cmr.repository.ContentIOException;
|
||||
import org.alfresco.service.cmr.repository.ContentWriter;
|
||||
import org.alfresco.util.GUID;
|
||||
import org.alfresco.util.TempFileProvider;
|
||||
import org.springframework.context.ApplicationEventPublisher;
|
||||
import org.springframework.context.support.StaticApplicationContext;
|
||||
|
||||
/**
|
||||
* Tests the content store replicator.
|
||||
*
|
||||
* @see org.alfresco.repo.content.replication.ContentStoreReplicator
|
||||
*
|
||||
* @author Derek Hulley
|
||||
*/
|
||||
@SuppressWarnings("unused")
|
||||
public class ContentStoreReplicatorTest extends TestCase
|
||||
{
|
||||
private static final String SOME_CONTENT = "The No. 1 Ladies' Detective Agency";
|
||||
|
||||
private ContentStoreReplicator replicator;
|
||||
private ContentStore sourceStore;
|
||||
private ContentStore targetStore;
|
||||
|
||||
@Override
|
||||
public void setUp() throws Exception
|
||||
{
|
||||
super.setUp();
|
||||
|
||||
// Create a dummy context for message broadcasting
|
||||
StaticApplicationContext ctx = new StaticApplicationContext();
|
||||
ctx.refresh();
|
||||
|
||||
File tempDir = TempFileProvider.getTempDir();
|
||||
// create the source file store
|
||||
String storeDir = tempDir.getAbsolutePath() + File.separatorChar + getName() + File.separatorChar + GUID.generate();
|
||||
sourceStore = new FileContentStore(ctx, storeDir);
|
||||
// create the target file store
|
||||
storeDir = tempDir.getAbsolutePath() + File.separatorChar + getName() + File.separatorChar + GUID.generate();
|
||||
targetStore = new FileContentStore(ctx, storeDir);
|
||||
|
||||
// create the replicator
|
||||
replicator = new ContentStoreReplicator();
|
||||
replicator.setSourceStore(sourceStore);
|
||||
replicator.setTargetStore(targetStore);
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates a source with some files and replicates in a single pass, checking the results.
|
||||
*/
|
||||
public void testSinglePassReplication() throws Exception
|
||||
{
|
||||
ContentWriter writer = sourceStore.getWriter(ContentStore.NEW_CONTENT_CONTEXT);
|
||||
writer.putContent("123");
|
||||
|
||||
// replicate
|
||||
replicator.start();
|
||||
|
||||
// wait a second
|
||||
synchronized(this)
|
||||
{
|
||||
this.wait(5000L);
|
||||
}
|
||||
|
||||
assertTrue("Target store doesn't have content added to source",
|
||||
targetStore.exists(writer.getContentUrl()));
|
||||
|
||||
// this was a single pass, so now more replication should be done
|
||||
writer = sourceStore.getWriter(ContentStore.NEW_CONTENT_CONTEXT);
|
||||
writer.putContent("456");
|
||||
|
||||
// wait a second
|
||||
synchronized(this)
|
||||
{
|
||||
this.wait(1000L);
|
||||
}
|
||||
|
||||
assertFalse("Replication should have been single-pass",
|
||||
targetStore.exists(writer.getContentUrl()));
|
||||
}
|
||||
|
||||
/**
|
||||
* Handler that merely records the URL
|
||||
*
|
||||
* @author Derek Hulley
|
||||
* @since 2.0
|
||||
*/
|
||||
private class UrlRecorder implements ContentUrlHandler
|
||||
{
|
||||
public Set<String> urls = new HashSet<String>(1027);
|
||||
public void handle(String contentUrl)
|
||||
{
|
||||
urls.add(contentUrl);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds content to the source while the replicator is going as fast as possible.
|
||||
* Just to make it more interesting, the content is sometimes put in the target
|
||||
* store as well.
|
||||
* <p>
|
||||
* Afterwards, some content is removed from the the target.
|
||||
* <p>
|
||||
* Then, finally, a check is performed to ensure that the source and target are
|
||||
* in synch.
|
||||
*/
|
||||
public void testContinuousReplication() throws Exception
|
||||
{
|
||||
replicator.start();
|
||||
|
||||
String duplicateUrl = null;
|
||||
// start the replicator - it won't wait between iterations
|
||||
for (int i = 0; i < 10; i++)
|
||||
{
|
||||
// put some content into both the target and source
|
||||
ContentWriter duplicateSourceWriter = sourceStore.getWriter(ContentStore.NEW_CONTENT_CONTEXT);
|
||||
duplicateUrl = duplicateSourceWriter.getContentUrl();
|
||||
try
|
||||
{
|
||||
ContentContext targetContentCtx = new ContentContext(null, duplicateUrl);
|
||||
ContentWriter duplicateTargetWriter = targetStore.getWriter(targetContentCtx);
|
||||
duplicateTargetWriter.putContent("Duplicate Target Content: " + i);
|
||||
duplicateSourceWriter.putContent(duplicateTargetWriter.getReader());
|
||||
}
|
||||
catch (ContentIOException e)
|
||||
{
|
||||
// This can happen because the replicator may have beaten us to it
|
||||
}
|
||||
|
||||
for (int j = 0; j < 100; j++)
|
||||
{
|
||||
// write content
|
||||
ContentWriter writer = sourceStore.getWriter(ContentStore.NEW_CONTENT_CONTEXT);
|
||||
writer.putContent("Repeated put: " + j);
|
||||
}
|
||||
}
|
||||
|
||||
// remove the last duplicated URL from the target
|
||||
targetStore.delete(duplicateUrl);
|
||||
|
||||
// allow time for the replicator to catch up
|
||||
synchronized(this)
|
||||
{
|
||||
this.wait(1000L);
|
||||
}
|
||||
|
||||
// check that we have an exact match of URLs
|
||||
UrlRecorder sourceUrls = new UrlRecorder();
|
||||
UrlRecorder targetUrls = new UrlRecorder();
|
||||
sourceStore.getUrls(sourceUrls);
|
||||
targetStore.getUrls(targetUrls);
|
||||
|
||||
assertTrue("Source must be equal to target", sourceUrls.urls.containsAll(targetUrls.urls));
|
||||
//targetUrls.urls.contains(sourceUrls.urls);
|
||||
}
|
||||
|
||||
/**
|
||||
* Call the replicator repeatedly to check that it prevents concurrent use
|
||||
*/
|
||||
public void testRepeatedReplication() throws Exception
|
||||
{
|
||||
for (int i = 0; i < 10; i++)
|
||||
{
|
||||
replicator.start();
|
||||
}
|
||||
}
|
||||
}
|
@@ -1,241 +0,0 @@
|
||||
/*
|
||||
* Copyright (C) 2005-2010 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 <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
package org.alfresco.repo.content.replication;
|
||||
|
||||
import java.io.File;
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashSet;
|
||||
import java.util.List;
|
||||
import java.util.Set;
|
||||
import java.util.concurrent.SynchronousQueue;
|
||||
import java.util.concurrent.ThreadPoolExecutor;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import org.alfresco.repo.content.AbstractWritableContentStoreTest;
|
||||
import org.alfresco.repo.content.ContentContext;
|
||||
import org.alfresco.repo.content.ContentStore;
|
||||
import org.alfresco.repo.content.ContentStore.ContentUrlHandler;
|
||||
import org.alfresco.repo.content.filestore.FileContentStore;
|
||||
import org.alfresco.service.cmr.repository.ContentIOException;
|
||||
import org.alfresco.service.cmr.repository.ContentReader;
|
||||
import org.alfresco.service.cmr.repository.ContentWriter;
|
||||
import org.alfresco.util.GUID;
|
||||
import org.alfresco.util.TempFileProvider;
|
||||
|
||||
/**
|
||||
* Tests read and write functionality for the replicating store.
|
||||
* <p>
|
||||
* By default, replication is off for both the inbound and outbound
|
||||
* replication. Specific tests change this.
|
||||
*
|
||||
* @see org.alfresco.repo.content.replication.ReplicatingContentStore
|
||||
*
|
||||
* @author Derek Hulley
|
||||
*/
|
||||
public class ReplicatingContentStoreTest extends AbstractWritableContentStoreTest
|
||||
{
|
||||
private static final String SOME_CONTENT = "The No. 1 Ladies' Detective Agency";
|
||||
|
||||
private ReplicatingContentStore replicatingStore;
|
||||
private ContentStore primaryStore;
|
||||
private List<ContentStore> secondaryStores;
|
||||
|
||||
@Override
|
||||
public void setUp() throws Exception
|
||||
{
|
||||
super.setUp();
|
||||
|
||||
File tempDir = TempFileProvider.getTempDir();
|
||||
// create a primary file store
|
||||
String storeDir = tempDir.getAbsolutePath() + File.separatorChar + GUID.generate();
|
||||
primaryStore = new FileContentStore(ctx, storeDir);
|
||||
// create some secondary file stores
|
||||
secondaryStores = new ArrayList<ContentStore>(3);
|
||||
for (int i = 0; i < 4; i++)
|
||||
{
|
||||
storeDir = tempDir.getAbsolutePath() + File.separatorChar + GUID.generate();
|
||||
FileContentStore store = new FileContentStore(ctx, storeDir);
|
||||
secondaryStores.add(store);
|
||||
// Only the first 3 are writable
|
||||
if (i >= 3)
|
||||
{
|
||||
store.setReadOnly(true);
|
||||
}
|
||||
}
|
||||
// Create the replicating store
|
||||
replicatingStore = new ReplicatingContentStore();
|
||||
replicatingStore.setPrimaryStore(primaryStore);
|
||||
replicatingStore.setSecondaryStores(secondaryStores);
|
||||
replicatingStore.setOutbound(false);
|
||||
replicatingStore.setInbound(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public ContentStore getStore()
|
||||
{
|
||||
return replicatingStore;
|
||||
}
|
||||
|
||||
/**
|
||||
* Performs checks necessary to ensure the proper replication of content for the given
|
||||
* URL
|
||||
*/
|
||||
private void checkForReplication(boolean inbound, boolean outbound, String contentUrl, String content)
|
||||
{
|
||||
if (inbound)
|
||||
{
|
||||
ContentReader reader = primaryStore.getReader(contentUrl);
|
||||
assertTrue("Content was not replicated into the primary store", reader.exists());
|
||||
assertEquals("The replicated content was incorrect", content, reader.getContentString());
|
||||
}
|
||||
if (outbound)
|
||||
{
|
||||
for (ContentStore store : secondaryStores)
|
||||
{
|
||||
// This is only required for writable stores
|
||||
if (!store.isWriteSupported())
|
||||
{
|
||||
continue;
|
||||
}
|
||||
ContentReader reader = store.getReader(contentUrl);
|
||||
assertTrue("Content was not replicated out to the secondary stores within a second", reader.exists());
|
||||
assertEquals("The replicated content was incorrect", content, reader.getContentString());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks that the url is present in each of the stores
|
||||
*
|
||||
* @param contentUrl
|
||||
* @param mustExist true if the content must exist, false if it must <b>not</b> exist
|
||||
*/
|
||||
private void checkForUrl(String contentUrl, boolean mustExist)
|
||||
{
|
||||
// check that the URL is present for each of the stores
|
||||
for (ContentStore store : secondaryStores)
|
||||
{
|
||||
final Set<String> urls = new HashSet<String>(1027);
|
||||
ContentUrlHandler handler = new ContentUrlHandler()
|
||||
{
|
||||
public void handle(String contentUrl)
|
||||
{
|
||||
urls.add(contentUrl);
|
||||
}
|
||||
};
|
||||
store.getUrls(handler);
|
||||
assertTrue("URL of new content not present in store", urls.contains(contentUrl) == mustExist);
|
||||
}
|
||||
}
|
||||
|
||||
public void testNoReplication() throws Exception
|
||||
{
|
||||
ContentWriter writer = getWriter();
|
||||
writer.putContent(SOME_CONTENT);
|
||||
|
||||
checkForReplication(false, false, writer.getContentUrl(), SOME_CONTENT);
|
||||
}
|
||||
|
||||
public void testOutboundReplication() throws Exception
|
||||
{
|
||||
replicatingStore.setOutbound(true);
|
||||
|
||||
// write some content
|
||||
ContentWriter writer = getWriter();
|
||||
writer.putContent(SOME_CONTENT);
|
||||
String contentUrl = writer.getContentUrl();
|
||||
|
||||
checkForReplication(false, true, contentUrl, SOME_CONTENT);
|
||||
|
||||
// check for outbound deletes
|
||||
replicatingStore.delete(contentUrl);
|
||||
checkForUrl(contentUrl, false);
|
||||
}
|
||||
|
||||
public void testAsyncOutboundReplication() throws Exception
|
||||
{
|
||||
ThreadPoolExecutor tpe = new ThreadPoolExecutor(1, 1, 60L, TimeUnit.SECONDS, new SynchronousQueue<Runnable>());
|
||||
|
||||
replicatingStore.setOutbound(true);
|
||||
replicatingStore.setOutboundThreadPoolExecutor(tpe);
|
||||
|
||||
// write some content
|
||||
ContentWriter writer = getWriter();
|
||||
writer.putContent(SOME_CONTENT);
|
||||
String contentUrl = writer.getContentUrl();
|
||||
|
||||
// wait for a second
|
||||
synchronized(this)
|
||||
{
|
||||
this.wait(1000L);
|
||||
}
|
||||
|
||||
checkForReplication(false, true, contentUrl, SOME_CONTENT);
|
||||
|
||||
// check for outbound deletes
|
||||
replicatingStore.delete(contentUrl);
|
||||
checkForUrl(contentUrl, false);
|
||||
}
|
||||
|
||||
public void testInboundReplication() throws Exception
|
||||
{
|
||||
replicatingStore.setInbound(false);
|
||||
|
||||
// pick a secondary store and write some content to it
|
||||
ContentStore secondaryStore = secondaryStores.get(2);
|
||||
ContentWriter writer = secondaryStore.getWriter(ContentContext.NULL_CONTEXT);
|
||||
writer.putContent(SOME_CONTENT);
|
||||
String contentUrl = writer.getContentUrl();
|
||||
|
||||
// get a reader from the replicating store
|
||||
ContentReader reader = replicatingStore.getReader(contentUrl);
|
||||
assertTrue("Reader must have been found in secondary store", reader.exists());
|
||||
|
||||
// set inbound replication on and repeat
|
||||
replicatingStore.setInbound(true);
|
||||
reader = replicatingStore.getReader(contentUrl);
|
||||
|
||||
// this time, it must have been replicated to the primary store
|
||||
checkForReplication(true, false, contentUrl, SOME_CONTENT);
|
||||
}
|
||||
|
||||
public void testTargetContentUrlExists()
|
||||
{
|
||||
replicatingStore.setOutbound(true);
|
||||
replicatingStore.setInbound(false);
|
||||
// pick a secondary store and write some content to it
|
||||
ContentStore secondaryStore = secondaryStores.get(2);
|
||||
ContentWriter secondaryWriter = secondaryStore.getWriter(ContentContext.NULL_CONTEXT);
|
||||
secondaryWriter.putContent("Content for secondary");
|
||||
String secondaryContentUrl = secondaryWriter.getContentUrl();
|
||||
|
||||
// Now write to the primary store
|
||||
ContentWriter replicatingWriter = replicatingStore.getWriter(new ContentContext(null, secondaryContentUrl));
|
||||
String replicatingContent = "Content for primary";
|
||||
try
|
||||
{
|
||||
replicatingWriter.putContent(replicatingContent);
|
||||
fail("Replication should fail when the secondary store already has the content");
|
||||
}
|
||||
catch (ContentIOException e)
|
||||
{
|
||||
// Expected
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user