blob: d8368f5b4e879baa5b6c5055f98146c867fb18f5 [file] [log] [blame]
/**
*
* 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.
*/
package org.apache.airavata.allocation.manager.db.utils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.sql.*;
import java.util.StringTokenizer;
/**
* This class creates the database tables required for airavata with default configuration this
* class creates derby database in server mode. User can specify required database in appropriate
* properties files.
*/
public class DatabaseCreator {
private final static Logger logger = LoggerFactory.getLogger(DatabaseCreator.class);
public enum DatabaseType {
derby("(?i).*derby.*"), mysql("(?i).*mysql.*"), other("");
private String pattern;
private DatabaseType(String matchingPattern) {
this.pattern = matchingPattern;
}
public String getMatchingPattern() {
return this.pattern;
}
}
private static DatabaseType[] supportedDatabase = new DatabaseType[] { DatabaseType.derby, DatabaseType.mysql };
private static Logger log = LoggerFactory.getLogger(DatabaseCreator.class);
private static final String delimiter = ";";
/**
* Creates database
*
* @throws Exception
*/
public static void createRegistryDatabase(String prefix, Connection conn) throws Exception {
createDatabase(prefix, conn);
}
/**
* Checks whether database tables are created by using select * on given table name
*
* @param tableName
* Table which should be existed
* @return <code>true</core> if checkSQL is success, else <code>false</code> .
*/
public static boolean isDatabaseStructureCreated(String tableName, Connection conn) {
try {
log.debug("Running a query to test the database tables existence.");
// check whether the tables are already created with a query
Statement statement = null;
try {
statement = conn.createStatement();
ResultSet rs = statement.executeQuery("select * from " + tableName);
if (rs != null) {
rs.close();
}
} finally {
try {
if (statement != null) {
statement.close();
}
} catch (SQLException e) {
return false;
}
}
} catch (SQLException e) {
return false;
}
return true;
}
/**
* executes given sql
*
* @param sql
* @throws Exception
*/
private static void executeSQL(String sql, Connection conn) throws Exception {
// Check and ignore empty statements
if ("".equals(sql.trim())) {
return;
}
Statement statement = null;
try {
log.debug("SQL : " + sql);
boolean ret;
int updateCount = 0, updateCountTotal = 0;
statement = conn.createStatement();
ret = statement.execute(sql);
updateCount = statement.getUpdateCount();
do {
if (!ret) {
if (updateCount != -1) {
updateCountTotal += updateCount;
}
}
ret = statement.getMoreResults();
if (ret) {
updateCount = statement.getUpdateCount();
}
} while (ret);
log.debug(sql + " : " + updateCountTotal + " rows affected");
SQLWarning warning = conn.getWarnings();
while (warning != null) {
log.info(warning + " sql warning");
warning = warning.getNextWarning();
}
conn.clearWarnings();
} catch (SQLException e) {
if (e.getSQLState().equals("X0Y32")) {
// eliminating the table already exception for the derby
// database
log.info("Table Already Exists", e);
} else {
throw new Exception("Error occurred while executing : " + sql, e);
}
} finally {
if (statement != null) {
try {
statement.close();
} catch (SQLException e) {
log.error("Error occurred while closing result set.", e);
}
}
}
}
/**
* computes relatational database type using database name
*
* @return DatabaseType
* @throws Exception
*
*/
public static DatabaseType getDatabaseType(Connection conn) throws Exception {
try {
if (conn != null && (!conn.isClosed())) {
DatabaseMetaData metaData = conn.getMetaData();
String databaseProductName = metaData.getDatabaseProductName();
return checkType(databaseProductName);
}
} catch (SQLException e) {
String msg = "Failed to create Airavata database." + e.getMessage();
log.error(msg, e);
throw new Exception(msg, e);
}
return DatabaseType.other;
}
/**
* Overloaded method with String input
*
* @return DatabaseType
* @throws Exception
*
*/
public static DatabaseType getDatabaseType(String dbUrl) throws Exception {
return checkType(dbUrl);
}
private static DatabaseType checkType(String text) throws Exception {
try {
if (text != null) {
for (DatabaseType type : supportedDatabase) {
if (text.matches(type.getMatchingPattern()))
return type;
}
}
String msg = "Unsupported database: " + text
+ ". Database will not be created automatically by the Airavata. "
+ "Please create the database using appropriate database scripts for " + "the database.";
throw new Exception(msg);
} catch (SQLException e) {
String msg = "Failed to create Airavatadatabase." + e.getMessage();
log.error(msg, e);
throw new Exception(msg, e);
}
}
/**
* Get scripts location which is prefix + "-" + databaseType + ".sql"
*
* @param prefix
* @param databaseType
* @return script location
*/
private static String getScriptLocation(String prefix, DatabaseType databaseType) {
String scriptName = prefix + "-" + databaseType + ".sql";
log.debug("Loading database script from :" + scriptName);
return scriptName;
}
private static void createDatabase(String prefix, Connection conn) throws Exception {
Statement statement = null;
try {
conn.setAutoCommit(false);
statement = conn.createStatement();
executeSQLScript(getScriptLocation(prefix, DatabaseCreator.getDatabaseType(conn)), conn);
conn.commit();
log.debug("Tables are created successfully.");
} catch (SQLException e) {
String msg = "Failed to create database tables for Airavata resource store. " + e.getMessage();
log.error(msg, e);
conn.rollback();
throw new Exception(msg, e);
} finally {
conn.setAutoCommit(true);
try {
if (statement != null) {
statement.close();
}
} catch (SQLException e) {
log.error("Failed to close statement.", e);
}
}
}
private static void executeSQLScript(String dbscriptName, Connection conn) throws Exception {
StringBuffer sql = new StringBuffer();
BufferedReader reader = null;
try {
InputStream is = DatabaseCreator.class.getClassLoader().getResourceAsStream(dbscriptName);
if(is == null) {
logger.info("Script file not found at " + dbscriptName + ". Uses default database script file");
DatabaseType databaseType = DatabaseCreator.getDatabaseType(conn);
if(databaseType.equals(DatabaseType.derby)){
is = DatabaseCreator.class.getClassLoader().getResourceAsStream("sharing-registry-derby.sql");
}else if(databaseType.equals(DatabaseType.mysql)){
is = DatabaseCreator.class.getClassLoader().getResourceAsStream("sharing-registry-mysql.sql");
}
}
reader = new BufferedReader(new InputStreamReader(is));
String line;
while ((line = reader.readLine()) != null) {
line = line.trim();
if (line.startsWith("//")) {
continue;
}
if (line.startsWith("--")) {
continue;
}
StringTokenizer st = new StringTokenizer(line);
if (st.hasMoreTokens()) {
String token = st.nextToken();
if ("REM".equalsIgnoreCase(token)) {
continue;
}
}
sql.append(" ").append(line);
// SQL defines "--" as a comment to EOL
// and in Oracle it may contain a hint
// so we cannot just remove it, instead we must end it
if (line.indexOf("--") >= 0) {
sql.append("\n");
}
if ((checkStringBufferEndsWith(sql, delimiter))) {
executeSQL(sql.substring(0, sql.length() - delimiter.length()), conn);
sql.replace(0, sql.length(), "");
}
}
// Catch any statements not followed by ;
if (sql.length() > 0) {
executeSQL(sql.toString(), conn);
}
} catch (IOException e) {
log.error("Error occurred while executing SQL script for creating Airavata database", e);
throw new Exception("Error occurred while executing SQL script for creating Airavata database", e);
} finally {
if (reader != null) {
reader.close();
}
}
}
/**
* Checks that a string buffer ends up with a given string. It may sound trivial with the existing JDK API but the
* various implementation among JDKs can make those methods extremely resource intensive and perform poorly due to
* massive memory allocation and copying. See
*
* @param buffer
* the buffer to perform the check on
* @param suffix
* the suffix
* @return <code>true</code> if the character sequence represented by the argument is a suffix of the character
* sequence represented by the StringBuffer object; <code>false</code> otherwise. Note that the result will
* be <code>true</code> if the argument is the empty string.
*/
public static boolean checkStringBufferEndsWith(StringBuffer buffer, String suffix) {
if (suffix.length() > buffer.length()) {
return false;
}
// this loop is done on purpose to avoid memory allocation performance
// problems on various JDKs
// StringBuffer.lastIndexOf() was introduced in jdk 1.4 and
// implementation is ok though does allocation/copying
// StringBuffer.toString().endsWith() does massive memory
// allocation/copying on JDK 1.5
// See http://issues.apache.org/bugzilla/show_bug.cgi?id=37169
int endIndex = suffix.length() - 1;
int bufferIndex = buffer.length() - 1;
while (endIndex >= 0) {
if (buffer.charAt(bufferIndex) != suffix.charAt(endIndex)) {
return false;
}
bufferIndex--;
endIndex--;
}
return true;
}
}