aboutsummaryrefslogtreecommitdiffstats
path: root/archiva-modules/archiva-base/archiva-filelock/src
diff options
context:
space:
mode:
authorOlivier Lamy <olamy@apache.org>2013-12-12 11:26:40 +0000
committerOlivier Lamy <olamy@apache.org>2013-12-12 11:26:40 +0000
commit6b23332e9def8fc0e75be883f8a18807f0fccbbd (patch)
tree96de94a165343d5f8a69abe6f2bdd3f580e2931a /archiva-modules/archiva-base/archiva-filelock/src
parent074386d154d4134fbfae4bc7642cd72d58422313 (diff)
downloadarchiva-6b23332e9def8fc0e75be883f8a18807f0fccbbd.tar.gz
archiva-6b23332e9def8fc0e75be883f8a18807f0fccbbd.zip
add a new module for file locking
git-svn-id: https://svn.apache.org/repos/asf/archiva/trunk@1550396 13f79535-47bb-0310-9956-ffa450edef68
Diffstat (limited to 'archiva-modules/archiva-base/archiva-filelock/src')
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/DefaultFileLockManager.java177
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockException.java13
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockManager.java18
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/Lock.java105
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/main/resources/META-INF/spring-context.xml34
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/test/cassandra-all-2.0.3.jarbin0 -> 3369436 bytes
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/test/java/org/apache/archiva/common/filelock/DefaultFileLockManagerTest.java208
-rw-r--r--archiva-modules/archiva-base/archiva-filelock/src/test/resources/log4j2-test.xml44
8 files changed, 599 insertions, 0 deletions
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/DefaultFileLockManager.java b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/DefaultFileLockManager.java
new file mode 100644
index 000000000..795760672
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/DefaultFileLockManager.java
@@ -0,0 +1,177 @@
+package org.apache.archiva.common.filelock;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.stereotype.Service;
+
+import java.io.File;
+import java.io.IOException;
+
+import java.nio.channels.OverlappingFileLockException;
+import java.util.concurrent.ConcurrentHashMap;
+import java.util.concurrent.ConcurrentMap;
+
+/**
+ * @author Olivier Lamy
+ */
+@Service("fileLockManager#default")
+public class DefaultFileLockManager
+ implements FileLockManager
+{
+ private static final ConcurrentMap<File, Lock> lockFiles = new ConcurrentHashMap<File, Lock>( 64 );
+
+ private boolean skipLocking = false;
+
+ private Logger log = LoggerFactory.getLogger( getClass() );
+
+ @Override
+ public Lock readFileLock( File file )
+ throws FileLockException
+ {
+ if ( skipLocking )
+ {
+ try
+ {
+ return new Lock( file, false );
+ }
+ catch ( IOException e )
+ {
+ throw new FileLockException( e.getMessage(), e );
+ }
+ }
+ Lock lock = lockFiles.get( file );
+ if ( lock == null )
+ {
+ try
+ {
+ lock = new Lock( file, false );
+ Lock current = lockFiles.putIfAbsent( file, lock );
+ if ( current != null )
+ {
+ lock = current;
+ }
+ return lock;
+ }
+ catch ( IOException e )
+ {
+ throw new FileLockException( e.getMessage(), e );
+ }
+ catch ( OverlappingFileLockException e )
+ {
+ log.debug( "OverlappingFileLockException: {}", e.getMessage() );
+ if ( lock == null )
+ {
+ lock = lockFiles.get( file );
+ }
+ }
+ }
+ // FIXME add a timeout on getting that!!!
+ while ( true )
+ {
+ log.debug( "wait read lock" );
+ synchronized ( lock )
+ {
+ if ( lock.getFileLock().isShared() || !lock.getFileLock().isValid() )
+ {
+ lock.addFileClient( Thread.currentThread() );
+ return lock;
+ }
+ }
+ }
+ //return lock;
+ }
+
+ @Override
+ public Lock writeFileLock( File file )
+ throws FileLockException
+ {
+ try
+ {
+ if ( skipLocking )
+ {
+ return new Lock( file, true );
+ }
+
+ // FIXME add a timeout on getting that!!!
+ while ( true )
+ {
+ Lock lock = lockFiles.get( file );
+ log.debug( "wait write lock" );
+ if ( lock != null )
+ {
+ synchronized ( lock )
+ {
+ if ( lock.getFileLock().isValid() || lock.getFileClients().size() > 0 )
+ {
+ continue;
+ }
+ return lock;
+ }
+ }
+ else
+ {
+ try
+ {
+ lock = new Lock( file, true );
+ }
+ catch ( OverlappingFileLockException e )
+ {
+ log.debug( "OverlappingFileLockException: {}", e.getMessage() );
+ if ( lock == null )
+ {
+ lock = lockFiles.get( file );
+ }
+
+ lock = lockFiles.get( file );
+ log.debug( "OverlappingFileLockException get: {}", lock );
+ }
+ Lock current = lockFiles.putIfAbsent( file, lock );
+ if ( current != null )
+ {
+ lock = current;
+ }
+ return lock;
+ }
+ }
+
+ }
+ catch ( IOException e )
+ {
+ throw new FileLockException( e.getMessage(), e );
+ }
+ }
+
+ @Override
+ public void release( Lock lock )
+ throws FileLockException
+ {
+ if ( lock == null )
+ {
+ log.debug( "skip releasing null" );
+ return;
+ }
+ if ( skipLocking )
+ {
+ return;
+ }
+ try
+ {
+ if ( lock.isWrite().get() )
+ {
+ lock.getFileLock().release();
+ }
+ synchronized ( lock )
+ {
+ lock.close();
+ if ( lock.getFileClients().size() < 1 )
+ {
+ lockFiles.remove( lock.getFile() );
+ }
+ }
+ }
+ catch ( IOException e )
+ {
+ throw new FileLockException( e.getMessage(), e );
+ }
+ }
+}
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockException.java b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockException.java
new file mode 100644
index 000000000..2d14859c0
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockException.java
@@ -0,0 +1,13 @@
+package org.apache.archiva.common.filelock;
+
+/**
+ * @author Olivier Lamy
+ */
+public class FileLockException
+ extends Exception
+{
+ public FileLockException( String s, Throwable throwable )
+ {
+ super( s, throwable );
+ }
+}
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockManager.java b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockManager.java
new file mode 100644
index 000000000..5f3832634
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/FileLockManager.java
@@ -0,0 +1,18 @@
+package org.apache.archiva.common.filelock;
+
+import java.io.File;
+
+/**
+ * @author Olivier Lamy
+ */
+public interface FileLockManager
+{
+ Lock writeFileLock( File file )
+ throws FileLockException;
+
+ Lock readFileLock( File file )
+ throws FileLockException;
+
+ void release( Lock lock )
+ throws FileLockException;
+}
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/Lock.java b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/Lock.java
new file mode 100644
index 000000000..6baabbc05
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/main/java/org/apache/archiva/common/filelock/Lock.java
@@ -0,0 +1,105 @@
+package org.apache.archiva.common.filelock;
+
+import java.io.File;
+import java.io.FileNotFoundException;
+import java.io.IOException;
+import java.io.RandomAccessFile;
+import java.nio.channels.FileLock;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.concurrent.atomic.AtomicBoolean;
+import java.util.concurrent.atomic.AtomicInteger;
+
+/**
+ * @author Olivier Lamy
+ */
+public class Lock
+{
+ private File file;
+
+ private AtomicBoolean write;
+
+ private final Map<Thread, AtomicInteger> fileClients = new HashMap<Thread, AtomicInteger>();
+
+ private FileLock fileLock;
+
+ public Lock( File file, boolean write )
+ throws FileNotFoundException, IOException
+ {
+ this.file = file;
+ this.write = new AtomicBoolean( write );
+ this.openLock( write );
+ }
+
+ public File getFile()
+ {
+ return file;
+ }
+
+ public AtomicBoolean isWrite()
+ {
+ return write;
+ }
+
+ public void setFile( File file )
+ {
+ this.file = file;
+ }
+
+ public void setWrite( boolean write )
+ {
+ this.write.set( write );
+ }
+
+ public FileLock getFileLock()
+ {
+ return fileLock;
+ }
+
+ public void setFileLock( FileLock fileLock )
+ {
+ this.fileLock = fileLock;
+ }
+
+ public Map<Thread, AtomicInteger> getFileClients()
+ {
+ return fileClients;
+ }
+
+ public void addFileClient( Thread thread )
+ {
+ this.fileClients.put( thread, new AtomicInteger( 1 ) );
+ }
+
+ public boolean removeFileClient( Thread thread )
+ {
+ return this.fileClients.remove( thread ) != null;
+ }
+
+ protected void close()
+ throws IOException
+ {
+ if ( this.write.get() )
+ {
+ this.fileLock.release();
+ fileClients.remove( Thread.currentThread() );
+ }
+ }
+
+ public void openLock( boolean write )
+ throws IOException
+ {
+ fileClients.put( Thread.currentThread(), new AtomicInteger( 1 ) );
+ RandomAccessFile raf = new RandomAccessFile( file, write ? "rw" : "r" );
+ this.fileLock = raf.getChannel().lock( 1, 1, !write );
+ }
+
+ @Override
+ public String toString()
+ {
+ final StringBuilder sb = new StringBuilder( "Lock{" );
+ sb.append( "file=" ).append( file );
+ sb.append( '}' );
+ return sb.toString();
+ }
+}
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/main/resources/META-INF/spring-context.xml b/archiva-modules/archiva-base/archiva-filelock/src/main/resources/META-INF/spring-context.xml
new file mode 100644
index 000000000..944c8d606
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/main/resources/META-INF/spring-context.xml
@@ -0,0 +1,34 @@
+<?xml version="1.0"?>
+
+<!--
+ ~ Licensed to the Apache Software Foundation (ASF) under one
+ ~ or more contributor license agreements. See the NOTICE file
+ ~ distributed with this work for additional information
+ ~ regarding copyright ownership. The ASF licenses this file
+ ~ to you under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0
+ ~
+ ~ Unless required by applicable law or agreed to in writing,
+ ~ software distributed under the License is distributed on an
+ ~ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ ~ KIND, either express or implied. See the License for the
+ ~ specific language governing permissions and limitations
+ ~ under the License.
+ -->
+<beans xmlns="http://www.springframework.org/schema/beans"
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+ xmlns:context="http://www.springframework.org/schema/context"
+ xsi:schemaLocation="http://www.springframework.org/schema/beans
+ http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
+ http://www.springframework.org/schema/context
+ http://www.springframework.org/schema/context/spring-context-3.0.xsd"
+ default-lazy-init="true">
+
+ <context:annotation-config/>
+ <context:component-scan base-package="org.apache.archiva.common.filelock"/>
+
+
+</beans> \ No newline at end of file
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/test/cassandra-all-2.0.3.jar b/archiva-modules/archiva-base/archiva-filelock/src/test/cassandra-all-2.0.3.jar
new file mode 100644
index 000000000..0a4b1c542
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/test/cassandra-all-2.0.3.jar
Binary files differ
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/test/java/org/apache/archiva/common/filelock/DefaultFileLockManagerTest.java b/archiva-modules/archiva-base/archiva-filelock/src/test/java/org/apache/archiva/common/filelock/DefaultFileLockManagerTest.java
new file mode 100644
index 000000000..86fb24dd3
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/test/java/org/apache/archiva/common/filelock/DefaultFileLockManagerTest.java
@@ -0,0 +1,208 @@
+package org.apache.archiva.common.filelock;
+
+import edu.umd.cs.mtc.MultithreadedTestCase;
+import edu.umd.cs.mtc.TestFramework;
+import org.apache.commons.io.FileUtils;
+import org.apache.commons.io.IOUtils;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.test.context.ContextConfiguration;
+import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
+
+import javax.inject.Inject;
+import javax.inject.Named;
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.FileOutputStream;
+import java.io.IOException;
+
+/**
+ * @author Olivier Lamy
+ */
+@RunWith( SpringJUnit4ClassRunner.class )
+@ContextConfiguration( locations = { "classpath*:/META-INF/spring-context.xml" } )
+public class DefaultFileLockManagerTest
+{
+
+ final Logger logger = LoggerFactory.getLogger( getClass() );
+
+ @Inject
+ @Named( value = "fileLockManager#default" )
+ FileLockManager fileLockManager;
+
+ class ConcurentFileWrite
+ extends MultithreadedTestCase
+ {
+
+ FileLockManager fileLockManager;
+
+ File file = new File( System.getProperty( "buildDirectory" ), "foo.txt" );
+
+ File largeJar = new File( System.getProperty( "basedir" ), "src/test/cassandra-all-2.0.3.jar" );
+
+ ConcurentFileWrite( FileLockManager fileLockManager )
+ throws IOException
+ {
+ this.fileLockManager = fileLockManager;
+ file.createNewFile();
+
+ }
+
+ @Override
+ public void initialize()
+ {
+
+ }
+
+ public void thread1()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread1" );
+ Lock lock = fileLockManager.writeFileLock( this.file );
+ try
+ {
+ lock.getFile().delete();
+ FileUtils.copyFile( largeJar, lock.getFile() );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread1 ok" );
+ }
+
+ public void thread2()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread2" );
+ Lock lock = fileLockManager.writeFileLock( this.file );
+ try
+ {
+ lock.getFile().delete();
+ FileUtils.copyFile( largeJar, lock.getFile() );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread2 ok" );
+ }
+
+ public void thread3()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread3" );
+ Lock lock = fileLockManager.readFileLock( this.file );
+ try
+ {
+ IOUtils.copy( new FileInputStream( lock.getFile() ),
+ new FileOutputStream( File.createTempFile( "foo", ".jar" ) ) );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread3 ok" );
+ }
+
+ public void thread4()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread4" );
+ Lock lock = fileLockManager.writeFileLock( this.file );
+ try
+ {
+ lock.getFile().delete();
+ FileUtils.copyFile( largeJar, lock.getFile() );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread4 ok" );
+ }
+
+ public void thread5()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread5" );
+ Lock lock = fileLockManager.writeFileLock( this.file );
+ try
+ {
+ lock.getFile().delete();
+ FileUtils.copyFile( largeJar, lock.getFile() );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread5 ok" );
+ }
+
+ public void thread6()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread6" );
+ Lock lock = fileLockManager.readFileLock( this.file );
+ try
+ {
+ IOUtils.copy( new FileInputStream( lock.getFile() ),
+ new FileOutputStream( File.createTempFile( "foo", ".jar" ) ) );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread6 ok" );
+ }
+
+ public void thread7()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread7" );
+ Lock lock = fileLockManager.writeFileLock( this.file );
+ try
+ {
+ lock.getFile().delete();
+ FileUtils.copyFile( largeJar, lock.getFile() );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread7 ok" );
+ }
+
+ public void thread8()
+ throws FileLockException, IOException
+ {
+ logger.info( "thread8" );
+ Lock lock = fileLockManager.readFileLock( this.file );
+ try
+ {
+ IOUtils.copy( new FileInputStream( lock.getFile() ),
+ new FileOutputStream( File.createTempFile( "foo", ".jar" ) ) );
+ }
+ finally
+ {
+ fileLockManager.release( lock );
+ }
+ logger.info( "thread8 ok" );
+ }
+
+
+
+ }
+
+ @Test
+ public void testWrite()
+ throws Throwable
+ {
+ ConcurentFileWrite concurentFileWrite = new ConcurentFileWrite( fileLockManager );
+ //concurentFileWrite.setTrace( true );
+ TestFramework.runOnce( concurentFileWrite );
+ }
+
+}
diff --git a/archiva-modules/archiva-base/archiva-filelock/src/test/resources/log4j2-test.xml b/archiva-modules/archiva-base/archiva-filelock/src/test/resources/log4j2-test.xml
new file mode 100644
index 000000000..93e519df7
--- /dev/null
+++ b/archiva-modules/archiva-base/archiva-filelock/src/test/resources/log4j2-test.xml
@@ -0,0 +1,44 @@
+<?xml version="1.0" encoding="UTF-8" ?>
+<!--
+ ~ Licensed to the Apache Software Foundation (ASF) under one
+ ~ or more contributor license agreements. See the NOTICE file
+ ~ distributed with this work for additional information
+ ~ regarding copyright ownership. The ASF licenses this file
+ ~ to you under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0
+ ~
+ ~ Unless required by applicable law or agreed to in writing,
+ ~ software distributed under the License is distributed on an
+ ~ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ ~ KIND, either express or implied. See the License for the
+ ~ specific language governing permissions and limitations
+ ~ under the License.
+ -->
+
+
+<configuration status="debug">
+
+
+
+ <appenders>
+ <Console name="console" target="SYSTEM_OUT">
+ <!--PatternLayout pattern="%d{HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n"/-->
+ <PatternLayout pattern="%highlight{%d{HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n}" />
+ </Console>
+
+ </appenders>
+ <loggers>
+
+
+ <logger name="org.apache.archiva.common.filelock" level="debug"/>
+
+ <root level="info" includeLocation="true">
+ <appender-ref ref="console"/>
+ </root>
+ </loggers>
+</configuration>
+
+