/OData Producer for PHP/library/ODataProducer/UriProcessor/QueryProcessor/ExpressionParser/Expressions/PropertyAccessExpression.php

# · PHP · 284 lines · 154 code · 21 blank · 109 comment · 26 complexity · e7822d89cc1a21fbaf8dbb2271287d43 MD5 · raw file

  1. <?php
  2. /**
  3. * Expression class specialized for a property access.
  4. *
  5. * PHP version 5.3
  6. *
  7. * @category ODataProducer
  8. * @package ODataProducer_UriProcessor_QueryProcessor_ExpressionParser_Expressions
  9. * @author Anu T Chandy <odataphpproducer_alias@microsoft.com>
  10. * @copyright 2011 Microsoft Corp. (http://www.microsoft.com)
  11. * @license New BSD license, (http://www.opensource.org/licenses/bsd-license.php)
  12. * @version SVN: 1.0
  13. * @link http://odataphpproducer.codeplex.com
  14. *
  15. */
  16. namespace ODataProducer\UriProcessor\QueryProcessor\ExpressionParser\Expressions;
  17. use ODataProducer\UriProcessor\QueryProcessor\FunctionDescription\FunctionDescription;
  18. use ODataProducer\UriProcessor\QueryProcessor\ExpressionParser\Expressions\LogicalExpression;
  19. use ODataProducer\UriProcessor\QueryProcessor\ExpressionParser\Expressions\AbstractExpression;
  20. use ODataProducer\Providers\Metadata\ResourceTypeKind;
  21. use ODataProducer\Providers\Metadata\ResourceProperty;
  22. use ODataProducer\Providers\Metadata\Type\Navigation;
  23. use ODataProducer\Providers\Metadata\Type\Boolean;
  24. /**
  25. * Expression class for property access.
  26. *
  27. * @category ODataProducer
  28. * @package ODataProducer_UriProcessor_QueryProcessor_ExpressionParser_Expressions
  29. * @author Anu T Chandy <odataphpproducer_alias@microsoft.com>
  30. * @copyright 2011 Microsoft Corp. (http://www.microsoft.com)
  31. * @license New BSD license, (http://www.opensource.org/licenses/bsd-license.php)
  32. * @version Release: 1.0
  33. * @link http://odataphpproducer.codeplex.com
  34. */
  35. class PropertyAccessExpression extends AbstractExpression
  36. {
  37. /**
  38. * @var PropertyAccessExpression
  39. */
  40. protected $parent;
  41. /**
  42. * @var PropertyAccessExpression
  43. */
  44. protected $child;
  45. /**
  46. * Resource property instance describes the property represented by
  47. * this expression
  48. *
  49. * @var ResourceProperty
  50. */
  51. protected $resourceProperty;
  52. /**
  53. * Creates new instance of PropertyAccessExpression
  54. *
  55. * @param PropertyAccessExpression $parent The parent expression
  56. * @param ResourceProperty $resourceProperty The ResourceProperty
  57. */
  58. public function __construct($parent, $resourceProperty)
  59. {
  60. $this->parent = $parent;
  61. $this->child = null;
  62. $this->nodeType = ExpressionType::PROPERTYACCESS;
  63. $this->resourceProperty = $resourceProperty;
  64. //If the property is primitive type, then _type will be primitve types
  65. //implementing IType
  66. if ($resourceProperty->getResourceType()->getResourceTypeKind() == ResourceTypeKind::PRIMITIVE) {
  67. $this->type = $resourceProperty->getResourceType()->getInstanceType();
  68. } else {
  69. //This is a navigation i.e. Complex, ResourceReference or Collection
  70. $this->type = new Navigation($resourceProperty->getResourceType());
  71. }
  72. if (!is_null($parent)) {
  73. $parent->setChild($this);
  74. }
  75. }
  76. /**
  77. * To set the child if any
  78. *
  79. * @param PropertyAccessExpression $child The child expression
  80. *
  81. * @return void
  82. */
  83. public function setChild($child)
  84. {
  85. $this->child = $child;
  86. }
  87. /**
  88. * To get the parent. If this property is property of entity
  89. * then return null, If this property is property of complex type
  90. * then return PropertyAccessExpression for the parent complex type
  91. *
  92. * @return PropertyAccessExpression
  93. */
  94. public function getParent()
  95. {
  96. return $this->parent;
  97. }
  98. /**
  99. * To get the child. Returns null if no child property
  100. *
  101. * @return PropertyAccessExpression
  102. */
  103. public function getChild()
  104. {
  105. return $this->child;
  106. }
  107. /**
  108. * Get the resource type of the property hold by this expression
  109. *
  110. * @return ResourceType
  111. */
  112. public function getResourceType()
  113. {
  114. return $this->resourceProperty->getResourceType();
  115. }
  116. /**
  117. * Get the ResourceProperty describing the property hold by this expression
  118. *
  119. * @return ResourceProperty
  120. */
  121. public function getResourceProperty()
  122. {
  123. return $this->resourceProperty;
  124. }
  125. /**
  126. * Gets collection of navigation (resource set reference or resource set)
  127. * properties used in this property access.
  128. *
  129. * @return array()/array(ResourceProperty) Returns empty array if no
  130. * navigation property is used else
  131. * array of ResourceProperty.
  132. */
  133. public function getNavigationPropertiesInThePath()
  134. {
  135. $basePropertyExpression = $this;
  136. while (($basePropertyExpression != null)
  137. && ($basePropertyExpression->parent != null)
  138. ) {
  139. $basePropertyExpression = $basePropertyExpression->parent;
  140. }
  141. $navigationPropertiesInThePath = array();
  142. while ($basePropertyExpression) {
  143. $resourceTypeKind
  144. = $basePropertyExpression->getResourceType()->getResourceTypeKind();
  145. if ($resourceTypeKind == ResourceTypeKind::ENTITY) {
  146. $navigationPropertiesInThePath[]
  147. = $basePropertyExpression->resourceProperty;
  148. } else {
  149. break;
  150. }
  151. $basePropertyExpression = $basePropertyExpression->child;
  152. }
  153. return $navigationPropertiesInThePath;
  154. }
  155. /**
  156. * Function to create a nullable expression subtree for checking the
  157. * nullablilty of parent (and current poperty optionally) properties
  158. *
  159. * @param boolean $includeMe Boolean flag indicating whether to include null
  160. * check for this property along with parents
  161. *
  162. * @return AbstractExpression Instance of UnaryExpression, LogicalExpression
  163. * or Null
  164. *
  165. */
  166. public function createNullableExpressionTree($includeMe)
  167. {
  168. $basePropertyExpression = $this;
  169. while (($basePropertyExpression != null)
  170. && ($basePropertyExpression->parent != null)
  171. ) {
  172. $basePropertyExpression = $basePropertyExpression->parent;
  173. }
  174. //This property is direct child of ResourceSet, no need to check
  175. //nullability for direct ResourceSet properties
  176. // ($c->CustomerID, $c->Order, $c->Address) unless $includeMe is true
  177. if ($basePropertyExpression == $this) {
  178. if ($includeMe) {
  179. return new UnaryExpression(
  180. new FunctionCallExpression(
  181. FunctionDescription::isNullCheckFunction(
  182. $basePropertyExpression->getType()
  183. ),
  184. array($basePropertyExpression)
  185. ),
  186. ExpressionType::NOT_LOGICAL,
  187. new Boolean()
  188. );
  189. }
  190. return null;
  191. }
  192. //This property is a property of a complex type or resource reference
  193. //$c->Order->OrderID, $c->Address->LineNumber,
  194. // $c->complex1->complex2->primitveVar
  195. //($c->Order != null),($c->Address != null),
  196. // (($c->complex1 != null) && ($c->complex1->complex2 != null))
  197. $expression = new UnaryExpression(
  198. new FunctionCallExpression(
  199. FunctionDescription::isNullCheckFunction(
  200. $basePropertyExpression->getType()
  201. ),
  202. array($basePropertyExpression)
  203. ),
  204. ExpressionType::NOT_LOGICAL,
  205. new Boolean()
  206. );
  207. while (($basePropertyExpression->getChild() != null)
  208. && ($basePropertyExpression->getChild()->getChild() != null)) {
  209. $basePropertyExpression = $basePropertyExpression->getChild();
  210. $expression2 = new UnaryExpression(
  211. new FunctionCallExpression(
  212. FunctionDescription::isNullCheckFunction(
  213. $basePropertyExpression->getType()
  214. ),
  215. array($basePropertyExpression)
  216. ),
  217. ExpressionType::NOT_LOGICAL,
  218. new Boolean()
  219. );
  220. $expression = new LogicalExpression(
  221. $expression, $expression2,
  222. ExpressionType::AND_LOGICAL
  223. );
  224. }
  225. if ($includeMe) {
  226. $basePropertyExpression = $basePropertyExpression->getChild();
  227. $expression2 = new UnaryExpression(
  228. new FunctionCallExpression(
  229. FunctionDescription::isNullCheckFunction(
  230. $basePropertyExpression->getType()
  231. ),
  232. array($basePropertyExpression)
  233. ),
  234. ExpressionType::NOT_LOGICAL,
  235. new Boolean()
  236. );
  237. $expression = new LogicalExpression(
  238. $expression, $expression2,
  239. ExpressionType::AND_LOGICAL
  240. );
  241. }
  242. return $expression;
  243. }
  244. /**
  245. * (non-PHPdoc)
  246. *
  247. * @see library/ODataProducer/QueryProcessor/Expressions/ODataProducer\QueryProcessor\Expressions.AbstractExpression::free()
  248. *
  249. * @return void
  250. */
  251. public function free()
  252. {
  253. if (!is_null($this->parent)) {
  254. $this->parent->free();
  255. unset($this->parent);
  256. }
  257. if (!is_null($this->child)) {
  258. $this->child->free();
  259. unset($this->child);
  260. }
  261. }
  262. }
  263. ?>