/* * 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.db.version; import com.google.common.annotations.VisibleForTesting; import com.google.common.collect.ImmutableSet; import java.util.Collections; import java.util.List; import java.util.Set; import javax.annotation.Nullable; import org.apache.ibatis.session.SqlSession; import org.sonar.db.MyBatis; public class DatabaseVersion { public static final int LAST_VERSION = 1_308; /** * The minimum supported version which can be upgraded. Lower * versions must be previously upgraded to LTS version. * Note that the value can't be less than current LTS version. */ public static final int MIN_UPGRADE_VERSION = 600; /** * List of all the tables. * This list is hardcoded because we didn't succeed in using java.sql.DatabaseMetaData#getTables() in the same way * for all the supported databases, particularly due to Oracle results. */ public static final Set TABLES = ImmutableSet.of( "active_dashboards", "active_rules", "active_rule_parameters", "activities", "authors", "ce_activity", "ce_queue", "ce_task_input", "dashboards", "duplications_index", "events", "file_sources", "groups", "groups_users", "group_roles", "issues", "issue_changes", "issue_filters", "issue_filter_favourites", "loaded_templates", "manual_measures", "measure_filters", "measure_filter_favourites", "metrics", "notifications", "permission_templates", "perm_templates_users", "perm_templates_groups", "perm_tpl_characteristics", "quality_gates", "quality_gate_conditions", "projects", "project_links", "project_measures", "project_qprofiles", "properties", "resource_index", "rules", "rules_parameters", "rules_profiles", "scanner_context", "schema_migrations", "snapshots", "users", "user_roles", "user_tokens", "widgets", "widget_properties" ); private MyBatis mybatis; public DatabaseVersion(MyBatis mybatis) { this.mybatis = mybatis; } @VisibleForTesting static Status getStatus(@Nullable Integer currentVersion, int lastVersion) { Status status = Status.FRESH_INSTALL; if (currentVersion != null) { if (currentVersion == lastVersion) { status = Status.UP_TO_DATE; } else if (currentVersion > lastVersion) { status = Status.REQUIRES_DOWNGRADE; } else { status = Status.REQUIRES_UPGRADE; } } return status; } public Status getStatus() { return getStatus(getVersion(), LAST_VERSION); } public Integer getVersion() { SqlSession session = mybatis.openSession(false); try { List versions = session.getMapper(SchemaMigrationMapper.class).selectVersions(); if (!versions.isEmpty()) { Collections.sort(versions); return versions.get(versions.size() - 1); } return null; } catch (RuntimeException e) { // The table SCHEMA_MIGRATIONS does not exist. // Ignore this exception -> it will be created by Ruby on Rails migrations. return null; } finally { MyBatis.closeQuietly(session); } } public enum Status { UP_TO_DATE, REQUIRES_UPGRADE, REQUIRES_DOWNGRADE, FRESH_INSTALL } }