001    /**
002     * Copyright 2004-2013 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.apache.torque.mojo;
017    
018    import java.io.File;
019    import java.io.IOException;
020    import java.io.OutputStream;
021    import java.util.ArrayList;
022    import java.util.List;
023    
024    import org.apache.commons.io.FileUtils;
025    import org.apache.commons.io.IOUtils;
026    import org.apache.commons.lang.StringUtils;
027    import org.apache.maven.plugin.AbstractMojo;
028    import org.apache.maven.plugin.MojoExecutionException;
029    import org.apache.torque.util.SimpleScanner;
030    
031    /**
032     * Examine SQL files and convert them to always use the delimiter <code>/</code> on it's own line. Convert any lines ending with
033     * <code>;</code> to use a <code>/</code> on it's own line.
034     *
035     * @goal convertsql
036     */
037    public class ConvertSQLMojo extends AbstractMojo {
038            private static final String FS = File.separator;
039    
040            /**
041             * The encoding to use when reading/writing files
042             *
043             * @parameter expression="${impex.encoding}" default-value="${project.build.sourceEncoding}"
044             */
045            private String encoding;
046    
047            /**
048             * The old delimiter used to terminate a SQL statement by being present at the end of a line.
049             *
050             * @parameter expression="${impex.oldDelimiter}" default-value=";"
051             * @required
052             */
053            private String oldDelimiter;
054    
055            /**
056             * The new delimiter to put on its own line
057             *
058             * @parameter expression="${impex.newDelimiter}" default-value="/"
059             * @required
060             */
061            private String newDelimiter;
062    
063            /**
064             * Directory to examine for .sql files
065             *
066             * @parameter expression="${impex.sourceDir}" default-value="${project.build.directory}/sql/source"
067             * @required
068             */
069            private File sourceDir;
070    
071            /**
072             * Directory to generate the converted files into
073             *
074             * @parameter expression="${impex.outputDir}" default-value="${project.build.directory}/sql/output"
075             * @required
076             */
077            private File outputDir;
078    
079            /**
080             * CSV list of regex patterns for files to include
081             *
082             * @parameter expression="${impex.includes}" default-value="\*\*\/*.sql"
083             * @required
084             */
085            private String includes;
086    
087            /**
088             * CSV list of regex patterns for files to exclude
089             *
090             * @parameter expression="${impex.excludes}" default-value="\*\*\/.svn/**,\*\*\/.git/**"
091             */
092            private String excludes;
093    
094            /**
095             * Liquibase injects a handful of metadata at the top of each SQL file that causes "noise". All of the checksums and actual SQL may stay
096             * the same, but the Liquibase metadata makes it appear as though the resulting SQL file has changed. By omitting the metadata you can
097             * run the same change set multiple times and get the exact same SQL file as output. This makes it much easier to perform validation
098             * checks on the SQL.
099             *
100             * @parameter expression="${impex.skipIrrelevantLiquibaseMetadataLines}" default-value="false"
101             */
102            private boolean skipIrrelevantLiquibaseMetadataLines;
103    
104            private List<String> liquibaseTokens = getLiquibaseTokens();
105    
106            @Override
107            public void execute() throws MojoExecutionException {
108                    try {
109                            // Make sure the sourceDir exists
110                            FileUtils.forceMkdir(sourceDir);
111                            getLog().info("Source Dir - " + sourceDir.getCanonicalPath());
112                            getLog().info("Output Dir - " + outputDir.getCanonicalPath());
113                            getLog().info("Includes - " + includes);
114                            getLog().info("Excludes - " + excludes);
115                            getLog().info("Old Delimiter - " + oldDelimiter);
116                            getLog().info("New Delimiter - " + newDelimiter);
117                            getLog().info("Encoding - " + encoding);
118                            List<File> files = getFiles();
119                            if (files == null || files.size() == 0) {
120                                    getLog().info("No files found");
121                                    return;
122                            }
123                            getLog().info("Found " + files.size() + " SQL files to convert");
124                            convert(files);
125                    } catch (Exception e) {
126                            throw new MojoExecutionException("Unexpected error", e);
127                    }
128            }
129    
130            protected void convert(List<File> files) throws IOException {
131                    for (File file : files) {
132                            convert(file);
133                    }
134            }
135    
136            protected List<String> getLiquibaseTokens() {
137                    List<String> tokens = new ArrayList<String>();
138                    tokens.add("-- Ran at:");
139                    tokens.add("-- Against:");
140                    tokens.add("-- Liquibase version:");
141                    tokens.add("--  Ran at:");
142                    tokens.add("--  Against:");
143                    tokens.add("--  Liquibase version:");
144                    return tokens;
145            }
146    
147            protected boolean isSkipLine(String line) {
148                    if (!skipIrrelevantLiquibaseMetadataLines) {
149                            return false;
150                    }
151                    for (String token : liquibaseTokens) {
152                            if (line.startsWith(token)) {
153                                    return true;
154                            }
155                    }
156                    return false;
157            }
158    
159            protected String getRelativePath(File dir, File file) throws IOException {
160                    String dirPath = dir.getCanonicalPath() + FS;
161                    String filePath = file.getCanonicalPath();
162                    return StringUtils.remove(filePath, dirPath);
163            }
164    
165            protected void convert(File file) throws IOException {
166                    String outputFilename = outputDir + FS + getRelativePath(sourceDir, file);
167                    File outputFile = new File(outputFilename);
168                    @SuppressWarnings("unchecked")
169                    List<String> lines = FileUtils.readLines(file, encoding);
170                    getLog().info("Writing " + outputFile.getCanonicalPath());
171                    OutputStream out = null;
172                    try {
173                            out = FileUtils.openOutputStream(outputFile);
174                            for (String line : lines) {
175                                    if (isSkipLine(line)) {
176                                            continue;
177                                    }
178                                    String convertedLine = getConvertedLine(line);
179                                    out.write(convertedLine.getBytes(encoding));
180                            }
181                    } finally {
182                            IOUtils.closeQuietly(out);
183                    }
184            }
185    
186            protected String getConvertedLine(String line) {
187                    String trimmed = StringUtils.trimToNull(line);
188                    if (StringUtils.endsWith(trimmed, oldDelimiter)) {
189                            int pos = line.lastIndexOf(oldDelimiter);
190                            return line.substring(0, pos) + IOUtils.LINE_SEPARATOR_UNIX + newDelimiter + IOUtils.LINE_SEPARATOR_UNIX;
191                    } else {
192                            return line + IOUtils.LINE_SEPARATOR_UNIX;
193                    }
194            }
195    
196            protected List<File> getFiles() throws IOException {
197                    getLog().info("Examining " + sourceDir.getCanonicalPath());
198                    String[] includeTokens = StringUtils.split(includes, ",");
199                    String[] excludeTokens = StringUtils.split(excludes, ",");
200                    SimpleScanner scanner = new SimpleScanner(sourceDir, includeTokens, excludeTokens);
201                    return scanner.getFiles();
202            }
203    
204            public File getSourceDir() {
205                    return sourceDir;
206            }
207    
208            public void setSourceDir(File inputDir) {
209                    this.sourceDir = inputDir;
210            }
211    
212            public File getOutputDir() {
213                    return outputDir;
214            }
215    
216            public void setOutputDir(File outputDir) {
217                    this.outputDir = outputDir;
218            }
219    
220            public String getIncludes() {
221                    return includes;
222            }
223    
224            public void setIncludes(String includes) {
225                    this.includes = includes;
226            }
227    
228            public String getExcludes() {
229                    return excludes;
230            }
231    
232            public void setExcludes(String excludes) {
233                    this.excludes = excludes;
234            }
235    
236            public String getEncoding() {
237                    return encoding;
238            }
239    
240            public void setEncoding(String encoding) {
241                    this.encoding = encoding;
242            }
243    
244            public String getOldDelimiter() {
245                    return oldDelimiter;
246            }
247    
248            public void setOldDelimiter(String oldDelimiter) {
249                    this.oldDelimiter = oldDelimiter;
250            }
251    
252            public String getNewDelimiter() {
253                    return newDelimiter;
254            }
255    
256            public void setNewDelimiter(String newDelimiter) {
257                    this.newDelimiter = newDelimiter;
258            }
259    
260            public boolean isSkipIrrelevantLiquibaseMetadataLines() {
261                    return skipIrrelevantLiquibaseMetadataLines;
262            }
263    
264            public void setSkipIrrelevantLiquibaseMetadataLines(boolean skipIrrelevantLiquibaseMetadataLines) {
265                    this.skipIrrelevantLiquibaseMetadataLines = skipIrrelevantLiquibaseMetadataLines;
266            }
267    
268    }