Don't worry about this ! The culprit was incorrect configuration in db
-----Original Message----- From: Simran Narula [mailto:snar...@avoka.com] Sent: Tuesday, 10 April 2012 1:55 PM To: user@cayenne.apache.org Subject: RE: Track Column level changes in Cayenne Hi Guys, Looks like need a little more help on below, I am getting The INSERT statement conflicted with the FOREIGN KEY constraint "X_Constraint". The conflict occurred in database "MyDatabase", table "dbo.X", column 'X_oid'. As raised earlier, I want to track column level changes on a particular entity, and for that Andrus advised me to get a parallel context and retrieve same object from the database and do the comparison Between the entity (that might have changed values) and fresh object that I have retrieved from the database using the parallel ObjectContext. What I am doing is below, On the save click on a page, I am saving the entity X like this: myEntityService.save(X) // where save() just checks if it's a new object and registers it, otherwise just commits the changes using commitChanges(); Then, I have preUpdate() callback on X Entity Class, like this: onPreUpdate() { if (hasPropertyValueChanged(properyNameConstant)) { ObjectContext context = DataContext.getThreadObjectContext(); Log l = (Log)context.newObject(Log.class); // populate log instance with whatever details addToLog(l); } } Where addToLog()... is simply addToManyTarget("Log", obj, true); ... and finally, I have the hasPropertyValueChanged is something like this... hasPropertyValueChanged(String properyName) { ObjectContext parallelContext = DataContext.createDataContext(false); ObjEntity objEntity = parallelContext.getEntityResolver().lookupObjEntity(entityClass); String pkName = CayenneUtils.getPkName(entityClass); ObjectId objectId = new ObjectId(objEntity.getName(), pkName, pkId); ObjectIdQuery objectIdQuery = new ObjectIdQuery(objectId, false, ObjectIdQuery.CACHE_REFRESH); MyBaseEntity persistedObject = (MyBaseEntity) DataObjectUtils.objectForQuery(parallelContext, objectIdQuery); // Compare the value in the current entity object (where we suspect the value to might have changed of a particular column) and the persisted object that we just obtained // if the value has been changed then return true... allowing the new Log to be created... } Once this is executed.. I get a The INSERT statement conflicted with the FOREIGN KEY constraint "X_Constraint". The conflict occurred in database "MyDatabase", table "dbo.X", column 'X_oid'. Which eventually causes the CommitException... Why am I getting this... ? I know its complaining about foreign key constraint, on my X table which is represented by X entity in the code examples above, the Log table has a foreign key reference to X_oid in X Table [Click] [error] handleException: org.apache.cayenne.CayenneRuntimeException: [v.3.0.2 Jun 11 2011 09:52:20] Commit Exception at org.apache.cayenne.access.DataContext.flushToParent(DataContext.java:1149) at org.apache.cayenne.access.DataContext.commitChanges(DataContext.java:1060) at XXX.CayenneService.commitChanges(CayenneService.java:X) at XXX.EntityService.save(EntityService.java:X) at XXX.onSaveClick(MyPage.java:X) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25) at java.lang.reflect.Method.invoke(Method.java:585) at org.apache.click.util.ClickUtils.invokeListener(ClickUtils.java:1949) at org.apache.click.util.ActionListenerAdaptor.onAction(ActionListenerAdaptor.java:59) at org.apache.click.ActionEventDispatcher.fireActionEvent(ActionEventDispatcher.java:248) at org.apache.click.ActionEventDispatcher.fireActionEvents(ActionEventDispatcher.java:193) at org.apache.click.ActionEventDispatcher$EventHolder.fireActionEvents(ActionEventDispatcher.java:430) at org.apache.click.ActionEventDispatcher.fireActionEvents(ActionEventDispatcher.java:226) at org.apache.click.ActionEventDispatcher.fireActionEvents(ActionEventDispatcher.java:212) at org.apache.click.ClickServlet.performOnProcess(ClickServlet.java:666) at org.apache.click.ClickServlet.processPage(ClickServlet.java:562) at org.apache.click.ClickServlet.handleRequest(ClickServlet.java:379) at org.apache.click.ClickServlet.doPost(ClickServlet.java:294) at javax.servlet.http.HttpServlet.service(HttpServlet.java:717) at javax.servlet.http.HttpServlet.service(HttpServlet.java:810) at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:290) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206) at xxx at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206) at xxx at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206) <...irrelevant lines truncated> at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206) at xxx at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235) at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206) at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:233) at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:191) at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:127) at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:102) at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:109) at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:298) at org.apache.coyote.http11.Http11Processor.process(Http11Processor.java:852) at org.apache.coyote.http11.Http11Protocol$Http11ConnectionHandler.process(Http11Protocol.java:588) at org.apache.tomcat.util.net.JIoEndpoint$Worker.run(JIoEndpoint.java:489) at java.lang.Thread.run(Thread.java:595) Caused by: com.microsoft.sqlserver.jdbc.SQLServerException: The INSERT statement conflicted with the FOREIGN KEY constraint "X_Contraint". The conflict occurred in database "MyDatabase", table "dbo.X", column 'X_oid'. at com.microsoft.sqlserver.jdbc.SQLServerException.makeFromDatabaseError(Unknown Source) at com.microsoft.sqlserver.jdbc.IOBuffer.processPackets(Unknown Source) at com.microsoft.sqlserver.jdbc.SQLServerStatement.sendExecute(Unknown Source) at com.microsoft.sqlserver.jdbc.SQLServerStatement.doExecuteUpdate(Unknown Source) at com.microsoft.sqlserver.jdbc.SQLServerPreparedStatement.executeUpdate(Unknown Source) at org.apache.cayenne.access.jdbc.BatchAction.runAsIndividualQueries(BatchAction.java:221) at org.apache.cayenne.access.jdbc.BatchAction.performAction(BatchAction.java:91) at org.apache.cayenne.dba.sqlserver.SQLServerBatchAction.performAction(SQLServerBatchAction.java:59) at org.apache.cayenne.access.DataNodeQueryAction.runQuery(DataNodeQueryAction.java:87) at org.apache.cayenne.access.DataNode.performQueries(DataNode.java:269) at org.apache.cayenne.access.DataDomainFlushAction.runQueries(DataDomainFlushAction.java:226) at org.apache.cayenne.access.DataDomainFlushAction.flush(DataDomainFlushAction.java:144) at org.apache.cayenne.access.DataDomain.onSyncFlush(DataDomain.java:824) at org.apache.cayenne.access.DataDomain$2.transform(DataDomain.java:791) at org.apache.cayenne.access.DataDomain.runInTransaction(DataDomain.java:850) at org.apache.cayenne.access.DataDomain.onSync(DataDomain.java:788) at org.apache.cayenne.access.DataContext.flushToParent(DataContext.java:1121) ... 65 more -----Original Message----- From: Simran Narula Sent: Thursday, 5 April 2012 1:54 PM To: 'user@cayenne.apache.org' Subject: RE: Track Column level changes in Cayenne Hey Andrus, Thanks for your reply that day. Just wanted to let you know this Worked !! Thanks heaps for help ! -----Original Message----- From: Andrus Adamchik [mailto:and...@objectstyle.org] Sent: Wednesday, 28 March 2012 6:50 PM To: user@cayenne.apache.org Subject: Re: Track Column level changes in Cayenne (reposting from stackoverflow answer) There are a few ways to approach this. Here is the one that IMO is the most transparent. The trick is to use a different ObjectContext from the one committing changes. Then you will get a separate copy of the object that will contain currently saved value: // 'this' refers to the DataObject being committed (assuming things happen in its callback) ObjectContext parallelContext = ... // create a new context here like you normally would // 3.1 API; 3.0.x has a similar method with a slightly different sig MyClass clone = parallelContext.localObject(this); // if you are ok with cached old value, ignore the 'invalidateObjects' call. // If not, uncomment it to ensure the object gets refetched. // Also 3.1 API. Should be easy to adjust for 3.0 // parallelContext.invalidateObjects(clone); Object oldValue = clone.getXyz(); Andrus On Mar 28, 2012, at 7:58 AM, Simran Narula wrote: > Hi, > > I need to log each action taken on my portal UI for e.g. > > Price changed for product x > > Assume all the information is contained within one table i.e.both price & the > product information. > > When the portal is loaded with product information, user can change the price > of the product. > > I want to check if the price has changed or not and for that I am using a > pre-update call back in Cayenne entity to check these changes, I want to > check if the price x is changed to Y and push another history record in the > Database. So far I have tried cache strategies - NO_CACHE and other > workarounds but no success yet. > > surely I can achieve this by other hacky means, but do not want to. My > question is... is there any way to track column level changes in Cayenne ? > > Thanks. > Simran > >