oql-parser.y 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. /*
  2. This is a LALR(1) grammar
  3. (seek for Lemon grammar to get some documentation from the Net)
  4. That doc was helpful: http://www.hwaci.com/sw/lemon/lemon.html
  5. To handle operators precedence we could have used the %left directive
  6. (we took another option, because that one was discovered right after...
  7. which option is the best for us?)
  8. Example:
  9. %left LOG_AND.
  10. %left LOG_OR.
  11. %nonassoc EQ NE GT GE LT LE.
  12. %left PLUS MINUS.
  13. %left TIMES DIVIDE MOD.
  14. %right EXP NOT.
  15. TODO : solve the 2 remaining shift-reduce conflicts (JOIN)
  16. */
  17. %name OQLParser_
  18. %declare_class {class OQLParserRaw}
  19. %syntax_error {
  20. throw new OQLParserException($this->m_sSourceQuery, $this->m_iLine, $this->m_iCol, $this->tokenName($yymajor), $TOKEN);
  21. }
  22. result ::= query(X). { $this->my_result = X; }
  23. result ::= condition(X). { $this->my_result = X; }
  24. query(A) ::= SELECT class_name(X) join_statement(J) where_statement(W). {
  25. A = new OqlObjectQuery(X, X, W, J);
  26. }
  27. query(A) ::= SELECT class_name(X) AS_ALIAS class_name(Y) join_statement(J) where_statement(W). {
  28. A = new OqlObjectQuery(X, Y, W, J);
  29. }
  30. /*
  31. query(A) ::= SELECT field_id(E) FROM class_name(X) join_statement(J) where_statement(W). {
  32. A = new OqlValueSetQuery(E, X, X, W, J);
  33. }
  34. query(A) ::= SELECT field_id(E) FROM class_name(X) AS_ALIAS class_name(Y) join_statement(J) where_statement(W). {
  35. A = new OqlValueSetQuery(E, X, Y, W, J);
  36. }
  37. */
  38. where_statement(A) ::= WHERE condition(C). { A = C;}
  39. where_statement(A) ::= . { A = null;}
  40. join_statement(A) ::= join_item(J) join_statement(S). {
  41. // insert the join statement on top of the existing list
  42. array_unshift(S, J);
  43. // and return the updated array
  44. A = S;
  45. }
  46. join_statement(A) ::= join_item(J). {
  47. A = Array(J);
  48. }
  49. join_statement(A) ::= . { A = null;}
  50. join_item(A) ::= JOIN class_name(X) AS_ALIAS class_name(Y) ON join_condition(C).
  51. {
  52. // create an array with one single item
  53. A = new OqlJoinSpec(X, Y, C);
  54. }
  55. join_item(A) ::= JOIN class_name(X) ON join_condition(C).
  56. {
  57. // create an array with one single item
  58. A = new OqlJoinSpec(X, X, C);
  59. }
  60. join_condition(A) ::= field_id(X) EQ field_id(Y). { A = new BinaryOqlExpression(X, '=', Y); }
  61. condition(A) ::= expression_prio4(X). { A = X; }
  62. expression_basic(A) ::= scalar(X). { A = X; }
  63. expression_basic(A) ::= field_id(X). { A = X; }
  64. expression_basic(A) ::= var_name(X). { A = X; }
  65. expression_basic(A) ::= func_name(X) PAR_OPEN arg_list(Y) PAR_CLOSE. { A = new FunctionOqlExpression(X, Y); }
  66. expression_basic(A) ::= PAR_OPEN expression_prio4(X) PAR_CLOSE. { A = X; }
  67. expression_basic(A) ::= expression_basic(X) list_operator(Y) list(Z). { A = new BinaryOqlExpression(X, Y, Z); }
  68. expression_prio1(A) ::= expression_basic(X). { A = X; }
  69. expression_prio1(A) ::= expression_prio1(X) operator1(Y) expression_basic(Z). { A = new BinaryOqlExpression(X, Y, Z); }
  70. expression_prio2(A) ::= expression_prio1(X). { A = X; }
  71. expression_prio2(A) ::= expression_prio2(X) operator2(Y) expression_prio1(Z). { A = new BinaryOqlExpression(X, Y, Z); }
  72. expression_prio3(A) ::= expression_prio2(X). { A = X; }
  73. expression_prio3(A) ::= expression_prio3(X) operator3(Y) expression_prio2(Z). { A = new BinaryOqlExpression(X, Y, Z); }
  74. expression_prio4(A) ::= expression_prio3(X). { A = X; }
  75. expression_prio4(A) ::= expression_prio4(X) operator4(Y) expression_prio3(Z). { A = new BinaryOqlExpression(X, Y, Z); }
  76. list(A) ::= PAR_OPEN scalar_list(X) PAR_CLOSE. {
  77. A = new ListOqlExpression(X);
  78. }
  79. scalar_list(A) ::= scalar(X). {
  80. A = array(X);
  81. }
  82. scalar_list(A) ::= scalar_list(L) COMA scalar(X). {
  83. array_push(L, X);
  84. A = L;
  85. }
  86. arg_list(A) ::= . {
  87. A = array();
  88. }
  89. arg_list(A) ::= argument(X). {
  90. A = array(X);
  91. }
  92. arg_list(A) ::= arg_list(L) COMA argument(X). {
  93. array_push(L, X);
  94. A = L;
  95. }
  96. argument(A) ::= expression_prio4(X). { A = X; }
  97. argument(A) ::= INTERVAL expression_prio4(X) interval_unit(Y). { A = new IntervalOqlExpression(X, Y); }
  98. interval_unit(A) ::= F_SECOND(X). { A = X; }
  99. interval_unit(A) ::= F_MINUTE(X). { A = X; }
  100. interval_unit(A) ::= F_HOUR(X). { A = X; }
  101. interval_unit(A) ::= F_DAY(X). { A = X; }
  102. interval_unit(A) ::= F_MONTH(X). { A = X; }
  103. interval_unit(A) ::= F_YEAR(X). { A = X; }
  104. scalar(A) ::= num_scalar(X). { A = X; }
  105. scalar(A) ::= str_scalar(X). { A = X; }
  106. num_scalar(A) ::= num_value(X). { A = new ScalarOqlExpression(X); }
  107. str_scalar(A) ::= str_value(X). { A = new ScalarOqlExpression(X); }
  108. field_id(A) ::= name(X). { A = new FieldOqlExpression(X); }
  109. field_id(A) ::= class_name(X) DOT name(Y). { A = new FieldOqlExpression(Y, X); }
  110. class_name(A) ::= name(X). { A=X; }
  111. var_name(A) ::= VARNAME(X). { A = new VariableOqlExpression(substr(X, 1)); }
  112. name(A) ::= NAME(X). {
  113. if (X[0] == '`')
  114. {
  115. $name = substr(X, 1, strlen(X) - 2);
  116. }
  117. else
  118. {
  119. $name = X;
  120. }
  121. A = new OqlName($name, $this->m_iColPrev);
  122. }
  123. num_value(A) ::= NUMVAL(X). {A=X;}
  124. str_value(A) ::= STRVAL(X). {A=stripslashes(substr(X, 1, strlen(X) - 2));}
  125. operator1(A) ::= num_operator1(X). {A=X;}
  126. operator2(A) ::= num_operator2(X). {A=X;}
  127. operator2(A) ::= str_operator(X). {A=X;}
  128. operator2(A) ::= EQ(X). {A=X;}
  129. operator2(A) ::= NOT_EQ(X). {A=X;}
  130. operator3(A) ::= LOG_AND(X). {A=X;}
  131. operator4(A) ::= LOG_OR(X). {A=X;}
  132. num_operator1(A) ::= MATH_DIV(X). {A=X;}
  133. num_operator1(A) ::= MATH_MULT(X). {A=X;}
  134. num_operator2(A) ::= MATH_PLUS(X). {A=X;}
  135. num_operator2(A) ::= MATH_MINUS(X). {A=X;}
  136. num_operator2(A) ::= GT(X). {A=X;}
  137. num_operator2(A) ::= LT(X). {A=X;}
  138. num_operator2(A) ::= GE(X). {A=X;}
  139. num_operator2(A) ::= LE(X). {A=X;}
  140. str_operator(A) ::= LIKE(X). {A=X;}
  141. str_operator(A) ::= NOT_LIKE(X). {A=X;}
  142. list_operator(A) ::= IN(X). {A=X;}
  143. list_operator(A) ::= NOT_IN(X). {A=X;}
  144. func_name(A) ::= F_IF(X). { A=X; }
  145. func_name(A) ::= F_ELT(X). { A=X; }
  146. func_name(A) ::= F_COALESCE(X). { A=X; }
  147. func_name(A) ::= F_CONCAT(X). { A=X; }
  148. func_name(A) ::= F_SUBSTR(X). { A=X; }
  149. func_name(A) ::= F_TRIM(X). { A=X; }
  150. func_name(A) ::= F_DATE(X). { A=X; }
  151. func_name(A) ::= F_DATE_FORMAT(X). { A=X; }
  152. func_name(A) ::= F_CURRENT_DATE(X). { A=X; }
  153. func_name(A) ::= F_NOW(X). { A=X; }
  154. func_name(A) ::= F_TIME(X). { A=X; }
  155. func_name(A) ::= F_TO_DAYS(X). { A=X; }
  156. func_name(A) ::= F_FROM_DAYS(X). { A=X; }
  157. func_name(A) ::= F_YEAR(X). { A=X; }
  158. func_name(A) ::= F_MONTH(X). { A=X; }
  159. func_name(A) ::= F_DAY(X). { A=X; }
  160. func_name(A) ::= F_DATE_ADD(X). { A=X; }
  161. func_name(A) ::= F_DATE_SUB(X). { A=X; }
  162. func_name(A) ::= F_ROUND(X). { A=X; }
  163. func_name(A) ::= F_FLOOR(X). { A=X; }
  164. func_name(A) ::= F_INET_ATON(X). { A=X; }
  165. func_name(A) ::= F_INET_NTOA(X). { A=X; }
  166. %code {
  167. class OQLParserException extends OQLException
  168. {
  169. public function __construct($sInput, $iLine, $iCol, $sTokenName, $sTokenValue)
  170. {
  171. $sIssue = "Unexpected token $sTokenName";
  172. parent::__construct($sIssue, $sInput, $iLine, $iCol, $sTokenValue);
  173. }
  174. }
  175. class OQLParser extends OQLParserRaw
  176. {
  177. // dirty, but working for us (no other mean to get the final result :-(
  178. protected $my_result;
  179. public function GetResult()
  180. {
  181. return $this->my_result;
  182. }
  183. // More info on the source query and the current position while parsing it
  184. // Data used when an exception is raised
  185. protected $m_iLine; // still not used
  186. protected $m_iCol;
  187. protected $m_iColPrev; // this is the interesting one, because the parser will reduce on the next token
  188. protected $m_sSourceQuery;
  189. public function __construct($sQuery)
  190. {
  191. $this->m_iLine = 0;
  192. $this->m_iCol = 0;
  193. $this->m_iColPrev = 0;
  194. $this->m_sSourceQuery = $sQuery;
  195. // no constructor - parent::__construct();
  196. }
  197. public function doParse($token, $value, $iCurrPosition = 0)
  198. {
  199. $this->m_iColPrev = $this->m_iCol;
  200. $this->m_iCol = $iCurrPosition;
  201. return parent::DoParse($token, $value);
  202. }
  203. public function doFinish()
  204. {
  205. $this->doParse(0, 0);
  206. return $this->my_result;
  207. }
  208. public function __destruct()
  209. {
  210. // Bug in the original destructor, causing an infinite loop !
  211. // This is a real issue when a fatal error occurs on the first token (the error could not be seen)
  212. if (is_null($this->yyidx))
  213. {
  214. $this->yyidx = -1;
  215. }
  216. parent::__destruct();
  217. }
  218. }
  219. }