View Javadoc

1   /**
2    * Copyright 2004-2012 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.db;
17  
18  import java.io.BufferedReader;
19  import java.io.IOException;
20  import java.io.PrintStream;
21  import java.io.Reader;
22  import java.sql.Connection;
23  import java.sql.ResultSet;
24  import java.sql.ResultSetMetaData;
25  import java.sql.SQLException;
26  import java.sql.SQLWarning;
27  import java.sql.Statement;
28  import java.util.ArrayList;
29  import java.util.Enumeration;
30  import java.util.List;
31  import java.util.StringTokenizer;
32  import java.util.Vector;
33  
34  import org.apache.commons.lang.StringEscapeUtils;
35  import org.apache.commons.logging.Log;
36  import org.apache.commons.logging.LogFactory;
37  
38  import static org.apache.commons.io.IOUtils.*;
39  import static org.kuali.db.JDBCUtils.*;
40  
41  /**
42   * Executes SQL statements
43   */
44  public class SQLExecutor {
45  	private static final Log log = LogFactory.getLog(SQLExecutor.class);
46  
47  	/**
48  	 * Call {@link #setOnError(String)} with this value to abort SQL command execution if an error is found.
49  	 */
50  	public static final String ON_ERROR_ABORT = "abort";
51  
52  	/**
53  	 * Call {@link #setOnError(String)} with this value to continue SQL command execution until all commands have been
54  	 * attempted, then abort the build if an SQL error occurred in any of the commands.
55  	 */
56  	public static final String ON_ERROR_ABORT_AFTER = "abortAfter";
57  
58  	/**
59  	 * Call {@link #setOnError(String)} with this value to continue SQL command execution if an error is found.
60  	 */
61  	public static final String ON_ERROR_CONTINUE = "continue";
62  
63  	Vector<Transaction> transactions;
64  	boolean keepFormat = true;
65  	String delimiterType = "row";
66  	String delimiter = "/";
67  	boolean printResultSet = false;;
68  	String onError = ON_ERROR_ABORT;
69  	boolean showheaders = true;
70  	String outputDelimiter = ",";
71  	int totalStatements;
72  	int successfulStatements;
73  	Connection conn;
74  	Statement statement;
75  	boolean autocommit = false;
76  	boolean escapeProcessing = true;
77  	boolean skipOnConnectionError;
78  	boolean connectionError;
79  	boolean append = false;
80  	List<DatabaseListener> listeners = new ArrayList<DatabaseListener>();
81  
82  	public void addListener(DatabaseListener listener) {
83  		listeners.add(listener);
84  	}
85  
86  	protected void fireMessageLogged(String message) {
87  		fireMessageLogged(message, MessagePriority.INFO);
88  	}
89  
90  	protected void fireMessageLogged(String message, MessagePriority priority) {
91  		DatabaseEvent event = new DatabaseEvent(message, priority);
92  		for (DatabaseListener listener : listeners) {
93  			listener.messageLogged(event);
94  		}
95  	}
96  
97  	protected void fireBeginTransaction(Transaction transaction) {
98  		DatabaseEvent event = new DatabaseEvent();
99  		event.setTransaction(transaction);
100 		for (DatabaseListener listener : listeners) {
101 			listener.beginTransaction(event);
102 		}
103 	}
104 
105 	protected void fireFinishTransaction(Transaction transaction) {
106 		DatabaseEvent event = new DatabaseEvent();
107 		event.setTransaction(transaction);
108 		for (DatabaseListener listener : listeners) {
109 			listener.finishTransaction(event);
110 		}
111 	}
112 
113 	protected DatabaseEvent getExecuteSQLEvent(int totalStatements, String sql) {
114 		DatabaseEvent event = new DatabaseEvent();
115 		event.setSql(sql);
116 		event.setTotalStatements(totalStatements);
117 		return event;
118 	}
119 
120 	protected void fireBeforeExecuteSQL(int totalStatements, String sql) {
121 		for (DatabaseListener listener : listeners) {
122 			listener.beforeExecuteSQL(getExecuteSQLEvent(totalStatements, sql));
123 		}
124 	}
125 
126 	protected void fireAfterExecuteSQL(int totalStatements, String sql) {
127 		for (DatabaseListener listener : listeners) {
128 			listener.afterExecuteSQL(getExecuteSQLEvent(totalStatements, sql));
129 		}
130 	}
131 
132 	protected void fireAfterProcessingSQLResults(int totalStatements, int successfulStatements, int updateCountTotal, String sql) {
133 		DatabaseEvent event = getExecuteSQLEvent(totalStatements, sql);
134 		event.setSuccessfulStatements(successfulStatements);
135 		event.setUpdateCountTotal(updateCountTotal);
136 		for (DatabaseListener listener : listeners) {
137 			listener.afterExecuteSQL(event);
138 		}
139 	}
140 
141 	public void info(String message) {
142 		log.info(message);
143 		fireMessageLogged(message);
144 	}
145 
146 	public void debug(String message) {
147 		log.debug(message);
148 		fireMessageLogged(message, MessagePriority.DEBUG);
149 	}
150 
151 	public void error(Throwable throwable, String message) {
152 		if (throwable == null) {
153 			log.error(message);
154 		} else {
155 			log.error(message, throwable);
156 		}
157 		DatabaseEvent event = new DatabaseEvent(message, MessagePriority.ERROR);
158 		event.setException(throwable);
159 		for (DatabaseListener listener : listeners) {
160 			listener.messageLogged(event);
161 		}
162 	}
163 
164 	public void error(String message) {
165 		error(null, message);
166 	}
167 
168 	public void executeSql(String sql) throws SQLException {
169 		Transaction transaction = new Transaction();
170 		transaction.setSqlCommand(sql);
171 		executeTransaction(transaction);
172 	}
173 
174 	public void executeTransaction(Transaction transaction) throws SQLException {
175 		this.transactions = new Vector<Transaction>();
176 		transactions.add(transaction);
177 		execute();
178 	}
179 
180 	public void execute() throws SQLException {
181 		try {
182 			statement = conn.createStatement();
183 			statement.setEscapeProcessing(escapeProcessing);
184 
185 			// Process all transactions
186 			for (Enumeration<Transaction> e = transactions.elements(); e.hasMoreElements();) {
187 				Transaction t = (Transaction) e.nextElement();
188 
189 				fireBeginTransaction(t);
190 				runTransaction(t, System.out);
191 				fireFinishTransaction(t);
192 
193 				if (!autocommit) {
194 					debug("Committing transaction");
195 					conn.commit();
196 				}
197 			}
198 		} catch (IOException e) {
199 			throw new SQLException(e);
200 		} catch (SQLException e) {
201 			if (!autocommit && ON_ERROR_ABORT.equalsIgnoreCase(getOnError())) {
202 				rollbackQuietly(conn);
203 			}
204 			throw e;
205 		} finally {
206 			closeQuietly(statement, conn);
207 		}
208 
209 		log.info(getSuccessfulStatements() + " of " + getTotalStatements() + " SQL statements executed successfully");
210 
211 		if (ON_ERROR_ABORT_AFTER.equalsIgnoreCase(getOnError()) && totalStatements != successfulStatements) {
212 			throw new SQLException("Some SQL statements failed to execute");
213 		}
214 	}
215 
216 	protected void runTransaction(Transaction t, PrintStream out) throws IOException, SQLException {
217 		Reader in = null;
218 		try {
219 			in = t.getReader();
220 			runStatements(in, out);
221 		} finally {
222 			closeQuietly(in);
223 		}
224 	}
225 
226 	/**
227 	 * Exec the sql statement.
228 	 * 
229 	 * @param sql
230 	 *            query to execute
231 	 * @param out
232 	 *            the outputstream
233 	 */
234 	protected void execSQL(String sql, PrintStream out) throws SQLException {
235 		// Check and ignore empty statements
236 		if ("".equals(sql.trim())) {
237 			return;
238 		}
239 
240 		ResultSet resultSet = null;
241 		try {
242 			totalStatements++;
243 			debug("SQL: " + sql);
244 
245 			boolean ret;
246 			int updateCountTotal = 0;
247 
248 			fireBeforeExecuteSQL(totalStatements, sql);
249 			ret = statement.execute(sql);
250 			fireAfterExecuteSQL(totalStatements, sql);
251 			do {
252 				if (!ret) {
253 					int updateCount = statement.getUpdateCount();
254 					if (updateCount != -1) {
255 						updateCountTotal += updateCount;
256 					}
257 				} else {
258 					resultSet = statement.getResultSet();
259 					if (printResultSet) {
260 						printResultSet(resultSet, out);
261 					}
262 				}
263 				ret = statement.getMoreResults();
264 			} while (ret);
265 
266 			debug(updateCountTotal + " rows affected");
267 
268 			if (printResultSet) {
269 				StringBuffer line = new StringBuffer();
270 				line.append(updateCountTotal).append(" rows affected");
271 				out.println(line);
272 			}
273 
274 			SQLWarning warning = conn.getWarnings();
275 			while (warning != null) {
276 				debug(warning + " sql warning");
277 				warning = warning.getNextWarning();
278 			}
279 			conn.clearWarnings();
280 			successfulStatements++;
281 			fireAfterProcessingSQLResults(totalStatements, successfulStatements, updateCountTotal, sql);
282 		} catch (SQLException e) {
283 			error("Failed to execute: " + sql + "\n\n" + e.getMessage());
284 			if (ON_ERROR_ABORT.equalsIgnoreCase(getOnError())) {
285 				throw e;
286 			}
287 		} finally {
288 			closeQuietly(resultSet);
289 		}
290 	}
291 
292 	/**
293 	 * read in lines and execute them
294 	 * 
295 	 * @param reader
296 	 *            the reader
297 	 * @param out
298 	 *            the outputstream
299 	 * @throws SQLException
300 	 * @throws IOException
301 	 */
302 	protected void runStatements(Reader reader, PrintStream out) throws SQLException, IOException {
303 		String line;
304 
305 		StringBuffer sql = new StringBuffer();
306 
307 		BufferedReader in = new BufferedReader(reader);
308 
309 		while ((line = in.readLine()) != null) {
310 			if (!keepFormat) {
311 				line = line.trim();
312 			}
313 
314 			if (!keepFormat) {
315 				if (line.startsWith("//")) {
316 					continue;
317 				}
318 				if (line.startsWith("--")) {
319 					continue;
320 				}
321 				StringTokenizer st = new StringTokenizer(line);
322 				if (st.hasMoreTokens()) {
323 					String token = st.nextToken();
324 					if ("REM".equalsIgnoreCase(token)) {
325 						continue;
326 					}
327 				}
328 			}
329 
330 			if (!keepFormat) {
331 				sql.append(" ").append(line);
332 			} else {
333 				sql.append("\n").append(line);
334 			}
335 
336 			// SQL defines "--" as a comment to EOL
337 			// and in Oracle it may contain a hint
338 			// so we cannot just remove it, instead we must end it
339 			if (!keepFormat) {
340 				if (SqlSplitter.containsSqlEnd(line, delimiter) == SqlSplitter.NO_END) {
341 					sql.append("\n");
342 				}
343 			}
344 
345 			if ((delimiterType.equals(DelimiterType.NORMAL) && SqlSplitter.containsSqlEnd(line, delimiter) > 0) || (delimiterType.equals(DelimiterType.ROW) && line.trim().equals(delimiter))) {
346 				execSQL(sql.substring(0, sql.length() - delimiter.length()), out);
347 				sql.setLength(0); // clean buffer
348 			}
349 		}
350 
351 		// Catch any statements not followed by ;
352 		if (!sql.toString().equals("")) {
353 			execSQL(sql.toString(), out);
354 		}
355 	}
356 
357 	/**
358 	 * print any results in the result set.
359 	 * 
360 	 * @param rs
361 	 *            the resultset to print information about
362 	 * @param out
363 	 *            the place to print results
364 	 * @throws SQLException
365 	 *             on SQL problems.
366 	 */
367 	protected void printResultSet(ResultSet rs, PrintStream out) throws SQLException {
368 		if (rs == null) {
369 			return;
370 		}
371 		debug("Processing new result set.");
372 		ResultSetMetaData md = rs.getMetaData();
373 		int columnCount = md.getColumnCount();
374 		StringBuffer line = new StringBuffer();
375 		if (showheaders) {
376 			boolean first = true;
377 			for (int col = 1; col <= columnCount; col++) {
378 				String columnValue = md.getColumnName(col);
379 
380 				if (columnValue != null) {
381 					columnValue = columnValue.trim();
382 
383 					if (",".equals(outputDelimiter)) {
384 						columnValue = StringEscapeUtils.escapeCsv(columnValue);
385 					}
386 				}
387 
388 				if (first) {
389 					first = false;
390 				} else {
391 					line.append(outputDelimiter);
392 				}
393 				line.append(columnValue);
394 			}
395 			out.println(line);
396 			line = new StringBuffer();
397 		}
398 		while (rs.next()) {
399 			boolean first = true;
400 			for (int col = 1; col <= columnCount; col++) {
401 				String columnValue = rs.getString(col);
402 				if (columnValue != null) {
403 					columnValue = columnValue.trim();
404 
405 					if (",".equals(outputDelimiter)) {
406 						columnValue = StringEscapeUtils.escapeCsv(columnValue);
407 					}
408 				}
409 
410 				if (first) {
411 					first = false;
412 				} else {
413 					line.append(outputDelimiter);
414 				}
415 				line.append(columnValue);
416 			}
417 			out.println(line);
418 			line = new StringBuffer();
419 		}
420 		out.println();
421 	}
422 
423 	public boolean isKeepFormat() {
424 		return keepFormat;
425 	}
426 
427 	public void setKeepFormat(boolean keepFormat) {
428 		this.keepFormat = keepFormat;
429 	}
430 
431 	public String getDelimiterType() {
432 		return delimiterType;
433 	}
434 
435 	public void setDelimiterType(String delimiterType) {
436 		this.delimiterType = delimiterType;
437 	}
438 
439 	public String getDelimiter() {
440 		return delimiter;
441 	}
442 
443 	public void setDelimiter(String delimiter) {
444 		this.delimiter = delimiter;
445 	}
446 
447 	public boolean isPrintResultSet() {
448 		return printResultSet;
449 	}
450 
451 	public void setPrintResultSet(boolean printResultSet) {
452 		this.printResultSet = printResultSet;
453 	}
454 
455 	public String getOnError() {
456 		return onError;
457 	}
458 
459 	public void setOnError(String onError) {
460 		this.onError = onError;
461 	}
462 
463 	public boolean isShowheaders() {
464 		return showheaders;
465 	}
466 
467 	public void setShowheaders(boolean showheaders) {
468 		this.showheaders = showheaders;
469 	}
470 
471 	public String getOutputDelimiter() {
472 		return outputDelimiter;
473 	}
474 
475 	public void setOutputDelimiter(String outputDelimiter) {
476 		this.outputDelimiter = outputDelimiter;
477 	}
478 
479 	public int getTotalStatements() {
480 		return totalStatements;
481 	}
482 
483 	public void setTotalStatements(int totalStatements) {
484 		this.totalStatements = totalStatements;
485 	}
486 
487 	public int getSuccessfulStatements() {
488 		return successfulStatements;
489 	}
490 
491 	public void setSuccessfulStatements(int successfulStatements) {
492 		this.successfulStatements = successfulStatements;
493 	}
494 
495 	public Connection getConn() {
496 		return conn;
497 	}
498 
499 	public void setConn(Connection conn) {
500 		this.conn = conn;
501 	}
502 
503 	public Statement getStatement() {
504 		return statement;
505 	}
506 
507 	public void setStatement(Statement statement) {
508 		this.statement = statement;
509 	}
510 
511 	public boolean isAutocommit() {
512 		return autocommit;
513 	}
514 
515 	public void setAutocommit(boolean autocommit) {
516 		this.autocommit = autocommit;
517 	}
518 
519 	public boolean isEscapeProcessing() {
520 		return escapeProcessing;
521 	}
522 
523 	public void setEscapeProcessing(boolean escapeProcessing) {
524 		this.escapeProcessing = escapeProcessing;
525 	}
526 
527 	public boolean isAppend() {
528 		return append;
529 	}
530 
531 	public void setAppend(boolean append) {
532 		this.append = append;
533 	}
534 
535 	public boolean isSkipOnConnectionError() {
536 		return skipOnConnectionError;
537 	}
538 
539 	public void setSkipOnConnectionError(boolean skipOnConnectionError) {
540 		this.skipOnConnectionError = skipOnConnectionError;
541 	}
542 
543 	public boolean isConnectionError() {
544 		return connectionError;
545 	}
546 
547 	public void setConnectionError(boolean connectionError) {
548 		this.connectionError = connectionError;
549 	}
550 
551 	public List<DatabaseListener> getListeners() {
552 		return listeners;
553 	}
554 
555 	public void setListeners(List<DatabaseListener> listeners) {
556 		this.listeners = listeners;
557 	}
558 
559 	public Vector<Transaction> getTransactions() {
560 		return transactions;
561 	}
562 
563 	public void setTransactions(Vector<Transaction> transactions) {
564 		this.transactions = transactions;
565 	}
566 }