/sql-processor/src/main/java/org/sqlproc/engine/plugin/DefaultBeanUtilsPlugin.java

http://github.com/hudec/sql-processor · Java · 758 lines · 614 code · 87 blank · 57 comment · 212 complexity · 8f8b43dfc897fc27a39abd1917d05244 MD5 · raw file

  1. package org.sqlproc.engine.plugin;
  2. import java.beans.BeanInfo;
  3. import java.beans.IndexedPropertyDescriptor;
  4. import java.beans.IntrospectionException;
  5. import java.beans.Introspector;
  6. import java.beans.PropertyDescriptor;
  7. import java.lang.reflect.Constructor;
  8. import java.lang.reflect.Field;
  9. import java.lang.reflect.InvocationTargetException;
  10. import java.lang.reflect.Method;
  11. import java.lang.reflect.Modifier;
  12. import java.lang.reflect.ParameterizedType;
  13. import java.util.Arrays;
  14. import java.util.concurrent.ConcurrentHashMap;
  15. import org.slf4j.Logger;
  16. import org.slf4j.LoggerFactory;
  17. import org.sqlproc.engine.SqlFeature;
  18. import org.sqlproc.engine.SqlRuntimeContext;
  19. import org.sqlproc.engine.SqlRuntimeException;
  20. /**
  21. * Standard bean utilities implementation.
  22. *
  23. * @author <a href="mailto:Vladimir.Hudec@gmail.com">Vladimir Hudec</a>
  24. */
  25. public class DefaultBeanUtilsPlugin implements BeanUtilsPlugin {
  26. protected ConcurrentHashMap<String, Constructor<?>> constructors = new ConcurrentHashMap<String, Constructor<?>>();
  27. protected ConcurrentHashMap<String, PropertyDescriptor[]> descriptors = new ConcurrentHashMap<String, PropertyDescriptor[]>();
  28. protected ConcurrentHashMap<String, Class<?>> types = new ConcurrentHashMap<String, Class<?>>();
  29. protected ConcurrentHashMap<String, Class<?>[]> parameterizedTypes = new ConcurrentHashMap<String, Class<?>[]>();
  30. protected ConcurrentHashMap<String, Method> getters = new ConcurrentHashMap<String, Method>();
  31. protected ConcurrentHashMap<String, GetterType> typeGetters = new ConcurrentHashMap<String, GetterType>();
  32. protected ConcurrentHashMap<String, Method> setters = new ConcurrentHashMap<String, Method>();
  33. protected ConcurrentHashMap<String, Method> methods = new ConcurrentHashMap<String, Method>();
  34. protected ConcurrentHashMap<String, Method> enumsIn = new ConcurrentHashMap<String, Method>();
  35. protected ConcurrentHashMap<String, Method> enumsOut = new ConcurrentHashMap<String, Method>();
  36. /**
  37. * The internal slf4j logger.
  38. */
  39. final Logger logger = LoggerFactory.getLogger(DefaultBeanUtilsPlugin.class);
  40. // instances
  41. protected Constructor<?> getInstanceConstructor(SqlRuntimeContext runtimeCtx, Class<?> clazz) {
  42. String keyName = clazz.getName();
  43. Constructor<?> ctor = constructors.get(keyName);
  44. if (ctor != null)
  45. return ctor;
  46. if ((clazz.getModifiers() & 0x0400) != 0) {
  47. logger.warn("getInstance: " + clazz + " is abstract");
  48. return null;
  49. }
  50. try {
  51. ctor = clazz.getConstructor();
  52. try {
  53. ctor.setAccessible(true);
  54. } catch (SecurityException se) {
  55. logger.warn("getInstance: " + clazz + " " + se.getMessage());
  56. }
  57. } catch (NoSuchMethodException e) {
  58. logger.warn("getInstance: " + clazz + " " + e.getMessage());
  59. } catch (SecurityException e) {
  60. logger.warn("getInstance: " + clazz + " " + e.getMessage());
  61. }
  62. if (ctor == null) {
  63. Constructor<?>[] ctors = clazz.getConstructors();
  64. for (int i = 0, size = ctors.length; i < size; i++) {
  65. Class<?>[] ctorParams = ctors[i].getParameterTypes();
  66. if (ctorParams.length == 0) {
  67. if (Modifier.isPublic(ctor.getModifiers())) {
  68. Class<?> _clazz = ctor.getDeclaringClass();
  69. if (Modifier.isPublic(_clazz.getModifiers()))
  70. ctor = ctors[i];
  71. }
  72. if (ctor != null) {
  73. try {
  74. ctor.setAccessible(true);
  75. } catch (SecurityException se) {
  76. logger.warn("getInstance: " + clazz + " " + se.getMessage());
  77. }
  78. }
  79. }
  80. }
  81. }
  82. if (ctor == null)
  83. return null;
  84. Constructor<?> ctorPrev = constructors.putIfAbsent(keyName, ctor);
  85. if (ctorPrev != null)
  86. return ctorPrev;
  87. return ctor;
  88. }
  89. /**
  90. * {@inheritDoc}
  91. */
  92. @Override
  93. public Object getInstance(SqlRuntimeContext runtimeCtx, Class<?> clazz) {
  94. Constructor<?> ctor = getInstanceConstructor(runtimeCtx, clazz);
  95. if (ctor == null) {
  96. logger.warn("getInstance: " + clazz + " can't get constructor");
  97. return null;
  98. }
  99. try {
  100. return ctor.newInstance();
  101. } catch (InstantiationException e) {
  102. logger.error("getInstance: " + clazz, e);
  103. return null;
  104. } catch (IllegalAccessException e) {
  105. logger.error("getInstance: " + clazz, e);
  106. return null;
  107. } catch (IllegalArgumentException e) {
  108. logger.error("getInstance: " + clazz, e);
  109. return null;
  110. } catch (InvocationTargetException e) {
  111. logger.error("getInstance: " + clazz, e);
  112. return null;
  113. }
  114. }
  115. protected PropertyDescriptor[] getDescriptors(Class<?> clazz) {
  116. String keyName = clazz.getName();
  117. PropertyDescriptor[] _descriptors = descriptors.get(keyName);
  118. if (_descriptors != null)
  119. return _descriptors;
  120. BeanInfo beanInfo = null;
  121. try {
  122. beanInfo = Introspector.getBeanInfo(clazz);
  123. } catch (IntrospectionException e) {
  124. logger.error("getDescriptors: " + clazz, e);
  125. return null;
  126. }
  127. _descriptors = beanInfo.getPropertyDescriptors();
  128. for (PropertyDescriptor pd : _descriptors) {
  129. if (pd instanceof IndexedPropertyDescriptor) {
  130. logger.warn(
  131. "getDescriptors: " + clazz + " unsupported IndexedPropertyDescriptor " + pd.getDisplayName());
  132. }
  133. }
  134. PropertyDescriptor[] _descriptorsPrev = descriptors.putIfAbsent(keyName, _descriptors);
  135. if (_descriptorsPrev != null)
  136. return _descriptorsPrev;
  137. return _descriptors;
  138. }
  139. protected PropertyDescriptor getAttributeDescriptor(Class<?> clazz, String attrName) {
  140. PropertyDescriptor descriptor = null;
  141. PropertyDescriptor[] descriptors = getDescriptors(clazz);
  142. if (descriptors != null) {
  143. for (PropertyDescriptor _descriptor : descriptors) {
  144. if (_descriptor.getName().equalsIgnoreCase(attrName)) {
  145. descriptor = _descriptor;
  146. break;
  147. }
  148. }
  149. }
  150. return descriptor;
  151. }
  152. // attributes
  153. @Override
  154. public Class<?> getAttributeType(SqlRuntimeContext runtimeCtx, Class<?> clazz, String attrName) {
  155. String keyName = clazz.getName() + "." + attrName;
  156. Class<?> attrType = types.get(keyName);
  157. if (attrType != null)
  158. return attrType;
  159. PropertyDescriptor descriptor = getAttributeDescriptor(clazz, attrName);
  160. if (descriptor == null) {
  161. logger.error("getAttributeType: there's no attribute " + attrName + " in " + clazz.getName());
  162. return null;
  163. }
  164. attrType = descriptor.getPropertyType();
  165. Class<?> attrTypePrev = types.putIfAbsent(keyName, attrType);
  166. if (attrTypePrev != null)
  167. return attrTypePrev;
  168. return attrType;
  169. }
  170. @Override
  171. public Class<?>[] getAttributeParameterizedTypes(SqlRuntimeContext runtimeCtx, Class<?> clazz, String attrName) {
  172. String keyName = clazz.getName() + "." + attrName;
  173. Class<?>[] attrTypes = parameterizedTypes.get(keyName);
  174. if (attrTypes != null)
  175. return attrTypes;
  176. PropertyDescriptor descriptor = getAttributeDescriptor(clazz, attrName);
  177. if (descriptor == null) {
  178. logger.error("getAttributeType: there's no attribute " + attrName + " in " + clazz.getName());
  179. return null;
  180. }
  181. try {
  182. Field f = clazz.getDeclaredField(attrName);
  183. if (f.getGenericType() != null && f.getGenericType() instanceof ParameterizedType) {
  184. int size = ((ParameterizedType) f.getGenericType()).getActualTypeArguments().length;
  185. attrTypes = new Class<?>[size];
  186. for (int i = 0; i < size; i++)
  187. attrTypes[i] = (Class<?>) ((ParameterizedType) f.getGenericType()).getActualTypeArguments()[i];
  188. }
  189. } catch (NoSuchFieldException | SecurityException e) {
  190. logger.error("getAttributeParameterizedType: " + clazz + " for " + attrName, e);
  191. return null;
  192. }
  193. if (attrTypes != null) {
  194. Class<?>[] attrTypesPrev = parameterizedTypes.putIfAbsent(keyName, attrTypes);
  195. if (attrTypesPrev != null)
  196. return attrTypesPrev;
  197. }
  198. return attrTypes;
  199. }
  200. // getters
  201. protected Method getGetter(SqlRuntimeContext runtimeCtx, Class<?> clazz, String attrName, boolean onlyCheck) {
  202. String keyName = clazz.getName() + "." + attrName;
  203. Method getter = getters.get(keyName);
  204. if (getter != null)
  205. return getter;
  206. PropertyDescriptor descriptor = getAttributeDescriptor(clazz, attrName);
  207. if (descriptor == null) {
  208. if (!onlyCheck)
  209. logger.error("getGetter: there's no attribute " + attrName + " in " + clazz.getName());
  210. return null;
  211. }
  212. getter = getMethod(clazz, descriptor.getReadMethod(), onlyCheck);
  213. if (getter == null)
  214. return null;
  215. Method getterPrev = getters.putIfAbsent(keyName, getter);
  216. if (getterPrev != null)
  217. return getterPrev;
  218. return getter;
  219. }
  220. /**
  221. * {@inheritDoc}
  222. */
  223. @Override
  224. public GetterType getGetterType(SqlRuntimeContext runtimeCtx, Class<?> clazz, String attrName) {
  225. String keyName = clazz.getName() + "." + attrName;
  226. GetterType getterType = typeGetters.get(keyName);
  227. if (getterType != null)
  228. return getterType;
  229. Method m = getGetter(runtimeCtx, clazz, attrName, false);
  230. if (m == null)
  231. return null;
  232. getterType = new GetterType(m);
  233. GetterType getterTypePrev = typeGetters.putIfAbsent(keyName, getterType);
  234. if (getterTypePrev != null)
  235. return getterTypePrev;
  236. return getterType;
  237. }
  238. /**
  239. * {@inheritDoc}
  240. */
  241. @Override
  242. public GetterType getGetterType(SqlRuntimeContext runtimeCtx, Object bean, String attrName) {
  243. return getGetterType(runtimeCtx, bean.getClass(), attrName);
  244. }
  245. /**
  246. * {@inheritDoc}
  247. */
  248. @Override
  249. public boolean checkAttribute(SqlRuntimeContext runtimeCtx, Object bean, String attrName) {
  250. return getGetter(runtimeCtx, bean.getClass(), attrName, true) != null;
  251. }
  252. /**
  253. * {@inheritDoc}
  254. */
  255. @Override
  256. public Object getAttribute(SqlRuntimeContext runtimeCtx, Object bean, String attrName) throws SqlRuntimeException {
  257. Method getter = getGetter(runtimeCtx, bean.getClass(), attrName, true);
  258. if (getter == null)
  259. throw new SqlRuntimeException("getAttribute(NoSuchMethodException): there's no getter for '" + attrName
  260. + "' in " + bean.getClass());
  261. return invokeMethod(runtimeCtx, bean, getter);
  262. }
  263. // setters
  264. protected Method getSetter(SqlRuntimeContext runtimeCtx, Class<?> clazz, String attrName, boolean onlyCheck,
  265. Class<?>... attrTypes) {
  266. String keyName = clazz.getName() + "." + attrName + attrTypes2String(attrTypes);
  267. Method _setter = setters.get(keyName);
  268. if (_setter != null)
  269. return _setter;
  270. PropertyDescriptor descriptor = getAttributeDescriptor(clazz, attrName);
  271. if (descriptor == null) {
  272. if (!onlyCheck)
  273. logger.error("getSetter: there's no attribute " + attrName + " in " + clazz.getName());
  274. return null;
  275. }
  276. _setter = getMethod(clazz, descriptor.getWriteMethod(), onlyCheck);
  277. if (_setter == null)
  278. return null;
  279. if (_setter.getParameterTypes() == null || _setter.getParameterTypes().length != 1) {
  280. if (!onlyCheck)
  281. logger.error("getSetter: there's no setter " + attrName + " in " + clazz.getName());
  282. return null;
  283. }
  284. Method setter = null;
  285. if (attrTypes == null || attrTypes.length == 0) {
  286. setter = _setter;
  287. } else {
  288. Class<?> setterType = _setter.getParameterTypes()[0];
  289. for (Class<?> _clazz : attrTypes) {
  290. if (_clazz.isAssignableFrom(setterType))
  291. setter = _setter;
  292. }
  293. }
  294. if (setter == null) {
  295. if (!onlyCheck)
  296. logger.error("getSetter: there's no setter " + attrName + " in " + clazz.getName());
  297. return null;
  298. }
  299. Method setterPrev = setters.putIfAbsent(keyName, setter);
  300. if (setterPrev != null)
  301. return setterPrev;
  302. return setter;
  303. }
  304. protected Method getSetter(SqlRuntimeContext runtimeCtx, Object bean, String attrName, boolean onlyCheck,
  305. Class<?>... attrTypes) {
  306. return getSetter(runtimeCtx, bean.getClass(), attrName, onlyCheck, attrTypes);
  307. }
  308. /**
  309. * {@inheritDoc}
  310. */
  311. @Override
  312. public boolean simpleSetAttribute(SqlRuntimeContext runtimeCtx, Object bean, String attrName, Object attrValue,
  313. Class<?>... attrTypes) {
  314. Method setter = getSetter(runtimeCtx, bean, attrName, true, attrTypes);
  315. if (setter != null) {
  316. invokeMethod(runtimeCtx, bean, setter, attrValue);
  317. return true;
  318. } else {
  319. return false;
  320. }
  321. }
  322. /**
  323. * {@inheritDoc}
  324. */
  325. @Override
  326. public void setAttribute(SqlRuntimeContext runtimeCtx, Object bean, String attrName, Object attrValue)
  327. throws SqlRuntimeException {
  328. Method setter = getSetter(runtimeCtx, bean, attrName, true);
  329. if (setter == null)
  330. throw new SqlRuntimeException("setAttribute(NoSuchMethodException): there's no setter for '" + attrName
  331. + "' in " + bean.getClass());
  332. invokeMethod(runtimeCtx, bean, setter, attrValue);
  333. }
  334. // methods invocation
  335. protected Method getMethod(Class<?> clazz, Method method, boolean onlyCheck) {
  336. if (method == null)
  337. return null;
  338. if (!Modifier.isPublic(method.getModifiers())) {
  339. if (!onlyCheck)
  340. logger.error("getMethod: " + method.toString() + " in " + clazz.getName() + " is not public");
  341. return null;
  342. }
  343. boolean sameClass = true;
  344. if (clazz == null) {
  345. clazz = method.getDeclaringClass();
  346. } else {
  347. sameClass = clazz.equals(method.getDeclaringClass());
  348. if (!method.getDeclaringClass().isAssignableFrom(clazz)) {
  349. if (!onlyCheck)
  350. logger.error("getMethod: " + clazz.getName() + " is not assignable from "
  351. + method.getDeclaringClass().getName());
  352. return null;
  353. }
  354. }
  355. if (Modifier.isPublic(clazz.getModifiers())) {
  356. if (!sameClass && !Modifier.isPublic(method.getDeclaringClass().getModifiers())) {
  357. try {
  358. method.setAccessible(true);
  359. } catch (SecurityException se) {
  360. logger.warn("getMethod: " + method.toString() + " in " + clazz + " " + se.getMessage());
  361. }
  362. }
  363. return method;
  364. }
  365. String methodName = method.getName();
  366. Class<?>[] parameterTypes = method.getParameterTypes();
  367. Method _method = getInterfaceMethod(clazz, methodName, parameterTypes);
  368. if (_method == null)
  369. _method = getSuperclassMethod(clazz, methodName, parameterTypes);
  370. if (_method == null)
  371. if (!onlyCheck)
  372. logger.error("getMethod: there's no method " + method.toString() + " in " + clazz.getName());
  373. return _method;
  374. }
  375. protected Method getInterfaceMethod(Class<?> clazz, String methodName, Class<?>[] parameterTypes) {
  376. Method method = null;
  377. for (; clazz != null; clazz = clazz.getSuperclass()) {
  378. Class<?>[] interfaces = clazz.getInterfaces();
  379. for (int i = 0; i < interfaces.length; i++) {
  380. if (!Modifier.isPublic(interfaces[i].getModifiers())) {
  381. continue;
  382. }
  383. try {
  384. method = interfaces[i].getDeclaredMethod(methodName, parameterTypes);
  385. } catch (NoSuchMethodException e) {
  386. }
  387. if (method != null) {
  388. return method;
  389. }
  390. method = getInterfaceMethod(interfaces[i], methodName, parameterTypes);
  391. if (method != null) {
  392. return method;
  393. }
  394. }
  395. }
  396. return null;
  397. }
  398. protected Method getSuperclassMethod(Class<?> clazz, String methodName, Class<?>[] parameterTypes) {
  399. Class<?> parentClazz = clazz.getSuperclass();
  400. while (parentClazz != null) {
  401. if (Modifier.isPublic(parentClazz.getModifiers())) {
  402. try {
  403. return parentClazz.getMethod(methodName, parameterTypes);
  404. } catch (NoSuchMethodException e) {
  405. return null;
  406. }
  407. }
  408. parentClazz = parentClazz.getSuperclass();
  409. }
  410. return null;
  411. }
  412. protected Method getMethod(Class<?> clazz, String methodName, boolean onlyCheck, Class<?>... parameterTypes) {
  413. String keyName = clazz.getName() + "." + methodName + attrTypes2String(parameterTypes);
  414. Method method = methods.get(keyName);
  415. if (method != null)
  416. return method;
  417. try {
  418. method = clazz.getMethod(methodName, parameterTypes);
  419. try {
  420. method.setAccessible(true);
  421. } catch (SecurityException se) {
  422. logger.warn("getMethod: not accessible method " + method.toString() + " in " + clazz.getName() + ": "
  423. + se.getMessage());
  424. }
  425. return method;
  426. } catch (NoSuchMethodException e) {
  427. if (!onlyCheck)
  428. logger.warn("getMethod: there's no method " + methodName + " in " + clazz.getName());
  429. }
  430. for (Method _method : clazz.getMethods()) {
  431. if (_method.getName().equals(methodName)) {
  432. Class<?>[] methodsParams = _method.getParameterTypes();
  433. int methodParamSize = methodsParams.length;
  434. if (methodParamSize == parameterTypes.length) {
  435. boolean match = true;
  436. for (int n = 0; n < methodParamSize; n++) {
  437. if (!areTheSameParameters(methodsParams[n], parameterTypes[n])) {
  438. match = false;
  439. break;
  440. }
  441. }
  442. if (match) {
  443. method = getMethod(clazz, _method, onlyCheck);
  444. if (method != null)
  445. break;
  446. }
  447. }
  448. }
  449. }
  450. if (method == null) {
  451. if (!onlyCheck)
  452. logger.error("getMethod: there's no method " + methodName + " in " + clazz.getName());
  453. } else {
  454. Method methodPrev = methods.putIfAbsent(keyName, method);
  455. if (methodPrev != null)
  456. return methodPrev;
  457. }
  458. return method;
  459. }
  460. protected final boolean areTheSameParameters(Class<?> methodParameterType, Class<?> parameterType) {
  461. if (methodParameterType.isAssignableFrom(parameterType)) {
  462. return true;
  463. }
  464. if (methodParameterType.isPrimitive()) {
  465. Class<?> parameterWrapperClazz = getPrimitiveWrapper(methodParameterType);
  466. if (parameterWrapperClazz != null) {
  467. return parameterWrapperClazz.equals(parameterType);
  468. }
  469. }
  470. return false;
  471. }
  472. protected Class<?> getPrimitiveWrapper(Class<?> primitiveType) {
  473. if (boolean.class.equals(primitiveType)) {
  474. return Boolean.class;
  475. } else if (float.class.equals(primitiveType)) {
  476. return Float.class;
  477. } else if (long.class.equals(primitiveType)) {
  478. return Long.class;
  479. } else if (int.class.equals(primitiveType)) {
  480. return Integer.class;
  481. } else if (short.class.equals(primitiveType)) {
  482. return Short.class;
  483. } else if (byte.class.equals(primitiveType)) {
  484. return Byte.class;
  485. } else if (double.class.equals(primitiveType)) {
  486. return Double.class;
  487. } else if (char.class.equals(primitiveType)) {
  488. return Character.class;
  489. } else {
  490. return null;
  491. }
  492. }
  493. protected Object invokeMethod(SqlRuntimeContext runtimeCtx, Object bean, Method method, Object... args) {
  494. try {
  495. if (!method.isAccessible())
  496. try {
  497. method.setAccessible(true);
  498. } catch (SecurityException se) {
  499. logger.warn("invokeMethod: " + bean.getClass() + " " + se.getMessage());
  500. }
  501. return method.invoke(bean, args);
  502. } catch (IllegalAccessException e) {
  503. throw new SqlRuntimeException(debugInfo("invokeMethod", bean, method, args), e);
  504. } catch (IllegalArgumentException e) {
  505. throw new SqlRuntimeException(debugInfo("invokeMethod", bean, method, args), e);
  506. } catch (InvocationTargetException e) {
  507. throw new SqlRuntimeException(debugInfo("invokeMethod", bean, method, args), e);
  508. }
  509. }
  510. /**
  511. * {@inheritDoc}
  512. */
  513. @Override
  514. public boolean checkMethod(SqlRuntimeContext runtimeCtx, Class<?> clazz, String methodName, Class<?>... argTypes) {
  515. return getMethod(clazz, methodName, true, argTypes) != null;
  516. }
  517. /**
  518. * {@inheritDoc}
  519. */
  520. @Override
  521. public boolean checkMethod(SqlRuntimeContext runtimeCtx, Object bean, String methodName, Object... args) {
  522. return getMethod(bean.getClass(), methodName, true, toParameterTypes(args)) != null;
  523. }
  524. /**
  525. * {@inheritDoc}
  526. */
  527. @Override
  528. public Object invokeMethod(SqlRuntimeContext runtimeCtx, Class<?> clazz, String methodName, Object... args)
  529. throws SqlRuntimeException {
  530. return invokeMethod(runtimeCtx, clazz, null, methodName, args);
  531. }
  532. /**
  533. * {@inheritDoc}
  534. */
  535. @Override
  536. public Object invokeMethod(SqlRuntimeContext runtimeCtx, Object bean, String methodName, Object... args)
  537. throws SqlRuntimeException {
  538. return invokeMethod(runtimeCtx, bean.getClass(), bean, methodName, args);
  539. }
  540. protected Object invokeMethod(SqlRuntimeContext runtimeCtx, Class<?> clazz, Object bean, String methodName,
  541. Object... args) throws SqlRuntimeException {
  542. Class<?>[] parameterTypes = toParameterTypes(args);
  543. Method method = getMethod(clazz, methodName, true, parameterTypes);
  544. if (method == null) {
  545. throw new SqlRuntimeException(debugInfo("invokeMethod(NoSuchMethodException)", bean, method, args));
  546. }
  547. try {
  548. return method.invoke(bean, args);
  549. } catch (IllegalAccessException e) {
  550. throw new SqlRuntimeException(debugInfo("invokeMethod", bean, method, args), e);
  551. } catch (IllegalArgumentException e) {
  552. throw new SqlRuntimeException(debugInfo("invokeMethod", bean, method, args), e);
  553. } catch (InvocationTargetException e) {
  554. throw new SqlRuntimeException(debugInfo("invokeMethod", bean, method, args), e);
  555. }
  556. }
  557. // enums
  558. /**
  559. * {@inheritDoc}
  560. */
  561. @Override
  562. public Object getEnumToValue(SqlRuntimeContext runtimeCtx, Object bean) {
  563. if (bean == null)
  564. return null;
  565. String keyName = bean.getClass().getName();
  566. Method method = enumsIn.get(keyName);
  567. if (method != null)
  568. return invokeMethod(runtimeCtx, bean, method);
  569. for (String methodName : runtimeCtx.getFeatures(SqlFeature.METHODS_ENUM_IN.name())) {
  570. method = getMethod(bean.getClass(), methodName, true);
  571. if (method != null)
  572. break;
  573. }
  574. if (method == null)
  575. return null;
  576. enumsIn.put(keyName, method);
  577. return invokeMethod(runtimeCtx, bean, method);
  578. }
  579. /**
  580. * {@inheritDoc}
  581. */
  582. @Override
  583. public Class<?> getEnumToClass(SqlRuntimeContext runtimeCtx, Class<?> clazz) {
  584. if (clazz == null)
  585. return null;
  586. String keyName = clazz.getName();
  587. Method method = enumsIn.get(keyName);
  588. if (method != null)
  589. return method.getReturnType();
  590. for (String methodName : runtimeCtx.getFeatures(SqlFeature.METHODS_ENUM_IN.name())) {
  591. method = getMethod(clazz, methodName, true);
  592. if (method != null)
  593. break;
  594. }
  595. if (method == null)
  596. return null;
  597. enumsIn.put(keyName, method);
  598. return method.getReturnType();
  599. }
  600. /**
  601. * {@inheritDoc}
  602. */
  603. @Override
  604. public Object getValueToEnum(SqlRuntimeContext runtimeCtx, Class<?> clazz, Object val) {
  605. if (val == null)
  606. return null;
  607. Class<?>[] parameterTypes = toParameterTypes(val);
  608. String keyName = clazz.getName() + attrTypes2String(parameterTypes);
  609. Method method = enumsOut.get(keyName);
  610. if (method != null)
  611. return invokeMethod(runtimeCtx, clazz, method, val);
  612. for (String methodName : runtimeCtx.getFeatures(SqlFeature.METHODS_ENUM_OUT.name())) {
  613. method = getMethod(clazz, methodName, true, parameterTypes);
  614. if (method != null)
  615. break;
  616. }
  617. if (method == null)
  618. return null;
  619. enumsOut.put(keyName, method);
  620. return invokeMethod(runtimeCtx, clazz, method, val);
  621. }
  622. // misc
  623. protected String attrTypes2String(Class<?>... attrTypes) {
  624. if (attrTypes == null || attrTypes.length == 0)
  625. return "";
  626. StringBuilder sb = new StringBuilder(".");
  627. boolean first = true;
  628. for (Class<?> attrType : attrTypes) {
  629. if (first)
  630. first = false;
  631. else
  632. sb.append(",");
  633. sb.append(attrType.getName());
  634. }
  635. return sb.toString();
  636. }
  637. protected Object[] toArray(Object arg) {
  638. Object[] args = null;
  639. if (arg != null) {
  640. args = new Object[] { arg };
  641. }
  642. return args;
  643. }
  644. protected Class<?>[] toParameterTypes(Object arg) {
  645. if (arg == null)
  646. return new Class[0];
  647. return new Class[] { arg.getClass() };
  648. }
  649. protected Class<?>[] toParameterTypes(Object[] args) {
  650. if (args == null)
  651. return new Class[0];
  652. Class<?>[] parameterTypes = new Class[args.length];
  653. for (int i = 0; i < args.length; i++) {
  654. parameterTypes[i] = args[i].getClass();
  655. }
  656. return parameterTypes;
  657. }
  658. protected String debugInfo(String msg, Object bean, Method method, Object... args) {
  659. StringBuilder sb = new StringBuilder(msg);
  660. sb.append(": bean=").append((bean != null) ? bean.getClass() : "null");
  661. sb.append(", method=").append((method != null) ? method.toString() : "null");
  662. if (args != null) {
  663. Class<?>[] parameterTypes = toParameterTypes(args);
  664. sb.append(", args=").append(attrTypes2String(parameterTypes));
  665. }
  666. if (method != null)
  667. sb.append(", method params=")
  668. .append((method.getParameterTypes() != null) ? Arrays.asList(method.getParameterTypes()) : "empty");
  669. return sb.toString();
  670. }
  671. }