diff options
author | Olivier Lamy <olamy@apache.org> | 2013-12-12 11:26:40 +0000 |
---|---|---|
committer | Olivier Lamy <olamy@apache.org> | 2013-12-12 11:26:40 +0000 |
commit | 6b23332e9def8fc0e75be883f8a18807f0fccbbd (patch) | |
tree | 96de94a165343d5f8a69abe6f2bdd3f580e2931a /archiva-modules/archiva-base/archiva-filelock/src | |
parent | 074386d154d4134fbfae4bc7642cd72d58422313 (diff) | |
download | archiva-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')
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 Binary files differnew file mode 100644 index 000000000..0a4b1c542 --- /dev/null +++ b/archiva-modules/archiva-base/archiva-filelock/src/test/cassandra-all-2.0.3.jar 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> + + |