1
2
3
4
5
6
7
8
9
10
11
12
13
14
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
43
44 public class SQLExecutor {
45 private static final Log log = LogFactory.getLog(SQLExecutor.class);
46
47
48
49
50 public static final String ON_ERROR_ABORT = "abort";
51
52
53
54
55
56 public static final String ON_ERROR_ABORT_AFTER = "abortAfter";
57
58
59
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
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
228
229
230
231
232
233
234 protected void execSQL(String sql, PrintStream out) throws SQLException {
235
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
294
295
296
297
298
299
300
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
337
338
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);
348 }
349 }
350
351
352 if (!sql.toString().equals("")) {
353 execSQL(sql.toString(), out);
354 }
355 }
356
357
358
359
360
361
362
363
364
365
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 }