001    // Copyright (c) 2001 Hursh Jain (http://www.mollypages.org) 
002    // The Molly framework is freely distributable under the terms of an
003    // MIT-style license. For details, see the molly pages web site at:
004    // http://www.mollypages.org/. Use, modify, have fun !
005    
006    package fc.jdbc;
007    
008    import java.io.*;
009    import java.util.*;
010    import java.util.regex.*;
011    import java.lang.reflect.*;
012    import fc.io.*;
013    import fc.util.*;
014    
015    /** 
016    Loads sql queries from a file. Storing queries in a file (when
017    the number of queries is greater than 10 or so, is a lot more 
018    manageable than embeddding these as strings in the program).
019    
020    <p>
021    In the query file:
022    <ul style="list-style: square inside url('')">
023    
024    <li>comments start with hash or //</li>
025    
026    <li>// comments can also end a line and anything after a // on a line is ignored.</li>
027    
028    <li>empty lines (just whitespaces) are ignored</li>
029    
030    <li>queries are of format:
031    <blockquote><pre>
032    queryname = querycontent
033      ...querycontent continued...
034      ;
035    </pre></blockquote>
036    </li>
037    <li>a ';' character by itself (on a separate line) ends a name = content
038    section. The query is stored and can be retrieved via it's name.
039    </li>
040    
041    <li><p>
042    a special section within the content is enclosed within <b>$...$</b>
043    and is processed when this file is read. This section (if specified) refers
044    to a corresponding molly DBO class. For example:
045    </p>
046    <blockquote><pre>
047    userDetailQuery = 
048      select 
049        <b>$</b>package.foo<b>$</b>, <b>$</b>x.y.bar<b>$</b> from
050      ...rest of query...
051    ;
052    </pre></blockquote>
053    In this example, <tt>package.foo</tt> will be replaced by the call to
054    <tt>package.<b>fooMgr</b>.columns()</tt> and <tt>$x.y.bar$</tt> will be replaced
055    by a call to <tt>x.y.<b>barMgr</b>.columns()</tt>. Note, the names must be
056    fully qualified (that is, including the package name) since otherwise we won't
057    be able to locate the corresponding Mgr class at runtime.
058    <p>
059    The <b>$...$</b> can end with an optional <i>prefix</i>
060    after the ending $, for example, $...$<b><font color=blue>xxx</font></b>
061    </p>
062    <p>
063    If present, the <tt>package.<b>fooMgr</b>.columns(<font
064    color=blue><i>prefix</i></font>)</tt> is invoked to get the column
065    names. A prefix other than the default <i>tablename</i>.colname
066    formulation, (such as <i>xxx</i>_colname) is necessary, if the queryname
067    uses a <i>tablename <b>AS</b> <font color=blue>xxx</font></i> abbreviation anywhere 
068    in the query (if an abbreviation is used via the SQL AS clause, then that
069    abbreviation must be used everywhere, not the tablename).
070    </p>
071    </li>
072    
073    <li>to access a constant static final class attribute, use <b>$$...$$</b>. This is useful when constant values (declared in code) are to be used in a query.
074    <blockquote><pre>
075    userDetailQuery = 
076      select 
077        <b>$$</b>x.y.SOME_VAL<b>$$</b> 
078      ...rest of query...
079    ;
080    </pre></blockquote>
081    This will replace the value of the java field/attribute <tt>x.y.XYZ</tt> (in declaring class <tt>y</tt> located in package <tt>x</tt>).
082    This method does a straight value interpolation and does not add SQL stringify quotes if they are 
083    needed. For String or character types, quotes should be manually placed around the replaced value, 
084    such as:
085    <blockquote><pre>
086    userDetailQuery = 
087      select 
088        <font color=red>"</font><b>$$</b>x.y.SOME_STRING_VALUE<b>$$</b><font color=red>"</font>
089    ;
090    </pre></blockquote>
091    <p>
092    Note: the fields accessed this way must be delared as <b>final</b> and <b>static</b>. If they are not, then a error will be thrown when processing the query.
093    </li>
094    
095    <hr>
096    
097    Here is a real world <i>example</i> of this file in action:
098    <p>
099    <ol>
100    
101    <li>
102    The following file is called <b>my.queries</b> (the name is arbitrary) and
103    stored under <i>WEB-INF/foo/bar</i>
104    <pre>
105    # Get all friends of X
106    #   - get list of friends for X (friend uids from friends table)
107    # - get info about those friend uids from the users table
108    userFriends = 
109      SELECT 
110      u.name, u.status, u.gender, u.likes, u.pictime, u.hbtime,
111      f.friend_uid, f.is_liked, f.is_unliked,
112      is_friend(u.uid, ?) as twoway
113      FROM
114      users as u, friends as f
115      WHERE
116      f.uid = ? 
117      and u.uid = f.friend_uid
118    ;
119    
120    # Get unread messages sent to user X
121    # Unread messages are grouped by the the sender id and message count per sender
122    #
123    unreadMessages = 
124      SELECT
125      from_uid, count(is_retrieved) as count
126      FROM 
127      messages 
128      WHERE 
129      to_uid = ? and is_retrieved = false 
130      GROUP BY 
131      from_uid
132    ;
133    
134    # Gets detailed information about place X
135    placeDetail = 
136      SELECT
137          $my.dbobj.location$
138      FROM 
139        location 
140      WHERE
141      location_id = ?
142    ;
143    </pre>
144    </li>
145    
146    <li>
147    This is then initialized/accessed from a servlet via the following code snippet:
148    <pre>
149    try 
150      {
151      //queryMgr is an instance variable in the servlet (can be later accessed
152      //from other methods). webroot is a file pointing to the root directory
153      //of this context.
154    
155      queryMgr = new QueryReader(
156        new File(webroot, "<b>WEB-INF/foo/bar/my.queries</b>"), log); 
157      log.info("Available queries: ", queryMgr.getQueries().keySet());
158      }
159    catch (IOException e) {
160      throw new ServletException(IOUtil.throwableToString(e));
161      }
162    </pre>
163    </li>
164    
165    <li>
166    Note: the placeDetail query in the example file contains the $..$
167    replaceable text <i>$my.dbobj.location$</i>. This means, that the
168    <i>my.dbobj.*</i> classes should be included in the invoking servlet's
169    classpath, such that the corresponding <i>my.dbobj.locationMgr</i>
170    class is found.
171    <p>
172    This would be ensured by putting the following at the top of the
173    servlet code above:
174    <p>
175    import <i>my.dbobj.*</i>;
176    </p>
177    </li>
178    </ol>
179    **/
180    public class QueryReader
181    {
182    private static final boolean dbg = false;
183    public static final int   TEST_FIELD_1 = 1;
184    public static final String  TEST_FIELD_2 = "hello";
185    
186    Log   log;
187    Map   queries = new LinkedHashMap();
188    
189    /** 
190    Creates a query reader that reads queries from the specified file,
191    using the UTF-8 encoding and a default logger.
192    
193    @throws IOException   on error reading from the file
194    */
195    public QueryReader(File f) throws IOException
196      {
197      this(f, Log.getDefault());
198      }
199      
200    /** 
201    Creates a query reader that reads queries from the specified file,
202    using the UTF-8 encoding and the specified logger.
203    
204    @throws IOException   on error reading from the file
205    */
206    public QueryReader(File f, Log logger) throws IOException
207      {
208      log = logger; 
209      BufferedReader in
210          = new BufferedReader(
211              new InputStreamReader(new FileInputStream(f), "UTF-8"));
212      
213      String line = null;
214      StringBuilder buf = new StringBuilder (1024);
215      
216      while ( (line = in.readLine()) != null)
217        {
218        String trimline = line.trim();  //this gets rid of spaces, empty newlines, etc
219    
220        if (trimline.length() == 0 
221          || trimline.startsWith("#") 
222          || trimline.startsWith("//")) 
223          {
224          if (dbg) System.out.println("Skipping: " + line);
225          continue;
226          }
227    
228        //this skips a series of lines containing ;
229        if (buf.length() == 0 && trimline.equals(";"))
230          {
231          if (dbg) System.out.println("Skipping: " + line);
232          continue;
233          }
234    
235        //ignore trailing comments starting with "//"
236        String[] split = line.split("\\s+//", 2);
237        line = split[0];
238      
239        if (dbg && split.length > 1) {
240          System.out.println("Splitting line with trailing //");
241          System.out.println("split=" + Arrays.asList(split));
242          }
243        
244        if (trimline.equals(";")) 
245          {
246          processBuffer(buf);
247          buf = new StringBuilder();
248          }
249        else if (line.trim().endsWith(";")) 
250          {
251          //Not in spec but added just for safety in case ';' appears 
252          //on the same line 
253          buf.append(line.substring(0, line.lastIndexOf(';')));
254          processBuffer(buf);  
255          buf = new StringBuilder();
256          }
257        else{
258          //append original line, not trimline, this allows us
259          //to keep original leading spaces in the query 
260          buf.append(line);
261          //this is important, either append a newline or a space
262          //otherwise separate lines in the query file run into each
263          //other IF there is no trailing/leading spaces on each
264          //line. newline good since it preserves original formatting
265          buf.append("\n");
266          }
267        }
268        
269      String remaining = buf.toString().trim();
270      if (remaining.length() > 0) {
271        log.error("No ending delimiter (';') seen, the following section was NOT processed: \n", remaining);
272        }
273        
274      in.close();
275      log.info("Processed: ", queries.size(), " queries from file: ", f.getAbsolutePath());
276      }
277    
278    //process 1 query at a time (invoked when ending ';' for each query is seen)
279    void processBuffer(StringBuilder buf)
280      {
281      String[] keyval = buf.toString().split("=", 2);
282    
283      if (dbg) {
284        System.out.println(Arrays.asList(keyval));
285        }
286    
287      if (keyval.length != 2) {
288        log.error("Query sections must be of form 'name = value'. Badly formed section, NOT processed: \n", buf); 
289        return;
290        }
291        
292      String name = keyval[0].trim();
293      String value = keyval[1].trim();
294    
295      if (queries.get(name) != null) {
296        log.error("This query name ", name, " already exists prior to this section. Duplicate name NOT processed: \n", buf);  
297        return;
298        }
299    
300      StringBuffer sb = new StringBuffer();
301      
302      //dbo columns pattern
303      //the first [^$] skips the $$ part (for field value expressions). 
304      Pattern p = Pattern.compile("[^$]\\$\\s*([a-zA-Z_0-9.]+)\\s*\\$([a-zA-Z_0-9.]*)");  //=>  \$(...)\$optionalprefix
305      Matcher m = p.matcher(value);
306      while (m.find()) 
307        {
308        if (dbg) System.out.println("Matched columns $regex: " + m.group());
309        String dbo = m.group(1);
310        String mgrName = dbo + "Mgr";
311    
312        String prefix = m.group(2);
313          
314        if (dbg) System.out.println("Manager name = " +  mgrName);
315        if (dbg) System.out.println("Column prefix name = " +  prefix);
316        
317        try {
318          Class mgr = Class.forName(mgrName, true,
319            Thread.currentThread().getContextClassLoader());
320        
321          Method method = null;
322          String columns = null;
323        
324          if (prefix != null && ! prefix.equals("")) {
325            method = mgr.getMethod("columns", new Class[] {String.class});
326            columns = (String) method.invoke(null, prefix);
327            }
328          else{
329            method = mgr.getMethod("columns", null);
330            columns = (String) method.invoke(null, null);
331            }
332          
333          m.appendReplacement(sb, columns);
334          if (dbg) System.out.println("replacing: [" + dbo + "] with [" + columns + "]");
335          }
336        catch (ClassNotFoundException e) {
337          log.error("Manager [", mgrName, "] for [$", dbo + "$] not found, this query will NOT be added. Query:\n-----------------------\n", buf, "\n-----------------------\n");
338          return;     
339          }
340        catch (Exception e2) {
341          log.error("Internal error while processing: ", buf);
342          log.error("This query was NOT added");
343          log.error(IOUtil.throwableToString(e2));
344          return;           
345          }
346        } //~while
347        
348      m.appendTail(sb);
349    
350      //constant field values pattern $$...$$
351      p = Pattern.compile("\\$\\$\\s*([a-zA-Z_0-9.]+)\\.([a-zA-Z_0-9]+)\\s*\\$\\$");  
352      m = p.matcher(sb.toString());
353      sb = new StringBuffer();
354      
355      while (m.find()) 
356        {
357        if (dbg) System.out.println("Matched constant field $regex: " + m.group());
358        String classname = m.group(1);
359        String fieldname = m.group(2);
360          
361        if (dbg) System.out.println("class name = " +  classname);
362        if (dbg) System.out.println("field name = " +  fieldname);
363        
364        try {
365          Class c = Class.forName(classname, true, Thread.currentThread().getContextClassLoader());
366        
367          Field field = c.getDeclaredField(fieldname);
368          int modifiers = field.getModifiers();
369          if (! Modifier.isStatic(modifiers) || ! Modifier.isFinal(modifiers)) {
370            throw new Exception("Darn! Field: [" + field + "] was not declared static or final. It must be both for this reference to work!");
371            }
372          if (! Modifier.isPublic(modifiers)) {
373            field.setAccessible(true);
374            }
375          
376          Object fieldval = field.get(null);  
377          
378          m.appendReplacement(sb, String.valueOf(fieldval));
379          if (dbg) System.out.println("replacing: [" + field + "] with [" + fieldval + "]");
380          }
381        catch (ClassNotFoundException e) {
382          log.error("Class [", classname, "] not found, this query will NOT be added. Query:\n-----------------------\n", buf, "\n-----------------------\n");
383          return;     
384          }
385        catch (NoSuchFieldException e) {
386          log.error("Field [", fieldname, "] not found, this query will NOT be added. Query:\n-----------------------\n", buf, "\n-----------------------\n");
387          return;     
388          }
389        catch (Exception e2) {
390          log.error("Internal error while processing: ", buf);
391          log.error("This query was NOT added");
392          log.error(IOUtil.throwableToString(e2));
393          return;           
394          }
395        } //~while
396    
397      m.appendTail(sb);
398    
399      queries.put(name, sb.toString()); 
400      }
401      
402    /** 
403    returns the query with the specified name or <tt>null</tt> if the query
404    does not exist. 
405    */
406    public String getQuery(String name) {
407      return (String) queries.get(name);
408      }
409    
410    /** 
411    returns the entire query map containing all successfully read queries.
412    */
413    public Map getQueries() {
414      return queries;
415      }
416    
417      
418    public static void main (String args[]) throws IOException
419      {
420      Args myargs = new Args(args);
421      String filestr = myargs.getRequired("file");
422      QueryReader qr = new QueryReader(new File(filestr));
423      System.out.println("----------------- processed queries ------------------");
424      System.out.println(qr.queries);
425      }
426    }