View Javadoc
1   /*
2    * Copyright (C) 2003-2012 David E. Berry
3    *
4    * This library is free software; you can redistribute it and/or
5    * modify it under the terms of the GNU Lesser General Public
6    * License as published by the Free Software Foundation; either
7    * version 2.1 of the License, or (at your option) any later version.
8    *
9    * This library is distributed in the hope that it will be useful,
10   * but WITHOUT ANY WARRANTY; without even the implied warranty of
11   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12   * Lesser General Public License for more details.
13   *
14   * You should have received a copy of the GNU Lesser General Public
15   * License along with this library; if not, write to the Free Software
16   * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
17   *
18   * A copy of the GNU Lesser General Public License may also be found at
19   * http://www.gnu.org/licenses/lgpl.txt
20   */
21  package org.synchronoss.cpo.parser;
22  
23  import org.slf4j.*;
24  
25  import java.io.StringReader;
26  import java.text.ParseException;
27  import java.util.*;
28  
29  /**
30   * @author Michael Bellomo
31   * @since 10/20/2008
32   */
33  public class BoundExpressionParser implements ExpressionParser {
34  
35    private static final Logger logger = LoggerFactory.getLogger(BoundExpressionParser.class);
36  
37    private final static String COMPARE_CHARS = " =<>!()";
38    private final static String SEPARATOR_CHARS = " .,()\n";
39  
40    private String expression;
41  
42    public BoundExpressionParser() {
43    }
44  
45    @Override
46    public String getExpression() {
47      return expression;
48    }
49  
50    public void setExpression(String expression) {
51      this.expression = expression;
52    }
53  
54    /**
55     * Returns the count of the bind markers in the expression
56     *
57     * @return the number of bind markers
58     */
59    @Override
60    public int countArguments() {
61      return getArgumentIndexes().size();
62    }
63  
64    private Collection<Integer> getArgumentIndexes() {
65      Collection<Integer> indexes = new ArrayList<>();
66  
67      if (expression != null) {
68        StringReader reader = new StringReader(expression);
69  
70        try {
71  
72          int idx = 0;
73          int rc = -1;
74          boolean inDoubleQuotes = false;
75          boolean inSingleQuotes = false;
76  
77          do {
78            rc = reader.read();
79            if (((char)rc) == '\'') {
80              inSingleQuotes = !inSingleQuotes;
81            } else if (((char)rc) == '"') {
82              inDoubleQuotes = !inDoubleQuotes;
83            } else if (!inSingleQuotes && !inDoubleQuotes && ((char)rc) == '?') {
84              indexes.add(idx);
85            }
86            idx++;
87          } while (rc != -1);
88        } catch (Exception e) {
89          logger.error("error counting bind markers");
90        }
91      }
92      return indexes;
93    }
94  
95    /**
96     * Returns a list of columns from the expression for each bind marker
97     *
98     * @return List of Strings for the columns for the bind markers
99     * @throws ParseException thrown if the expression cannot be parsed
100    */
101   @Override
102   public List<String> parse() throws ParseException {
103 
104     if (expression == null)
105       throw new ParseException("The expression is null", -1);
106 
107     if (logger.isDebugEnabled())
108       logger.debug("Expression: " + expression);
109 
110     // expression is empty, nothing we can do
111     if (expression.length() < 1)
112       return null;
113 
114     // no question marks, nothing to do
115     if (!expression.contains("?")) {
116       return null;
117     }
118 
119     // upper case the expression, to make things easier
120     expression = expression.toUpperCase();
121 
122     List<String> colList = new ArrayList<>();
123 
124     if (expression.startsWith("INSERT")) {
125       // expression is in the format of:  insert into table(col1, col2...) values(val1, val2...)
126       // so we'll use the parens () to parse
127 
128       int colParenStart = expression.indexOf("(");
129       if (colParenStart == -1)
130         throw new ParseException("Unable to locate starting parenthesis for the column names.", -1);
131 
132       int colParenEnd = expression.indexOf(")", colParenStart);
133       if (colParenEnd == -1)
134         throw new ParseException("Unable to locate ending parenthesis for the column names.", -1);
135 
136       int valParenStart = expression.indexOf("(", colParenEnd);
137       if (valParenStart == -1)
138         throw new ParseException("Unable to locate starting parenthesis for the column values.", -1);
139 
140       // use the last close paren, this will make weird inner select stuff work,
141       // but it won't be able to guess the inner select values
142       int valParenEnd = expression.lastIndexOf(")");
143       if (valParenEnd == -1)
144         throw new ParseException("Unable to locate ending parenthesis for the column values.", -1);
145 
146       String[] cols = expression.substring(colParenStart + 1, colParenEnd).split(",");
147       String[] vals = expression.substring(valParenStart + 1, valParenEnd).split(",");
148 
149       // if cols or vals is null, it means we couldn't find any
150       if (cols == null || vals == null)
151         return null;
152 
153       if (logger.isDebugEnabled()) {
154         logger.debug("Found cols: " + cols.length);
155         logger.debug("Found vals: " + vals.length);
156       }
157 
158       if (cols.length != vals.length)
159         throw new ParseException("You seem to have " + cols.length + " columns, and " + vals.length + " values.\n\nThose numbers should be equal.", -1);
160 
161       // filter out columns that we're not providing values for
162       for (int i = 0; i < vals.length; i++) {
163         String val = vals[i];
164         if (val.trim().equals("?")) {
165           // just a ?, use the col
166           colList.add(cols[i].trim());
167         } else if (val.contains("?")) {
168           // more than just a ?, parse the val
169           ExpressionParser qp = new BoundExpressionParser();
170           qp.setExpression(val);
171           colList.addAll(qp.parse());
172         }
173       }
174     } else {
175       // expression is in the format of:  ...col1 = ? , col2 = ?...
176       // so we'll have to move left to right from the ? looking for the field name
177 
178       int startIdx = 0;
179       Collection<Integer> indexes = getArgumentIndexes();
180       for (int qIdx : indexes) {
181         String chunk = expression.substring(startIdx, qIdx);
182 
183         if (logger.isDebugEnabled())
184           logger.debug("Chunk [" + chunk + "]");
185 
186         int idx = chunk.length() - 1;
187         int fieldStartIdx = -1;
188         int fieldEndIdx = -1;
189 
190         boolean found = false;
191         boolean inFunction = false;
192         while (!found && (idx >= 0)) {
193           char c = chunk.charAt(idx);
194 
195           if (fieldEndIdx == -1) {
196             // if the character is a ( this might be a function like UPPER(), try to parse that out
197             if (!inFunction && c == '(') {
198               inFunction = true;
199             } else if (inFunction && COMPARE_CHARS.indexOf(c) != -1) {
200               inFunction = false;
201             }
202 
203             // till we find the first char of the end of the field name, ignore compare chars
204             if (!inFunction && COMPARE_CHARS.indexOf(c) == -1) {
205               // found a char, must be the end of the field name
206               fieldEndIdx = idx;
207             }
208           } else {
209             // if we find a separator, we've reached the beginning of the field name
210             if (SEPARATOR_CHARS.indexOf(c) >= 0) {
211               fieldStartIdx = idx + 1;
212               found = true;
213             }
214           }
215           idx--;
216         }
217         if (found) {
218           String col = chunk.substring(fieldStartIdx, fieldEndIdx + 1);
219           colList.add(col);
220         }
221 
222         // move the starting index to where the ? was
223         startIdx = qIdx + 1;
224       }
225     }
226 
227     return colList;
228   }
229 }