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 006package fc.jdbc; 007 008import java.io.*; 009import java.util.*; 010import java.util.regex.*; 011import java.lang.reflect.*; 012import fc.io.*; 013import fc.util.*; 014 015/** 016Loads sql queries from a file. Storing queries in a file (when 017the number of queries is greater than 10 or so, is a lot more 018manageable than embeddding these as strings in the program). 019 020<p> 021In 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> 032queryname = querycontent 033 ...querycontent continued... 034 ; 035</pre></blockquote> 036</li> 037<li>a ';' character by itself (on a separate line) ends a name = content 038section. The query is stored and can be retrieved via it's name. 039</li> 040 041<li><p> 042a special section within the content is enclosed within <b>$...$</b> 043and is processed when this file is read. This section (if specified) refers 044to a corresponding molly DBO class. For example: 045</p> 046<blockquote><pre> 047userDetailQuery = 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> 053In 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 055by a call to <tt>x.y.<b>barMgr</b>.columns()</tt>. Note, the names must be 056fully qualified (that is, including the package name) since otherwise we won't 057be able to locate the corresponding Mgr class at runtime. 058<p> 059The <b>$...$</b> can end with an optional <i>prefix</i> 060after the ending $, for example, $...$<b><font color=blue>xxx</font></b> 061</p> 062<p> 063If present, the <tt>package.<b>fooMgr</b>.columns(<font 064color=blue><i>prefix</i></font>)</tt> is invoked to get the column 065names. A prefix other than the default <i>tablename</i>.colname 066formulation, (such as <i>xxx</i>_colname) is necessary, if the queryname 067uses a <i>tablename <b>AS</b> <font color=blue>xxx</font></i> abbreviation anywhere 068in the query (if an abbreviation is used via the SQL AS clause, then that 069abbreviation 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> 075userDetailQuery = 076 select 077 <b>$$</b>x.y.SOME_VAL<b>$$</b> 078 ...rest of query... 079; 080</pre></blockquote> 081This 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>). 082This method does a straight value interpolation and does not add SQL stringify quotes if they are 083needed. For String or character types, quotes should be manually placed around the replaced value, 084such as: 085<blockquote><pre> 086userDetailQuery = 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> 092Note: 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 097Here is a real world <i>example</i> of this file in action: 098<p> 099<ol> 100 101<li> 102The following file is called <b>my.queries</b> (the name is arbitrary) and 103stored 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 108userFriends = 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# 123unreadMessages = 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 135placeDetail = 136 SELECT 137 $my.dbobj.location$ 138 FROM 139 location 140 WHERE 141 location_id = ? 142; 143</pre> 144</li> 145 146<li> 147This is then initialized/accessed from a servlet via the following code snippet: 148<pre> 149try 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 } 159catch (IOException e) { 160 throw new ServletException(IOUtil.throwableToString(e)); 161 } 162</pre> 163</li> 164 165<li> 166Note: the placeDetail query in the example file contains the $..$ 167replaceable text <i>$my.dbobj.location$</i>. This means, that the 168<i>my.dbobj.*</i> classes should be included in the invoking servlet's 169classpath, such that the corresponding <i>my.dbobj.locationMgr</i> 170class is found. 171<p> 172This would be ensured by putting the following at the top of the 173servlet code above: 174<p> 175import <i>my.dbobj.*</i>; 176</p> 177</li> 178</ol> 179**/ 180public final class QueryReader 181{ 182private static final boolean dbg = false; 183public static final int TEST_FIELD_1 = 1; 184public static final String TEST_FIELD_2 = "hello"; 185 186Log log; 187Map queries = new LinkedHashMap(); 188 189/** 190Creates a query reader that reads queries from the specified file, 191using the UTF-8 encoding and a default logger. 192 193@throws IOException on error reading from the file 194*/ 195public QueryReader(File f) throws IOException 196 { 197 this(f, Log.getDefault()); 198 } 199 200/** 201Creates a query reader that reads queries from the specified file, 202using the UTF-8 encoding and the specified logger. 203 204@throws IOException on error reading from the file 205*/ 206public 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) 279void 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/** 403returns the query with the specified name or <tt>null</tt> if the query 404does not exist. 405*/ 406public String getQuery(String name) { 407 return (String) queries.get(name); 408 } 409 410/** 411returns the entire query map containing all successfully read queries. 412*/ 413public Map getQueries() { 414 return queries; 415 } 416 417 418public 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}