/fcrepo-security/fcrepo-security-pdp/src/main/java/org/fcrepo/server/security/xacml/pdp/decorator/PolicyIndexInvocationHandler.java

https://github.com/fcrepo/fcrepo · Java · 379 lines · 241 code · 80 blank · 58 comment · 33 complexity · 893888fae0e56a54643319b0db96f6b0 MD5 · raw file

  1. /* The contents of this file are subject to the license and copyright terms
  2. * detailed in the license directory at the root of the source tree (also
  3. * available online at http://fedora-commons.org/license/).
  4. */
  5. package org.fcrepo.server.security.xacml.pdp.decorator;
  6. import java.io.IOException;
  7. import java.io.InputStream;
  8. import java.lang.reflect.InvocationTargetException;
  9. import java.lang.reflect.Method;
  10. import org.apache.commons.io.IOUtils;
  11. import org.fcrepo.server.Server;
  12. import org.fcrepo.server.errors.GeneralException;
  13. import org.fcrepo.server.errors.InitializationException;
  14. import org.fcrepo.server.proxy.AbstractInvocationHandler;
  15. import org.fcrepo.server.proxy.ModuleConfiguredInvocationHandler;
  16. import org.fcrepo.server.security.xacml.pdp.data.FedoraPolicyStore;
  17. import org.fcrepo.server.security.xacml.pdp.data.PolicyIndex;
  18. import org.fcrepo.server.security.xacml.pdp.data.PolicyIndexException;
  19. import org.fcrepo.server.storage.DOManager;
  20. import org.slf4j.Logger;
  21. import org.slf4j.LoggerFactory;
  22. /**
  23. * A {@link java.lang.reflect.InvocationHandler InvocationHandler} responsible
  24. * for updating the FeSL XACML policy index whenever API-M
  25. * invocations result in changes to a policy stored in a Fedora object.
  26. *
  27. * @author Stephen Bayliss
  28. * @version $Id$
  29. */
  30. public class PolicyIndexInvocationHandler
  31. extends AbstractInvocationHandler
  32. implements ModuleConfiguredInvocationHandler {
  33. /** Logger for this class. */
  34. private static Logger LOG =
  35. LoggerFactory.getLogger(PolicyIndexInvocationHandler.class.getName());
  36. private DOManager m_DOManager = null;
  37. // for updating the policy cache
  38. private PolicyIndex m_policyIndex = null;
  39. public void setPolicyIndex(PolicyIndex policyIndex) {
  40. m_policyIndex = policyIndex;
  41. }
  42. public void setDOManager(DOManager manager) {
  43. m_DOManager = manager;
  44. }
  45. public void init(Server server) throws InitializationException {
  46. m_DOManager = (DOManager)server.getBean(DOManager.class.getName());
  47. m_policyIndex = (PolicyIndex)server.getBean(PolicyIndex.class.getName());
  48. if (m_DOManager == null) {
  49. LOG.error("DOManager module was not set");
  50. throw new InitializationException("DOManager module was not set");
  51. }
  52. if (m_policyIndex == null) {
  53. LOG.error("PolicyIndex was not set");
  54. throw new InitializationException("PolicyIndex was not set");
  55. }
  56. }
  57. /**
  58. * {@inheritDoc}
  59. */
  60. @Override
  61. public Object invoke(Object proxy, Method method, Object[] args)
  62. throws Throwable {
  63. // get the method parameters and "type"
  64. ManagementMethodInvocation managementMethod = new ManagementMethodInvocation(method, args);
  65. // if it's not a method that requires policy cache modifications, quit
  66. if (managementMethod.action.equals(ManagementMethodInvocation.Action.NA))
  67. return invokeTarget(target, method, args);
  68. Object returnValue = null;
  69. // holds the state of the Fedora policy object before and after the API method is invoked
  70. PolicyObject policyBefore = null;
  71. PolicyObject policyAfter = null;
  72. // validation note:
  73. // validation is carried out as part of the API methods
  74. // policy index update occurs after the management method has been invoked
  75. // therefore if a validation error occurs the object will not have changed,
  76. // and the resulting exception means the policy index won't be updated (and therefore remain in sync)
  77. switch (managementMethod.target) {
  78. case DIGITALOBJECT: // API methods operating on the object
  79. switch (managementMethod.component) {
  80. case STATE: // object state change
  81. // if the requested object state is null, then there's no state change - no change to policy index
  82. if (managementMethod.parameters.objectState != null && managementMethod.parameters.objectState.length() > 0) {
  83. // get the "before" object and determine if it was indexed in the cache
  84. policyBefore = new PolicyObject(m_DOManager,
  85. managementMethod.parameters.context,
  86. managementMethod.parameters.pid,
  87. null,
  88. null,
  89. null);
  90. boolean indexedBefore = policyBefore.isPolicyActive();
  91. returnValue = invokeTarget(target, method, args);
  92. // get the object after the state change and determine if it should be indexed
  93. policyAfter = new PolicyObject(m_DOManager,
  94. managementMethod.parameters.context,
  95. managementMethod.parameters.pid,
  96. managementMethod.parameters.objectState,
  97. null,
  98. null);
  99. // if the indexing status has changed, make appropriate changes to the policy cache/index
  100. if (indexedBefore != policyAfter.isPolicyActive()) {
  101. if (policyAfter.isPolicyActive() ) {
  102. addPolicy(managementMethod.parameters.pid, policyAfter.getDsContent());
  103. } else {
  104. deletePolicy(managementMethod.parameters.pid);
  105. }
  106. }
  107. }
  108. break;
  109. case CONTENT: // object ingest or purge
  110. switch (managementMethod.action){
  111. case CREATE: // ingest
  112. // do the API call
  113. returnValue = invokeTarget(target, method, args);
  114. // get the ingested policy - note ingested object PID is the return value
  115. policyAfter = new PolicyObject(m_DOManager,
  116. managementMethod.parameters.context,
  117. (String)returnValue,
  118. null,
  119. null,
  120. null);
  121. // add to the cache if required
  122. if (policyAfter.isPolicyActive()) {
  123. addPolicy((String)returnValue, policyAfter.getDsContent());
  124. }
  125. break;
  126. case DELETE: // purge
  127. // get the policy object that existed prior to ingest and see if it was indexed
  128. policyBefore = new PolicyObject(m_DOManager,
  129. managementMethod.parameters.context,
  130. managementMethod.parameters.pid,
  131. null,
  132. null,
  133. null);
  134. boolean wasIndexed = policyBefore.isPolicyActive();
  135. // do the API call
  136. returnValue = invokeTarget(target, method, args);
  137. // if the policy was indexed, delete it from the cache/index
  138. if (wasIndexed)
  139. deletePolicy(managementMethod.parameters.pid);
  140. break;
  141. default:
  142. }
  143. break;
  144. default:
  145. }
  146. break;
  147. case DATASTREAM: // operations on datastreams
  148. // note, DS ID can be null - server-assigned ID
  149. if (managementMethod.parameters.dsID != null && managementMethod.parameters.dsID.equals(FedoraPolicyStore.FESL_POLICY_DATASTREAM)) {
  150. switch (managementMethod.component) {
  151. case STATE: // datastream state change
  152. // get the object prior to the API call and see if it was indexed/cached
  153. policyBefore = new PolicyObject(m_DOManager,
  154. managementMethod.parameters.context,
  155. managementMethod.parameters.pid,
  156. null,
  157. managementMethod.parameters.dsID,
  158. null);
  159. boolean wasIndexed = policyBefore.isPolicyActive();
  160. // do the API call
  161. returnValue = invokeTarget(target, method, args);
  162. // the object after the call
  163. policyAfter = new PolicyObject(m_DOManager,
  164. managementMethod.parameters.context,
  165. managementMethod.parameters.pid,
  166. null,
  167. managementMethod.parameters.dsID,
  168. managementMethod.parameters.dsState);
  169. // if indexing status has changed, update the cache/index
  170. if (wasIndexed != policyAfter.isPolicyActive()) {
  171. if (policyAfter.isPolicyActive()) {
  172. addPolicy(managementMethod.parameters.pid, policyAfter.getDsContent());
  173. } else {
  174. deletePolicy(managementMethod.parameters.pid);
  175. }
  176. }
  177. break;
  178. case CONTENT: // datastream add, modify, purge
  179. switch (managementMethod.action){
  180. case CREATE:
  181. // do the API call
  182. returnValue = invokeTarget(target, method, args);
  183. // get the policy object after the method call
  184. policyAfter = new PolicyObject(m_DOManager,
  185. managementMethod.parameters.context,
  186. managementMethod.parameters.pid,
  187. null,
  188. managementMethod.parameters.dsID,
  189. managementMethod.parameters.dsState);
  190. if (policyAfter.isPolicyActive())
  191. addPolicy(managementMethod.parameters.pid, policyAfter.getDsContent());
  192. break;
  193. case DELETE:
  194. // get the policy object before the call and see if it was indexed
  195. policyBefore = new PolicyObject(m_DOManager,
  196. managementMethod.parameters.context,
  197. managementMethod.parameters.pid,
  198. null,
  199. managementMethod.parameters.dsID,
  200. managementMethod.parameters.dsState);
  201. wasIndexed = policyBefore.isPolicyActive();
  202. // invoke the method
  203. returnValue = invokeTarget(target, method, args);
  204. policyAfter = new PolicyObject(m_DOManager,
  205. managementMethod.parameters.context,
  206. managementMethod.parameters.pid,
  207. null,
  208. managementMethod.parameters.dsID,
  209. null);
  210. if (wasIndexed) {
  211. //if policy is still active after, only a version was purged and the policy has changed. Update
  212. if (policyAfter.isPolicyActive()) {
  213. updatePolicy(managementMethod.parameters.pid, policyAfter.getDsContent());
  214. //policy is not active after, then it was deleted and needs to be removed from index.
  215. } else {
  216. deletePolicy(managementMethod.parameters.pid);
  217. }
  218. }
  219. break;
  220. case UPDATE:
  221. // do the API call, get the policy object after the call
  222. returnValue = invokeTarget(target, method, args);
  223. policyAfter = new PolicyObject(m_DOManager,
  224. managementMethod.parameters.context,
  225. managementMethod.parameters.pid,
  226. null,
  227. managementMethod.parameters.dsID,
  228. null);
  229. if (policyAfter.isPolicyActive())
  230. updatePolicy(managementMethod.parameters.pid, policyAfter.getDsContent());
  231. break;
  232. default:
  233. }
  234. break;
  235. default:
  236. }
  237. }
  238. break;
  239. default:
  240. }
  241. // if API call not made as part of the above (ie no action was required on policy cache), do it now
  242. if (returnValue == null)
  243. returnValue = invokeTarget(target,method, args);
  244. return returnValue;
  245. }
  246. /**
  247. * Invoke the underlying method, catching any InvocationTargetException and rethrowing the target exception
  248. */
  249. private Object invokeTarget(Object target, Method method, Object[] args) throws Throwable {
  250. Object returnValue;
  251. try {
  252. returnValue = method.invoke(target, args);
  253. } catch(InvocationTargetException ite) {
  254. throw ite.getTargetException();
  255. }
  256. return returnValue;
  257. }
  258. // TODO: invalidate PEP cache on the following operations
  259. private void addPolicy(String pid, InputStream dsContent) throws GeneralException {
  260. LOG.info("Adding policy " + pid);
  261. String policy;
  262. try {
  263. policy = IOUtils.toString(dsContent);
  264. dsContent.close();
  265. m_policyIndex.addPolicy(pid, policy);
  266. } catch (IOException e) {
  267. throw new GeneralException("Error adding policy " + pid + " to policy index: " + e.getMessage(), e);
  268. } catch (PolicyIndexException e) {
  269. throw new GeneralException("Error adding policy " + pid + " to policy index: " + e.getMessage(), e);
  270. }
  271. }
  272. private void updatePolicy(String pid, InputStream dsContent) throws GeneralException {
  273. LOG.info("Updating policy " + pid);
  274. String policy;
  275. try {
  276. policy = IOUtils.toString(dsContent);
  277. dsContent.close();
  278. m_policyIndex.updatePolicy(pid, policy);
  279. } catch (IOException e) {
  280. throw new GeneralException("Error adding policy " + pid + " to policy index: " + e.getMessage(), e);
  281. } catch (PolicyIndexException e) {
  282. throw new GeneralException("Error adding policy " + pid + " to policy index: " + e.getMessage(), e);
  283. }
  284. }
  285. /**
  286. * Remove the specified policy from the cache
  287. * @param pid
  288. * @throws PolicyIndexException
  289. */
  290. private void deletePolicy(String pid) throws GeneralException {
  291. LOG.debug("Deleting policy " + pid);
  292. try {
  293. // in case the policy doesn't exist (corrupt policy index, failed to add because invalid, etc)
  294. m_policyIndex.deletePolicy(pid);
  295. } catch (PolicyIndexException e) {
  296. throw new GeneralException("Error deleting policy " + pid + " from policy index: " + e.getMessage(), e);
  297. }
  298. }
  299. }