aboutsummaryrefslogtreecommitdiffstats
path: root/sonar-batch/src
diff options
context:
space:
mode:
authorDuarte Meneses <duarte.meneses@sonarsource.com>2016-03-15 14:58:03 +0100
committerDuarte Meneses <duarte.meneses@sonarsource.com>2016-03-16 14:16:03 +0100
commita280429889a21f2f258041caa42264570017c31a (patch)
tree94fe06e6ddce06d238efa26b3b493e8b7a8f6969 /sonar-batch/src
parent712c66d95af8076fe3b4efd7392a61672dc7b656 (diff)
downloadsonarqube-a280429889a21f2f258041caa42264570017c31a.tar.gz
sonarqube-a280429889a21f2f258041caa42264570017c31a.zip
SONAR-7124 Concurrent access to WS cache allowed
Diffstat (limited to 'sonar-batch/src')
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/scan/MutableProjectReactorProvider.java12
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/scan/ProjectLock.java18
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/scan/ProjectScanContainer.java2
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/scan/WorkDirectoryCleaner.java63
-rw-r--r--sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndexer.java4
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/scan/ProjectLockTest.java25
-rw-r--r--sonar-batch/src/test/java/org/sonar/batch/scan/WorkDirectoryCleanerTest.java79
7 files changed, 166 insertions, 37 deletions
diff --git a/sonar-batch/src/main/java/org/sonar/batch/scan/MutableProjectReactorProvider.java b/sonar-batch/src/main/java/org/sonar/batch/scan/MutableProjectReactorProvider.java
index c3d34a23a30..afde1e18c4d 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/scan/MutableProjectReactorProvider.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/scan/MutableProjectReactorProvider.java
@@ -19,11 +19,8 @@
*/
package org.sonar.batch.scan;
-import java.io.File;
-import java.io.IOException;
import org.picocontainer.injectors.ProviderAdapter;
import org.sonar.api.batch.bootstrap.ProjectReactor;
-import org.sonar.core.util.FileUtils;
public class MutableProjectReactorProvider extends ProviderAdapter {
private ProjectReactor reactor = null;
@@ -31,16 +28,7 @@ public class MutableProjectReactorProvider extends ProviderAdapter {
public ProjectReactor provide(ProjectReactorBuilder builder) {
if (reactor == null) {
reactor = builder.execute();
- cleanDirectory(reactor.getRoot().getWorkDir());
}
return reactor;
}
-
- private static void cleanDirectory(File dir) {
- try {
- FileUtils.cleanDirectory(dir);
- } catch (IOException e) {
- throw new IllegalStateException("Failed to recreate working directory: " + dir.getAbsolutePath(), e);
- }
- }
}
diff --git a/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectLock.java b/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectLock.java
index d37987adfe6..d52f49cfd88 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectLock.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectLock.java
@@ -20,24 +20,30 @@
package org.sonar.batch.scan;
import org.sonar.batch.bootstrap.Slf4jLogger;
-
import org.sonar.home.cache.DirectoryLock;
import org.picocontainer.Startable;
import org.sonar.api.batch.bootstrap.ProjectReactor;
+import java.io.IOException;
import java.nio.channels.OverlappingFileLockException;
+import java.nio.file.Files;
import java.nio.file.Path;
public class ProjectLock implements Startable {
- public static final String LOCK_FILE_NAME = ".sonar_lock";
-
- private DirectoryLock lock;
+ private final DirectoryLock lock;
public ProjectLock(ProjectReactor projectReactor) {
- Path directory = projectReactor.getRoot().getBaseDir().toPath();
+ Path directory = projectReactor.getRoot().getWorkDir().toPath();
+ try {
+ if (!Files.exists(directory)) {
+ Files.createDirectories(directory);
+ }
+ } catch (IOException e) {
+ throw new IllegalStateException("Failed to create work directory", e);
+ }
this.lock = new DirectoryLock(directory.toAbsolutePath(), new Slf4jLogger());
}
-
+
public void tryLock() {
try {
if (!lock.tryLock()) {
diff --git a/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectScanContainer.java b/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectScanContainer.java
index f92f07acce8..ff8108e4871 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectScanContainer.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/scan/ProjectScanContainer.java
@@ -111,6 +111,7 @@ public class ProjectScanContainer extends ComponentContainer {
addBatchComponents();
lock = getComponentByType(ProjectLock.class);
lock.tryLock();
+ getComponentByType(WorkDirectoryCleaner.class).execute();
addBatchExtensions();
Settings settings = getComponentByType(Settings.class);
if (settings != null && settings.getBoolean(CoreProperties.PROFILING_LOG_PROPERTY)) {
@@ -139,6 +140,7 @@ public class ProjectScanContainer extends ComponentContainer {
props,
DefaultAnalysisMode.class,
ProjectReactorBuilder.class,
+ WorkDirectoryCleaner.class,
new MutableProjectReactorProvider(),
new ImmutableProjectReactorProvider(),
ProjectBuildersExecutor.class,
diff --git a/sonar-batch/src/main/java/org/sonar/batch/scan/WorkDirectoryCleaner.java b/sonar-batch/src/main/java/org/sonar/batch/scan/WorkDirectoryCleaner.java
new file mode 100644
index 00000000000..b4857bf6afb
--- /dev/null
+++ b/sonar-batch/src/main/java/org/sonar/batch/scan/WorkDirectoryCleaner.java
@@ -0,0 +1,63 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact AT sonarsource DOT com
+ *
+ * This program 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.
+ *
+ * This program 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 this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.batch.scan;
+
+import org.sonar.api.batch.bootstrap.ProjectReactor;
+import org.sonar.core.util.FileUtils;
+import org.sonar.home.cache.DirectoryLock;
+
+import java.io.IOException;
+import java.nio.file.DirectoryStream;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.util.Iterator;
+
+public class WorkDirectoryCleaner {
+ private Path workDir;
+
+ public WorkDirectoryCleaner(ProjectReactor projectReactor) {
+ workDir = projectReactor.getRoot().getWorkDir().toPath();
+ }
+
+ public void execute() {
+ if (!Files.exists(workDir)) {
+ return;
+ }
+
+ try (DirectoryStream<Path> stream = list()) {
+
+ Iterator<Path> it = stream.iterator();
+ while (it.hasNext()) {
+ FileUtils.deleteQuietly(it.next().toFile());
+ }
+ } catch (IOException e) {
+ throw new IllegalStateException("Failed to clean working directory: " + workDir.toString(), e);
+ }
+ }
+
+ private DirectoryStream<Path> list() throws IOException {
+ return Files.newDirectoryStream(workDir, new DirectoryStream.Filter<Path>() {
+ @Override
+ public boolean accept(Path entry) throws IOException {
+ return !DirectoryLock.LOCK_FILE_NAME.equals(entry.getFileName().toString());
+ }
+ });
+ }
+}
diff --git a/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndexer.java b/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndexer.java
index 0ed036192b2..9ebf91886a0 100644
--- a/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndexer.java
+++ b/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndexer.java
@@ -31,8 +31,8 @@ import org.sonar.api.batch.fs.internal.DefaultInputDir;
import org.sonar.api.batch.fs.internal.DefaultInputFile;
import org.sonar.api.scan.filesystem.PathResolver;
import org.sonar.api.utils.MessageException;
-import org.sonar.batch.scan.ProjectLock;
import org.sonar.batch.util.ProgressReport;
+import org.sonar.home.cache.DirectoryLock;
import java.io.File;
import java.io.IOException;
@@ -217,7 +217,7 @@ public class FileIndexer {
@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
- if (!Files.isHidden(file) && !ProjectLock.LOCK_FILE_NAME.equals(file.getFileName().toString())) {
+ if (!Files.isHidden(file) && !DirectoryLock.LOCK_FILE_NAME.equals(file.getFileName().toString())) {
indexFile(inputFileBuilder, fileSystem, status, file, type);
}
return FileVisitResult.CONTINUE;
diff --git a/sonar-batch/src/test/java/org/sonar/batch/scan/ProjectLockTest.java b/sonar-batch/src/test/java/org/sonar/batch/scan/ProjectLockTest.java
index d2139dd1bfd..c03240b7341 100644
--- a/sonar-batch/src/test/java/org/sonar/batch/scan/ProjectLockTest.java
+++ b/sonar-batch/src/test/java/org/sonar/batch/scan/ProjectLockTest.java
@@ -22,11 +22,11 @@ package org.sonar.batch.scan;
import org.junit.rules.ExpectedException;
import org.sonar.api.batch.bootstrap.ProjectDefinition;
import org.sonar.api.batch.bootstrap.ProjectReactor;
+import org.sonar.home.cache.DirectoryLock;
import java.io.File;
import java.nio.file.Files;
import java.nio.file.Path;
-import java.nio.file.Paths;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;
@@ -48,25 +48,25 @@ public class ProjectLockTest {
public void setUp() {
lock = setUpTest(tempFolder.getRoot());
}
-
+
private ProjectLock setUpTest(File file) {
ProjectReactor projectReactor = mock(ProjectReactor.class);
ProjectDefinition projectDefinition = mock(ProjectDefinition.class);
when(projectReactor.getRoot()).thenReturn(projectDefinition);
- when(projectDefinition.getBaseDir()).thenReturn(file);
+ when(projectDefinition.getWorkDir()).thenReturn(file);
return new ProjectLock(projectReactor);
}
@Test
public void tryLock() {
- Path lockFilePath = tempFolder.getRoot().toPath().resolve(ProjectLock.LOCK_FILE_NAME);
+ Path lockFilePath = tempFolder.getRoot().toPath().resolve(DirectoryLock.LOCK_FILE_NAME);
lock.tryLock();
assertThat(Files.exists(lockFilePath)).isTrue();
assertThat(Files.isRegularFile(lockFilePath)).isTrue();
lock.stop();
- assertThat(Files.exists(lockFilePath)).isFalse();
+ assertThat(Files.exists(lockFilePath)).isTrue();
}
@Test
@@ -76,7 +76,7 @@ public class ProjectLockTest {
lock.tryLock();
lock.tryLock();
}
-
+
@Test
/**
* If there is an error starting up the scan, we'll still try to unlock even if the lock
@@ -94,18 +94,9 @@ public class ProjectLockTest {
lock.tryLock();
lock.stop();
}
-
- @Test
- public void errorLock() {
- lock = setUpTest(Paths.get("path", "that", "wont", "exist", "ever").toFile());
- exception.expect(IllegalStateException.class);
- exception.expectMessage("Failed to create lock in");
- lock.tryLock();
- }
-
+
@Test
- public void errorDeleteLock() {
- lock = setUpTest(Paths.get("path", "that", "wont", "exist", "ever").toFile());
+ public void unLockWithNoLock() {
lock.stop();
}
diff --git a/sonar-batch/src/test/java/org/sonar/batch/scan/WorkDirectoryCleanerTest.java b/sonar-batch/src/test/java/org/sonar/batch/scan/WorkDirectoryCleanerTest.java
new file mode 100644
index 00000000000..8bd3d5c4e3a
--- /dev/null
+++ b/sonar-batch/src/test/java/org/sonar/batch/scan/WorkDirectoryCleanerTest.java
@@ -0,0 +1,79 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact AT sonarsource DOT com
+ *
+ * This program 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.
+ *
+ * This program 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 this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.batch.scan;
+
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.TemporaryFolder;
+import org.sonar.api.batch.bootstrap.ProjectDefinition;
+import org.sonar.api.batch.bootstrap.ProjectReactor;
+import org.sonar.home.cache.DirectoryLock;
+
+import java.io.File;
+import java.io.IOException;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+public class WorkDirectoryCleanerTest {
+ private WorkDirectoryCleaner cleaner;
+ @Rule
+ public TemporaryFolder temp = new TemporaryFolder();
+
+ @Before
+ public void setUp() throws IOException {
+ // create files to clean
+ temp.newFile();
+ File newFolder = temp.newFolder();
+ File fileInFolder = new File(newFolder, "test");
+ fileInFolder.createNewFile();
+
+ File lock = new File(temp.getRoot(), DirectoryLock.LOCK_FILE_NAME);
+ lock.createNewFile();
+
+ // mock project
+ ProjectReactor projectReactor = mock(ProjectReactor.class);
+ ProjectDefinition projectDefinition = mock(ProjectDefinition.class);
+ when(projectReactor.getRoot()).thenReturn(projectDefinition);
+ when(projectDefinition.getWorkDir()).thenReturn(temp.getRoot());
+
+ assertThat(temp.getRoot().list().length).isGreaterThan(1);
+ cleaner = new WorkDirectoryCleaner(projectReactor);
+ }
+
+ @Test
+ public void testNonExisting() {
+ temp.delete();
+ cleaner.execute();
+ }
+
+ @Test
+ public void testClean() {
+ File lock = new File(temp.getRoot(), DirectoryLock.LOCK_FILE_NAME);
+ cleaner.execute();
+
+ assertThat(temp.getRoot()).exists();
+ assertThat(lock).exists();
+ assertThat(temp.getRoot().list()).containsOnly(DirectoryLock.LOCK_FILE_NAME);
+ }
+
+}