View Javadoc

1   package org.kuali.common.util.service;
2   
3   import java.io.File;
4   import java.io.IOException;
5   import java.io.PrintStream;
6   import java.util.ArrayList;
7   import java.util.List;
8   
9   import org.apache.commons.io.IOUtils;
10  import org.apache.commons.lang3.StringUtils;
11  import org.kuali.common.util.Assert;
12  import org.kuali.common.util.CollectionUtils;
13  import org.kuali.common.util.FormatUtils;
14  import org.kuali.common.util.LocationUtils;
15  import org.kuali.common.util.PrintlnStreamConsumer;
16  import org.slf4j.Logger;
17  import org.slf4j.LoggerFactory;
18  
19  public class DefaultMySqlDumpService extends DefaultExecService implements MySqlDumpService {
20  
21  	private static final Logger logger = LoggerFactory.getLogger(DefaultMySqlDumpService.class);
22  
23  	@Override
24  	public void dump(String username, String password, String hostname, String database, File outputFile) {
25  		MySqlDumpContext context = new MySqlDumpContext();
26  		context.setExecutable(DEFAULT_EXECUTABLE);
27  		context.setUsername(username);
28  		context.setPassword(password);
29  		context.setHostname(hostname);
30  		context.setDatabase(database);
31  		context.setOutputFile(outputFile);
32  		dump(context);
33  	}
34  
35  	@Override
36  	public void dump(List<String> options, String database, File outputFile) {
37  		MySqlDumpContext context = new MySqlDumpContext();
38  		context.setExecutable(DEFAULT_EXECUTABLE);
39  		context.setOptions(options);
40  		context.setDatabase(database);
41  		context.setOutputFile(outputFile);
42  		dump(context);
43  	}
44  
45  	@Override
46  	public void dump(MySqlDumpContext context) {
47  		Assert.notNull(context.getDatabase(), "database is null");
48  		Assert.notNull(context.getOutputFile(), "output file is null");
49  		Assert.notNull(context.getExecutable(), "executable is null");
50  		fillInOptions(context);
51  		DefaultExecContext dec = getExecContext(context);
52  		beforeDump(context);
53  		dump(dec, context);
54  	}
55  
56  	protected void beforeDump(MySqlDumpContext context) {
57  		String username = StringUtils.trimToEmpty(context.getUsername());
58  		String hostname = StringUtils.trimToEmpty(context.getHostname());
59  		int port = context.getPort();
60  		String database = context.getDatabase();
61  		String path = LocationUtils.getCanonicalPath(context.getOutputFile());
62  		Object[] args = { username, hostname, port, database, path };
63  		logger.info("Dumping [{}@{}:{}/{}] -> [{}]", args);
64  	}
65  
66  	protected void dump(DefaultExecContext context, MySqlDumpContext msdc) {
67  		PrintStream out = null;
68  		try {
69  			out = LocationUtils.openPrintStream(msdc.getOutputFile());
70  			PrintlnStreamConsumer standardOutConsumer = new PrintlnStreamConsumer(out, msdc.getIgnorers());
71  			context.setStandardOutConsumer(standardOutConsumer);
72  			long start = System.currentTimeMillis();
73  			int result = execute(context);
74  			long elapsed = System.currentTimeMillis() - start;
75  			if (result != 0) {
76  				throw new IllegalStateException("Non-zero exit value - " + result);
77  			}
78  			afterDump(msdc, elapsed, standardOutConsumer.getLineCount(), standardOutConsumer.getSkipCount());
79  		} catch (IOException e) {
80  			throw new IllegalStateException("Unexpected IO error", e);
81  		} finally {
82  			IOUtils.closeQuietly(out);
83  		}
84  	}
85  
86  	protected void afterDump(MySqlDumpContext context, long elapsed, long lineCount, long skippedCount) {
87  		long length = context.getOutputFile().length();
88  		String time = FormatUtils.getTime(elapsed);
89  		String size = FormatUtils.getSize(length);
90  		String rate = FormatUtils.getRate(elapsed, length);
91  		String lines = FormatUtils.getCount(lineCount);
92  		String skipped = FormatUtils.getCount(skippedCount);
93  		Object[] args = { time, size, rate, lines, skipped };
94  		logger.info("Dump completed. [Time:{}, Size:{}, Rate:{}, Lines:{}  Skipped:{}]", args);
95  	}
96  
97  	protected DefaultExecContext getExecContext(MySqlDumpContext context) {
98  		List<String> args = getArgs(context);
99  		DefaultExecContext dec = new DefaultExecContext();
100 		dec.setExecutable(context.getExecutable());
101 		dec.setArgs(args);
102 		return dec;
103 	}
104 
105 	/**
106 	 * <code>mysqldump</code> invocation looks like this:
107 	 *
108 	 * <pre>
109 	 * mysqldump [OPTIONS] database [tables]
110 	 * </pre>
111 	 */
112 	protected List<String> getArgs(MySqlDumpContext context) {
113 		List<String> args = new ArrayList<String>();
114 		args.addAll(CollectionUtils.toEmptyList(context.getOptions()));
115 		args.add(context.getDatabase());
116 		args.addAll(CollectionUtils.toEmptyList(context.getTables()));
117 		return args;
118 	}
119 
120 	/**
121 	 * Create (or update) the list of options for this context
122 	 */
123 	protected void fillInOptions(MySqlDumpContext context) {
124 		// Get a handle to the existing options list, or create new one
125 		List<String> options = context.getOptions() == null ? new ArrayList<String>() : context.getOptions();
126 		// Insert the options we are explicitly managing at the front of the list
127 		options.add(0, "--port=" + context.getPort());
128 		if (!StringUtils.isBlank(context.getHostname())) {
129 			options.add(0, "--host=" + context.getHostname());
130 		}
131 		if (!StringUtils.isBlank(context.getPassword())) {
132 			options.add(0, "--password=" + context.getPassword());
133 		}
134 		if (!StringUtils.isBlank(context.getUsername())) {
135 			options.add(0, "--user=" + context.getUsername());
136 		}
137 		// Just in case there were no options to begin with
138 		context.setOptions(options);
139 	}
140 }