From 05cfd5783f5307bbcedb0b9205cebef11dae22df Mon Sep 17 00:00:00 2001 From: jprocter <jprocter@compbio.dundee.ac.uk> Date: Wed, 22 Mar 2006 14:12:54 +0000 Subject: [PATCH] finished the Appdata/ClientDocument update mechanism. Untested! git-svn-id: https://svn.lifesci.dundee.ac.uk/svn/repository/trunk@198 be28352e-c001-0410-b1a7-c7978e42abec --- .../vamsas/client/simpleclient/ClientDocument.java | 79 +++++++++++++------- .../vamsas/client/simpleclient/SimpleClient.java | 15 +++- .../client/simpleclient/SimpleClientAppdata.java | 53 ++++++++++--- 3 files changed, 109 insertions(+), 38 deletions(-) diff --git a/src/org/vamsas/client/simpleclient/ClientDocument.java b/src/org/vamsas/client/simpleclient/ClientDocument.java index 6b1d7fc..f89f713 100644 --- a/src/org/vamsas/client/simpleclient/ClientDocument.java +++ b/src/org/vamsas/client/simpleclient/ClientDocument.java @@ -3,6 +3,7 @@ */ package org.vamsas.client.simpleclient; +import java.io.IOException; import java.util.Vector; import org.apache.commons.logging.Log; @@ -10,9 +11,11 @@ import org.apache.commons.logging.LogFactory; import org.vamsas.client.ClientHandle; import org.vamsas.client.IClientAppdata; import org.vamsas.client.IClientDocument; +import org.vamsas.client.UserHandle; import org.vamsas.client.Vobject; import org.vamsas.client.VorbaId; import org.vamsas.objects.core.ApplicationData; +import org.vamsas.objects.core.User; import org.vamsas.objects.core.VAMSAS; import org.vamsas.objects.core.VamsasDocument; import org.vamsas.objects.utils.AppDataReference; @@ -346,8 +349,11 @@ public class ClientDocument extends org.vamsas.client.ClientDocument implements return null; } protected boolean updateSessionDocument() throws java.io.IOException { - if (!isModified() && !scappd.isModified()) + boolean docupdate = true; // 'non-serious' problems below set this false + if (!isModified() && !scappd.isModified()) { + log.debug("Document update not necessary. returning false."); return false; + } VamsasSession session = sclient._session; log.debug("updating Session Document in "+session.sessionDir); // update the VamsasDocument structure with any new appData's. @@ -355,10 +361,11 @@ public class ClientDocument extends org.vamsas.client.ClientDocument implements log.debug("Attempting to update session "+sclient.session.getSessionUrn()); if (scappd.isModified()) { ClientHandle client = sclient.client; + UserHandle user = sclient.user; scappd.closeForWriting(); - long newAppd = scappd.newAppData.sessionFile.length(); - if (scappd.appsGlobal==null) { + log.debug("Creating new appData entry for this application..."); + // first write for this application - add a new section in document ApplicationData appd = scappd.appsGlobal = new ApplicationData(); appd.setName(client.getClientName()); appd.setUrn(client.getClientUrn()); @@ -366,40 +373,61 @@ public class ClientDocument extends org.vamsas.client.ClientDocument implements doc.addApplicationData(appd); // embed or jarEntry ? - for now only jarEntry's are dealt with. appd.setDataReference(AppDataReference.uniqueAppDataReference(doc, sclient.client.getClientUrn())); + log.debug("... created."); } if (scappd.newAppData!=null && scappd.newAppData.sessionFile.exists()) { + log.debug("Beginning update for new Global Appdata..."); + // new global appdata to write. if (scappd.appsGlobal.getData()!=null) { scappd.appsGlobal.setData(null); scappd.appsGlobal.setDataReference(AppDataReference.uniqueAppDataReference(doc, sclient.client.getClientUrn())); } - - - + // LATER: use a switch to decide if the data should be written as a reference or as an embedded data chunk + scappd.updateAnAppdataEntry(archive, scappd.appsGlobal, scappd.newAppData); + log.debug("...Successfully updated Global Appdata Entry."); } - if (scappd.usersData==null) { - + if (scappd.newUserData!=null && scappd.newUserData.sessionFile.exists()) { + log.debug("Beginning to update Users Appdata entry...."); + if (scappd.usersData==null) { + // create new user appdata + scappd.usersData = new User(); + scappd.usersData.setFullname(user.getFullName()); + scappd.usersData.setOrganization(user.getOrganization()); + scappd.appsGlobal.addUser(scappd.usersData); + } + User appd = scappd.usersData; + if (appd.getData()!=null || appd.getDataReference()==null) { + // LATER make standard appDataReference constructor for client+user + appd.setData(null); + String safe_username = user.getFullName(); + int t = safe_username.indexOf(" "); + if (t!=-1) { + safe_username = safe_username.substring(t); + } + appd.setDataReference(AppDataReference.uniqueAppDataReference(doc, sclient.client.getClientUrn()+safe_username)); + } + scappd.updateAnAppdataEntry(archive, scappd.usersData, scappd.newUserData); + log.debug("...Successfully updated user AppData entry."); } - } - cdocument.archive.putVamsasDocument(vdoc); - // write the appHandle to the lastupdate file. - } - if (scappd!=null) { - if (scappd.accessedDocument) + log.debug("Updating Document..."); + // now update the document. + try { + archive.putVamsasDocument(doc); + log.debug("Successfully written document entry."); } - va = new org.vamsas.client.simpleclient.VamsasArchive(newf, false, true, sfile); - doc = va.getVamsasDocument(); - doc.addVAMSAS(Core.getDemoVamsas()); - doc.addApplicationData(makeDemoAppdata(va, - "org.vamsas.test.simpleclient.VamsasArchive", "another old Bugger esq", "rescinded")); - if (va.transferRemainingAppDatas()) - log.info("Remain appdatas were transferred."); + catch (Exception e) { + log.error("Marshalling error for vamsas document.",e); + docupdate = false; // pass on the (probable) object validation error + } + if (archive.transferRemainingAppDatas()) + log.debug("Remaining appdatas were transferred."); else - log.warn("No appdatas were transferred. This is wrong."); - va.putVamsasDocument(doc); - va.closeArchive(); + log.debug("No remaining appdatas were transferred. (Correct?)"); + archive.closeArchive(); + log.debug("...successully finished and closed."); - return true; // successfully updated. + return docupdate; // no errors ? } /* (non-Javadoc) * @see java.lang.Object#finalize() @@ -411,6 +439,7 @@ public class ClientDocument extends org.vamsas.client.ClientDocument implements scappd = null; } // TODO local garbage collection + super.finalize(); } diff --git a/src/org/vamsas/client/simpleclient/SimpleClient.java b/src/org/vamsas/client/simpleclient/SimpleClient.java index 084c7f5..970bcfa 100644 --- a/src/org/vamsas/client/simpleclient/SimpleClient.java +++ b/src/org/vamsas/client/simpleclient/SimpleClient.java @@ -100,7 +100,7 @@ public class SimpleClient implements IClient { throw new Exception("Failed to import data from "+importingArchive, e); } } - + /* * (non-Javadoc) * LATER: check that build substitution variables are correct @@ -273,8 +273,17 @@ public class SimpleClient implements IClient { if (log.isDebugEnabled()) log.debug("updateDocument for "+session.getSessionUrn()+" with unmodified IClientDocument."); } else { - if (cdocument.updateSessionDocument()) - log.warn("Session document did not update properly for session directory "+_session.sessionDir); + try { + if (!cdocument.updateSessionDocument()) { + log.warn("Session document did not update properly for session directory "+_session.sessionDir); + // cdocument.archive.cancelArchive(); // LATER: could be done - would need to prevent updateSessionDocument closing the archive. + _session.slog.warn("Session Document updated but may not be valid (false return from org.vamsas.simpleclient.ClientDocument.updateSessionDocument()"); + } + } + catch (IOException e) { + log.warn("IO Problems when updating document!",e); + _session.slog.error("IO problems when attempting to update document."); + } } // garbage collect the ClientDocument instance. try { diff --git a/src/org/vamsas/client/simpleclient/SimpleClientAppdata.java b/src/org/vamsas/client/simpleclient/SimpleClientAppdata.java index 5bda46e..a2d8912 100644 --- a/src/org/vamsas/client/simpleclient/SimpleClientAppdata.java +++ b/src/org/vamsas/client/simpleclient/SimpleClientAppdata.java @@ -3,6 +3,7 @@ */ package org.vamsas.client.simpleclient; +import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; @@ -10,6 +11,7 @@ import java.io.DataInput; import java.io.DataInputStream; import java.io.DataOutput; import java.io.DataOutputStream; +import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.util.Vector; @@ -339,6 +341,7 @@ public class SimpleClientAppdata implements IClientAppdata { } try { apdfile.lockFile(); + // LATER: Refactor these local AppDatastream IO stuff to their own class. JarOutputStream dstrm = new JarOutputStream( new BufferedOutputStream(new java.io.FileOutputStream(apdfile.sessionFile))); @@ -358,6 +361,45 @@ public class SimpleClientAppdata implements IClientAppdata { apdfile.unlockFile(); return null; } + /** + * copy data from the appData jar file to an appropriately + * referenced jar or Data entry for the given ApplicationData + * Assumes the JarFile is properly closed. + * @param vdoc session Document handler + * @param appd the AppData whose block is being updated + * @param apdjar the new data in a Jar written by this class + */ + protected void updateAnAppdataEntry(VamsasArchive vdoc, AppData appd, SessionFile apdjar) throws IOException { + if (apdjar==null || apdjar.sessionFile==null || !apdjar.sessionFile.exists()) { + throw new IOException("No temporary Appdata to recover and transfer."); + } + if (vdoc==null) { + log.fatal("FATAL! NO DOCUMENT TO WRITE TO!"); + throw new IOException("FATAL! NO DOCUMENT TO WRITE TO!"); + } + log.debug("Recovering AppData entry from "+apdjar.sessionFile); + JarInputStream istrm = new JarInputStream(new BufferedInputStream(new FileInputStream(apdjar.sessionFile))); + JarEntry je=null; + while (istrm.available()>0 && (je=istrm.getNextJarEntry())!=null && !je.getName().equals("appData_entry.dat")) { + if (je!=null) + log.debug("Ignoring extraneous entry "+je.getName()); + } + if (istrm.available()>0 && je!=null) { + log.debug("Found appData_entry.dat in Jar"); + String ref = appd.getDataReference(); + if (ref==null) { + throw new IOException("Null AppData.DataReference passed."); + } + if (vdoc.writeAppdataFromStream(ref, istrm)) { + log.debug("Entry updated successfully."); + } else { + throw new IOException("writeAppdataFromStream did not return true - expect future badness."); // LATER - verify why this might occur. + } + } else { + throw new IOException("Couldn't find appData_entry.dat in temporary jar file "+apdjar.sessionFile.getAbsolutePath()); + } + istrm.close(); + } /* (non-Javadoc) * @see org.vamsas.client.IClientAppdata#getClientOutputStream() */ @@ -466,16 +508,7 @@ public class SimpleClientAppdata implements IClientAppdata { newUserDataStream.close(); } } - /** - * copy data from the appData jar file to an appropriately - * referenced jar or Data entry for the given ApplicationData - * @param vdoc session Document handler - * @param appd the AppData whose block is being updated - * @param apdjar the new data in a Jar written by this class - */ - protected void updateAnAppdataEntry(VamsasArchive vdoc, AppData appd, SessionFile apdjar) { - - } + /** * -- 1.7.10.2