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
107
108
109
110
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
122
123 protected void fillInOptions(MySqlDumpContext context) {
124
125 List<String> options = context.getOptions() == null ? new ArrayList<String>() : context.getOptions();
126
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
138 context.setOptions(options);
139 }
140 }