summaryrefslogtreecommitdiffstats
path: root/sonar-server/src/main
diff options
context:
space:
mode:
authorSimon Brandhof <simon.brandhof@gmail.com>2013-09-20 18:26:19 +0200
committerSimon Brandhof <simon.brandhof@gmail.com>2013-09-20 18:26:36 +0200
commit52ef49cd29b15da8799547ea34d8f543b68065ef (patch)
treee909bd7acbe447c1a2f878bae1df6674266a1e2c /sonar-server/src/main
parentcdca5c7a1300d91377b89d0a2a4bdfbc98049270 (diff)
downloadsonarqube-52ef49cd29b15da8799547ea34d8f543b68065ef.tar.gz
sonarqube-52ef49cd29b15da8799547ea34d8f543b68065ef.zip
SONAR-4691 do not extensively use Oracle rollback segments
Diffstat (limited to 'sonar-server/src/main')
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/DatabaseMigration.java6
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrations.java30
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrator.java18
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Progress.java7
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Referentials.java59
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverter.java104
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverters.java59
-rw-r--r--sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationMigration.java89
-rw-r--r--sonar-server/src/main/java/org/sonar/server/platform/Platform.java5
9 files changed, 214 insertions, 163 deletions
diff --git a/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigration.java b/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigration.java
index 2e99e4b30bb..f49d40d0505 100644
--- a/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigration.java
+++ b/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigration.java
@@ -19,14 +19,12 @@
*/
package org.sonar.server.db;
-import org.sonar.core.persistence.Database;
-
/**
- * Java alternative of ActiveRecord::Migration.
+ * Java alternative of ActiveRecord::Migration. Do not forget to declare implementation classes in {@link DatabaseMigrations#CLASSES}
* @since 3.7
*/
public interface DatabaseMigration {
- void execute(Database db);
+ void execute();
}
diff --git a/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrations.java b/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrations.java
new file mode 100644
index 00000000000..23e4a8a52b0
--- /dev/null
+++ b/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrations.java
@@ -0,0 +1,30 @@
+/*
+ * SonarQube, open source software quality management tool.
+ * Copyright (C) 2008-2013 SonarSource
+ * mailto:contact AT sonarsource DOT com
+ *
+ * SonarQube 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.
+ *
+ * SonarQube 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.server.db;
+
+import org.sonar.server.db.DatabaseMigration;
+import org.sonar.server.db.migrations.violation.ViolationMigration;
+
+public interface DatabaseMigrations {
+
+ Class<? extends DatabaseMigration>[] CLASSES = new Class[]{
+ ViolationMigration.class
+ };
+}
diff --git a/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrator.java b/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrator.java
index 6b36e055134..c8012b3ca65 100644
--- a/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrator.java
+++ b/sonar-server/src/main/java/org/sonar/server/db/DatabaseMigrator.java
@@ -40,10 +40,12 @@ public class DatabaseMigrator implements ServerComponent {
private final MyBatis myBatis;
private final Database database;
+ private final DatabaseMigration[] migrations;
- public DatabaseMigrator(MyBatis myBatis, Database database) {
+ public DatabaseMigrator(MyBatis myBatis, Database database, DatabaseMigration[] migrations) {
this.myBatis = myBatis;
this.database = database;
+ this.migrations = migrations;
}
/**
@@ -72,10 +74,9 @@ public class DatabaseMigrator implements ServerComponent {
}
public void executeMigration(String className) {
+ DatabaseMigration migration = getMigration(className);
try {
- Class<DatabaseMigration> migrationClass = (Class<DatabaseMigration>) Class.forName(className);
- DatabaseMigration migration = migrationClass.newInstance();
- migration.execute(database);
+ migration.execute();
} catch (Exception e) {
// duplication between log and exception because webapp does not correctly log initial stacktrace
@@ -85,6 +86,15 @@ public class DatabaseMigrator implements ServerComponent {
}
}
+ private DatabaseMigration getMigration(String className) {
+ for (DatabaseMigration migration : migrations) {
+ if (migration.getClass().getName().equals(className)) {
+ return migration;
+ }
+ }
+ throw new IllegalArgumentException("Database migration not found: " + className);
+ }
+
@VisibleForTesting
protected void createSchema(Connection connection, String dialectId) {
DdlUtils.createSchema(connection, dialectId);
diff --git a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Progress.java b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Progress.java
index fe1175a8d7b..8057855c913 100644
--- a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Progress.java
+++ b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Progress.java
@@ -52,14 +52,13 @@ class Progress extends TimerTask {
public void run() {
int totalIssues = counter.get();
long durationMinutes = (System.currentTimeMillis() - start) / 60000L;
- int frequency = 0, remaining = 0;
+ int remaining = 0;
if (durationMinutes > 0) {
- frequency = (int) (totalIssues / durationMinutes);
+ int frequency = (int) (totalIssues / durationMinutes);
remaining = (totalViolations - totalIssues) / frequency;
}
logger.info(String.format(
- "%d%% [%d/%d violations, %d violations/minute, %d minutes remaining]",
- (100 * totalIssues) / totalViolations, totalIssues, totalViolations, frequency, remaining)
+ "%d%% [%d/%d violations, %d minutes remaining]", (100 * totalIssues) / totalViolations, totalIssues, totalViolations, remaining)
);
}
}
diff --git a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Referentials.java b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Referentials.java
index d4d28258488..35ac7c60c81 100644
--- a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Referentials.java
+++ b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/Referentials.java
@@ -32,18 +32,24 @@ import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Map;
+import java.util.Queue;
+import java.util.concurrent.ConcurrentLinkedQueue;
class Referentials {
- private final Database database;
+
+ static final int VIOLATION_GROUP_SIZE = 1000;
+
private final Map<Long, String> loginsByUserId;
private final Map<Long, String> plansById;
- private final int totalViolations;
+ private final Queue<long[]> groupsOfViolationIds;
+
+ // int is enough, it allows to upgrade up to 2 billions violations !
+ private int totalViolations = 0;
Referentials(Database database) throws SQLException {
- this.database = database;
- loginsByUserId = selectLongString("select id,login from users");
- plansById = selectLongString("select id,kee from action_plans");
- totalViolations = selectInt("select count(id) from rule_failures");
+ loginsByUserId = selectLongString(database, "select id,login from users");
+ plansById = selectLongString(database, "select id,kee from action_plans");
+ groupsOfViolationIds = initGroupOfViolationIds(database);
}
@CheckForNull
@@ -60,7 +66,20 @@ class Referentials {
return totalViolations;
}
- private Map<Long, String> selectLongString(String sql) throws SQLException {
+ @CheckForNull
+ Long[] pollGroupOfViolationIds() {
+ long[] longs = groupsOfViolationIds.poll();
+ if (longs == null) {
+ return null;
+ }
+ Long[] objects = new Long[longs.length];
+ for (int i = 0; i < longs.length; i++) {
+ objects[i] = new Long(longs[i]);
+ }
+ return objects;
+ }
+
+ private Map<Long, String> selectLongString(Database database, String sql) throws SQLException {
Connection connection = database.getDataSource().getConnection();
try {
return new QueryRunner().query(connection, sql, new ResultSetHandler<Map<Long, String>>() {
@@ -78,17 +97,33 @@ class Referentials {
}
}
- private int selectInt(String sql) throws SQLException {
+ private Queue<long[]> initGroupOfViolationIds(Database database) throws SQLException {
Connection connection = database.getDataSource().getConnection();
+ connection.setAutoCommit(false);
Statement stmt = null;
ResultSet rs = null;
try {
stmt = connection.createStatement();
- rs = stmt.executeQuery(sql);
- if (rs.next()) {
- return rs.getInt(1);
+ stmt.setFetchSize(10000);
+ rs = stmt.executeQuery("select id from rule_failures");
+ ConcurrentLinkedQueue<long[]> queue = new ConcurrentLinkedQueue<long[]>();
+
+ totalViolations = 0;
+ long[] block = new long[VIOLATION_GROUP_SIZE];
+ int cursor = 0;
+ while (rs.next()) {
+ block[cursor++] = rs.getLong(1);
+ totalViolations++;
+ if (cursor == VIOLATION_GROUP_SIZE) {
+ queue.add(block);
+ block = new long[VIOLATION_GROUP_SIZE];
+ cursor = 0;
+ }
+ }
+ if (cursor > 0) {
+ queue.add(block);
}
- return 0;
+ return queue;
} finally {
DbUtils.closeQuietly(connection, stmt, rs);
}
diff --git a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverter.java b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverter.java
index 1d57602224c..6c819e69913 100644
--- a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverter.java
+++ b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverter.java
@@ -36,8 +36,9 @@ import java.sql.SQLException;
import java.util.List;
import java.util.Map;
import java.util.UUID;
+import java.util.concurrent.Callable;
-class ViolationConverter implements Runnable {
+class ViolationConverter implements Callable<Object> {
private static final long ONE_YEAR = 365L * 24 * 60 * 60 * 1000;
private static final Date ONE_YEAR_AGO = new Date(System.currentTimeMillis() - ONE_YEAR);
@@ -70,6 +71,7 @@ class ViolationConverter implements Runnable {
private static final String USER_ID = "userId";
private static final String SEVERITY_MAJOR = "MAJOR";
+
private static final String SQL_ISSUE_COLUMNS = "kee, component_id, root_component_id, rule_id, severity, manual_severity, message, line, effort_to_fix, status, resolution, " +
"checksum, reporter, assignee, action_plan_key, issue_attributes, issue_creation_date, issue_update_date, created_at, updated_at";
@@ -79,40 +81,78 @@ class ViolationConverter implements Runnable {
private static final String SQL_INSERT_ISSUE_CHANGE = "INSERT INTO issue_changes(kee, issue_key, user_login, change_type, change_data, created_at, updated_at)" +
" VALUES (?, ?, ?, 'comment', ?, ?, ?)";
+ private static final String SQL_DELETE_RULE_FAILURES;
+
+ static {
+ StringBuilder sb = new StringBuilder("delete rule_failures where ");
+ for (int i = 0; i < Referentials.VIOLATION_GROUP_SIZE; i++) {
+ if (i > 0) {
+ sb.append(" or ");
+ }
+ sb.append("id=?");
+ }
+ SQL_DELETE_RULE_FAILURES = sb.toString();
+ }
+
+ static final String SQL_SELECT_RULE_FAILURES;
+
+ static {
+ StringBuilder sb = new StringBuilder("select rev.id as reviewId, s.project_id as projectId, rf.rule_id as ruleId, " +
+ " rf.failure_level as failureLevel, rf.message as message, rf.line as line, " +
+ " rf.cost as cost, rf.created_at as createdAt, rf.checksum as checksum, rev.user_id as reviewReporterId, " +
+ " rev.assignee_id as reviewAssigneeId, rev.status as reviewStatus, " +
+ " rev.severity as reviewSeverity, rev.resolution as reviewResolution, rev.manual_severity as reviewManualSeverity, " +
+ " rev.data as reviewData, rev.updated_at as reviewUpdatedAt, " +
+ " s.root_project_id as rootProjectId, rev.manual_violation as reviewManualViolation, planreviews.action_plan_id as planId " +
+ " from rule_failures rf " +
+ " inner join snapshots s on s.id=rf.snapshot_id " +
+ " left join reviews rev on rev.rule_failure_permanent_id=rf.permanent_id " +
+ " left join action_plans_reviews planreviews on planreviews.review_id=rev.id " +
+ " where ");
+ for (int i = 0; i < Referentials.VIOLATION_GROUP_SIZE; i++) {
+ if (i > 0) {
+ sb.append(" or ");
+ }
+ sb.append("rf.id=?");
+ }
+ SQL_SELECT_RULE_FAILURES = sb.toString();
+ }
+
private final Database db;
- private final Object[] violationIds;
private final Referentials referentials;
private final Progress progress;
- ViolationConverter(Referentials referentials, Database db, Object[] violationIds, Progress progress) {
+ ViolationConverter(Referentials referentials, Database db, Progress progress) {
this.referentials = referentials;
this.db = db;
- this.violationIds = violationIds;
this.progress = progress;
}
@Override
- public void run() {
- convert(selectRows());
+ public Object call() throws Exception {
+ Long[] violationIds = referentials.pollGroupOfViolationIds();
+ while (violationIds != null) {
+ List<Map<String, Object>> rows = selectRows(violationIds);
+ convert(rows, violationIds);
+
+ violationIds = referentials.pollGroupOfViolationIds();
+ }
+ return null;
}
- private List<Map<String, Object>> selectRows() {
+ private List<Map<String, Object>> selectRows(Long[] violationIds) throws SQLException {
Connection readConnection = null;
try {
readConnection = db.getDataSource().getConnection();
ViolationHandler violationHandler = new ViolationHandler();
- return new QueryRunner().query(readConnection, violationHandler.SQL, violationHandler, violationIds);
-
- } catch (SQLException e) {
- //TODO
- throw new IllegalStateException();
+ return new QueryRunner().query(readConnection, SQL_SELECT_RULE_FAILURES, violationHandler, violationIds);
} finally {
DbUtils.closeQuietly(readConnection);
}
}
- private void convert(List<Map<String, Object>> rows) {
+ private void convert(List<Map<String, Object>> rows, Long[] violationIds) throws SQLException {
Connection readConnection = null;
Connection writeConnection = null;
try {
@@ -123,6 +163,7 @@ class ViolationConverter implements Runnable {
List<Object[]> allParams = Lists.newArrayList();
List<Map<String, Object>> allComments = Lists.newArrayList();
+ QueryRunner runner = new QueryRunner();
for (Map<String, Object> row : rows) {
Long componentId = (Long) row.get(PROJECT_ID);
if (componentId == null) {
@@ -151,7 +192,7 @@ class ViolationConverter implements Runnable {
reporter = referentials.userLogin((Long) row.get(REVIEW_REPORTER_ID));
}
- List<Map<String, Object>> comments = new QueryRunner().query(readConnection, ReviewCommentsHandler.SQL + reviewId, new ReviewCommentsHandler());
+ List<Map<String, Object>> comments = runner.query(readConnection, ReviewCommentsHandler.SQL + reviewId, new ReviewCommentsHandler());
for (Map<String, Object> comment : comments) {
comment.put(ISSUE_KEY, issueKey);
allComments.add(comment);
@@ -180,15 +221,12 @@ class ViolationConverter implements Runnable {
params[19] = updatedAt;
allParams.add(params);
}
- new QueryRunner().batch(writeConnection, SQL_INSERT_ISSUE, allParams.toArray(new Object[allParams.size()][]));
- writeConnection.commit();
-
+ runner.batch(writeConnection, SQL_INSERT_ISSUE, allParams.toArray(new Object[allParams.size()][]));
insertComments(writeConnection, allComments);
- progress.increment(rows.size());
+ runner.update(writeConnection, SQL_DELETE_RULE_FAILURES, violationIds);
- } catch (SQLException e) {
- //TODO
- throw new IllegalStateException();
+ writeConnection.commit();
+ progress.increment(rows.size());
} finally {
DbUtils.closeQuietly(readConnection);
@@ -214,7 +252,6 @@ class ViolationConverter implements Runnable {
}
if (!allParams.isEmpty()) {
new QueryRunner().batch(writeConnection, SQL_INSERT_ISSUE_CHANGE, allParams.toArray(new Object[allParams.size()][]));
- writeConnection.commit();
}
}
@@ -236,29 +273,6 @@ class ViolationConverter implements Runnable {
private static class ViolationHandler extends AbstractListHandler<Map<String, Object>> {
private static final Map<Integer, String> SEVERITIES = ImmutableMap.of(1, Severity.INFO, 2, Severity.MINOR, 3, Severity.MAJOR, 4, Severity.CRITICAL, 5, Severity.BLOCKER);
- static final String SQL;
-
- static {
- StringBuilder sb = new StringBuilder("select rev.id as reviewId, s.project_id as projectId, rf.rule_id as ruleId, " +
- " rf.failure_level as failureLevel, rf.message as message, rf.line as line, " +
- " rf.cost as cost, rf.created_at as createdAt, rf.checksum as checksum, rev.user_id as reviewReporterId, " +
- " rev.assignee_id as reviewAssigneeId, rev.status as reviewStatus, " +
- " rev.severity as reviewSeverity, rev.resolution as reviewResolution, rev.manual_severity as reviewManualSeverity, " +
- " rev.data as reviewData, rev.updated_at as reviewUpdatedAt, " +
- " s.root_project_id as rootProjectId, rev.manual_violation as reviewManualViolation, planreviews.action_plan_id as planId " +
- " from rule_failures rf " +
- " inner join snapshots s on s.id=rf.snapshot_id " +
- " left join reviews rev on rev.rule_failure_permanent_id=rf.permanent_id " +
- " left join action_plans_reviews planreviews on planreviews.review_id=rev.id " +
- " where ");
- for (int i = 0; i < ViolationMigration.GROUP_SIZE; i++) {
- if (i > 0) {
- sb.append(" or ");
- }
- sb.append("rf.id=?");
- }
- SQL = sb.toString();
- }
@Override
protected Map<String, Object> handleRow(ResultSet rs) throws SQLException {
diff --git a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverters.java b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverters.java
index a643fa7fdc9..492d95a6228 100644
--- a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverters.java
+++ b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationConverters.java
@@ -19,41 +19,54 @@
*/
package org.sonar.server.db.migrations.violation;
+import com.google.common.collect.Lists;
+import org.sonar.api.config.Settings;
import org.sonar.core.persistence.Database;
+import java.util.List;
import java.util.Timer;
-import java.util.concurrent.*;
+import java.util.concurrent.Callable;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.Future;
class ViolationConverters {
- static final int MAX_THREADS = 5;
+ static final int DEFAULT_THREADS = 5;
+ static final String THREADS_PROPERTY = "sonar.violationMigration.threads";
+ private final Settings settings;
- private final ExecutorService executorService;
- private final Database database;
- private final Referentials referentials;
- private final Progress progress;
- private final Timer timer;
-
- ViolationConverters(Database db, Referentials referentials) {
- this.database = db;
- this.referentials = referentials;
+ ViolationConverters(Settings settings) {
+ this.settings = settings;
+ }
- this.progress = new Progress(referentials.totalViolations());
- timer = new Timer(Progress.THREAD_NAME);
+ void execute(Referentials referentials, Database db) throws Exception {
+ Progress progress = new Progress(referentials.totalViolations());
+ Timer timer = new Timer(Progress.THREAD_NAME);
timer.schedule(progress, Progress.DELAY_MS, Progress.DELAY_MS);
- BlockingQueue<Runnable> blockingQueue = new ArrayBlockingQueue<Runnable>(MAX_THREADS);
- RejectedExecutionHandler rejectedExecutionHandler = new ThreadPoolExecutor.CallerRunsPolicy();
- this.executorService = new ThreadPoolExecutor(0, MAX_THREADS, 5L, TimeUnit.SECONDS, blockingQueue, rejectedExecutionHandler);
- }
+ List<Callable<Object>> converters = Lists.newArrayList();
+ for (int i = 0; i < numberOfThreads(); i++) {
+ converters.add(new ViolationConverter(referentials, db, progress));
+ }
+ ExecutorService executor = Executors.newFixedThreadPool(converters.size());
+ List<Future<Object>> results = executor.invokeAll(converters);
- void convert(Object[] violationIds) {
- executorService.execute(new ViolationConverter(referentials, database, violationIds, progress));
+ executor.shutdown();
+ for (Future result : results) {
+ result.get();
+ }
+ timer.cancel();
}
- void waitForFinished() throws InterruptedException {
- executorService.shutdown();
- executorService.awaitTermination(10L, TimeUnit.SECONDS);
- timer.cancel();
+ int numberOfThreads() {
+ int threads = settings.getInt(THREADS_PROPERTY);
+ if (threads < 0) {
+ throw new IllegalArgumentException(String.format("Bad value of %s: %d", THREADS_PROPERTY, threads));
+ }
+ if (threads == 0) {
+ threads = DEFAULT_THREADS;
+ }
+ return threads;
}
}
diff --git a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationMigration.java b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationMigration.java
index 86a94ddef0c..8dae2f15034 100644
--- a/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationMigration.java
+++ b/sonar-server/src/main/java/org/sonar/server/db/migrations/violation/ViolationMigration.java
@@ -19,17 +19,13 @@
*/
package org.sonar.server.db.migrations.violation;
-import org.apache.commons.dbutils.DbUtils;
-import org.apache.commons.dbutils.QueryRunner;
-import org.apache.commons.dbutils.ResultSetHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
+import org.sonar.api.config.Settings;
import org.sonar.api.utils.MessageException;
import org.sonar.core.persistence.Database;
import org.sonar.server.db.DatabaseMigration;
-import java.sql.Connection;
-import java.sql.ResultSet;
import java.sql.SQLException;
/**
@@ -37,15 +33,21 @@ import java.sql.SQLException;
*/
public class ViolationMigration implements DatabaseMigration {
- public static final int GROUP_SIZE = 1000;
+
+ private final Settings settings;
private Logger logger = LoggerFactory.getLogger(ViolationMigration.class);
+ private final Database db;
+
+ public ViolationMigration(Database database, Settings settings) {
+ this.db = database;
+ this.settings = settings;
+ }
@Override
- public void execute(Database db) {
+ public void execute() {
try {
- truncateIssueTables(db);
- migrate(db);
+ migrate();
} catch (SQLException e) {
logger.error("Fail to convert violations to issues", e);
@@ -58,72 +60,17 @@ public class ViolationMigration implements DatabaseMigration {
}
}
- private void truncateIssueTables(Database db) throws SQLException {
- Connection connection = null;
- try {
- QueryRunner runner = new QueryRunner();
- connection = db.getDataSource().getConnection();
- connection.setAutoCommit(true);
-
- // lower-case table names for SQLServer....
- runner.update(connection, "TRUNCATE TABLE issues");
- runner.update(connection, "TRUNCATE TABLE issue_changes");
-
- } finally {
- DbUtils.closeQuietly(connection);
- }
-
- }
-
- public void migrate(Database db) throws Exception {
+ public void migrate() throws Exception {
+ logger.info("Initialize input");
Referentials referentials = new Referentials(db);
- if (referentials.totalViolations() > 0) {
- ViolationConverters converters = new ViolationConverters(db, referentials);
- Connection readConnection = db.getDataSource().getConnection();
- try {
- new QueryRunner().query(readConnection, "select id from rule_failures", new ViolationIdHandler(converters));
- } finally {
- DbUtils.closeQuietly(readConnection);
- }
- converters.waitForFinished();
- }
- }
-
- private static class ViolationIdHandler implements ResultSetHandler {
- private final ViolationConverters converters;
- private ViolationIdHandler(ViolationConverters converters) {
- this.converters = converters;
- }
-
- @Override
- public Object handle(ResultSet rs) throws SQLException {
- // int is enough, it allows to upgrade up to 2 billions violations !
- int total = 0;
- int cursor = 0;
+ if (referentials.totalViolations() > 0) {
+ logger.info("Migrate {} violations", referentials.totalViolations());
- Object[] violationIds = new Object[GROUP_SIZE];
- while (rs.next()) {
- long violationId = rs.getLong(1);
- violationIds[cursor++] = violationId;
- if (cursor == GROUP_SIZE) {
- converters.convert(violationIds);
- violationIds = new Object[GROUP_SIZE];
- cursor = 0;
- }
- total++;
- }
- if (cursor > 0) {
- for (int i = 0; i < violationIds.length; i++) {
- if (violationIds[i] == null) {
- violationIds[i] = -1;
- }
- }
- converters.convert(violationIds);
- }
- LoggerFactory.getLogger(getClass()).info(String.format("%d violations migrated to issues", total));
- return null;
+ ViolationConverters converters = new ViolationConverters(settings);
+ converters.execute(referentials, db);
}
}
+
}
diff --git a/sonar-server/src/main/java/org/sonar/server/platform/Platform.java b/sonar-server/src/main/java/org/sonar/server/platform/Platform.java
index b473ca01d81..44b95743aa8 100644
--- a/sonar-server/src/main/java/org/sonar/server/platform/Platform.java
+++ b/sonar-server/src/main/java/org/sonar/server/platform/Platform.java
@@ -75,8 +75,10 @@ import org.sonar.server.component.DefaultComponentFinder;
import org.sonar.server.component.DefaultRubyComponentService;
import org.sonar.server.configuration.Backup;
import org.sonar.server.configuration.ProfilesManager;
+import org.sonar.server.db.DatabaseMigration;
import org.sonar.server.db.DatabaseMigrator;
import org.sonar.server.db.EmbeddedDatabaseFactory;
+import org.sonar.server.db.DatabaseMigrations;
import org.sonar.server.issue.*;
import org.sonar.server.notifications.NotificationCenter;
import org.sonar.server.notifications.NotificationService;
@@ -175,6 +177,9 @@ public final class Platform {
rootContainer.addSingleton(MyBatis.class);
rootContainer.addSingleton(DefaultServerUpgradeStatus.class);
rootContainer.addSingleton(DatabaseServerCompatibility.class);
+ for (Class<? extends DatabaseMigration> migrationClass : DatabaseMigrations.CLASSES) {
+ rootContainer.addSingleton(migrationClass);
+ }
rootContainer.addSingleton(DatabaseMigrator.class);
rootContainer.addSingleton(DatabaseVersion.class);
for (Class daoClass : DaoUtils.getDaoClasses()) {