From: jprocter <jprocter@compbio.dundee.ac.uk>
Date: Fri, 20 Jan 2006 18:03:29 +0000 (+0000)
Subject: partially implemented simpleclient.EventGenerator
X-Git-Tag: Release_0.2~362
X-Git-Url: http://source.jalview.org/gitweb/?a=commitdiff_plain;h=a612f023ecf10c639a99540f79894e52a2151aba;p=vamsas.git

partially implemented simpleclient.EventGenerator

git-svn-id: https://svn.lifesci.dundee.ac.uk/svn/repository/trunk@158 be28352e-c001-0410-b1a7-c7978e42abec
---

diff --git a/src/org/vamsas/client/Events.java b/src/org/vamsas/client/Events.java
index fea1b79..00512a7 100644
--- a/src/org/vamsas/client/Events.java
+++ b/src/org/vamsas/client/Events.java
@@ -45,6 +45,8 @@ public class Events {
    * created. Any client that handles this should call the
    * IClient.getDocument(), update and then IClient.updateDocument in the same
    * handler thread.
+   * EventName: <Vamsas-session URN>
+   * NewValue: org.vamsas.client.IClient for session.
    */
   public static final String DOCUMENT_FINALIZEAPPDATA = "org.vamsas.client.events.DocumentFinalizeAppData";
 
diff --git a/src/org/vamsas/client/simpleclient/EventGeneratorThread.java b/src/org/vamsas/client/simpleclient/EventGeneratorThread.java
new file mode 100644
index 0000000..e60a674
--- /dev/null
+++ b/src/org/vamsas/client/simpleclient/EventGeneratorThread.java
@@ -0,0 +1,117 @@
+package org.vamsas.client.simpleclient;
+
+import java.beans.PropertyChangeEvent;
+import java.beans.PropertyChangeSupport;
+import java.util.Hashtable;
+
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+import org.vamsas.client.Events;
+
+/**
+ * monitors watcher objects and generates events.
+ */
+public class EventGeneratorThread extends Thread implements Runnable {
+  private static Log log = LogFactory.getLog(EventGeneratorThread.class);
+  private SimpleClient client;
+  private Hashtable handlers; // manager object
+  private VamsasSession session;
+  
+  protected FileWatcher clientfile=null;
+  protected FileWatcher vamsasfile=null;
+  protected FileWatcher storeFile=null;
+  
+  private boolean watch=false;
+  
+  
+  EventGeneratorThread(VamsasSession s, SimpleClient _client, Hashtable eventhandlers) {
+    if (eventhandlers==null || s==null || _client==null)
+      throw new Error("Null arguments to EventGeneratorThread constructor.");
+    handlers = eventhandlers;
+    session = s;
+    client = _client;
+    setName(s.sessionDir.getName());
+    initWatchers();
+  }
+  
+  private void initWatchers() {
+    if (clientfile==null)
+      clientfile = session.getClientWatcher();
+    if (vamsasfile ==null)
+      vamsasfile = session.getDocWatcher();
+    if (storeFile == null)
+      storeFile = session.getStoreWatcher();
+    clientfile.setState();
+    vamsasfile.setState();
+    storeFile.setState();
+  }
+  boolean ownsf = false;
+  /**
+   * scans all watchers and fires changeEvents if necessary
+   * @return number of events generated.
+   */
+  private int checkforEvents() {
+    Lock watchlock;
+    //TODO : leave slog.info messages for the events that occur.
+    int raised=0;
+    // could make this general - but for now keep simple
+    if ((watchlock=storeFile.getChangedState())!=null) {
+      // TODO: define the storeFile semaphore mechanism : file exists - all clients inform their apps, and then the client that wrote the file should delete the file (it should hold the lock to it).
+      if (storeFile.exists) { 
+        PropertyChangeSupport h = (PropertyChangeSupport) handlers.get(Events.DOCUMENT_FINALIZEAPPDATA);
+        if (h!=null) {
+          log.debug("Triggering DOCUMENT_FINALIZEAPPDATA");
+          raised++;
+          h.firePropertyChange(client.getSessionUrn(), null, client);
+          // expect client to 
+          vamsasfile.setState();
+        }
+      }
+    }
+    if ((watchlock=clientfile.getChangedState())!=null) {
+      // see what happened to the clientfile - compare our internal version with the one in the file, or just send the updated list out...?
+      raised++;
+    }
+    if ((watchlock=vamsasfile.getChangedState())!=null) {
+      // pass IClientDocument instance to app handler ?
+    }
+    return raised;
+  }
+  
+  private void initEvents() {
+    
+  }
+  
+  /**
+   * probably don't need any of these below.
+   */
+  /* (non-Javadoc)
+   * @see java.lang.Thread#destroy()
+   */
+  public void destroy() {
+    super.destroy();
+  }
+  /* (non-Javadoc)
+   * @see java.lang.Thread#interrupt()
+   */
+  public void interrupt() {
+    // TODO Auto-generated method stub
+    super.interrupt();
+  }
+  /* (non-Javadoc)
+   * @see java.lang.Thread#isInterrupted()
+   */
+  public boolean isInterrupted() {
+    // TODO Auto-generated method stub
+    return super.isInterrupted();
+  }
+  /* (non-Javadoc)
+   * @see java.lang.Thread#run()
+   */
+  public void run() {
+    // TODO Auto-generated method stub
+    super.run();
+  }
+  
+
+}
diff --git a/src/org/vamsas/client/simpleclient/VamsasFile.java b/src/org/vamsas/client/simpleclient/VamsasFile.java
index 6d5f13c..ffb1485 100644
--- a/src/org/vamsas/client/simpleclient/VamsasFile.java
+++ b/src/org/vamsas/client/simpleclient/VamsasFile.java
@@ -59,30 +59,25 @@ public class VamsasFile extends SessionFile {
     
   }
   */
-  
   /**
-   * gets a locked Reader for the vamsas document.
-   * @return reader for vamsasdocument.xml enrty
+   * public interface for getting a lock.
+   * The lock object is internally referenced 
+   * so the lock will persist even after the
+   * return value of the method goes out of scope.
+   * @return null if lock couldn't be got or a valid Lock object.
    */
   public Lock getLock() {
-    lockFile();
-    return fileLock;
+    if (lockFile())
+      return fileLock;
+    return null;
   }
+  /**
+   * explicitly unlocks vamsas file.
+   * if you have called getLock() you *must* 
+   * call this to release the lock.
+   */
   public void unLock() {
     this.unlockFile();
   }
-  public java.io.Reader getDocumentReader() {
-    
-    try {
-      JarFile session = new JarFile("vamsasJar");
-      //JarEntry vamsasDocument = session.getJarEntry("vamsasDocument.xml");
-      //return new InputStreamReader(session.getInputStream(vamsasDocument));
-    } catch (IOException e) {
-      // TODO Auto-generated catch block
-      e.printStackTrace();
-    }
-    return null;
-  }
-  
   
 }
diff --git a/src/org/vamsas/client/simpleclient/VamsasSession.java b/src/org/vamsas/client/simpleclient/VamsasSession.java
index ef76d71..242cf36 100644
--- a/src/org/vamsas/client/simpleclient/VamsasSession.java
+++ b/src/org/vamsas/client/simpleclient/VamsasSession.java
@@ -1,6 +1,13 @@
 package org.vamsas.client.simpleclient;
 
 import java.io.File;
+import java.io.IOException;
+
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+import org.apache.log4j.Appender;
+import org.apache.log4j.Logger;
+import org.apache.log4j.FileAppender;
 /**
  * 
  * Vamsas client is intialised with a path to create live session directories. 
@@ -41,11 +48,133 @@ import java.io.File;
 
 public class VamsasSession {
   /**
-   * Holds the file handlers for a session.
+   * Holds the file handlers and watchers for a session.
+   */
+  /**
+   * TODO: make sessionDir signature to allow VamsasSession instances to recognise the form of the directory
+   */
+  /**
+   * indicator file for informing other processes that 
+   * they should finalise their vamsas datasets for 
+   * storing in a vamsas archive.
+   */
+  public static final String CLOSEANDSAVE_FILE="exiting";
+  /**
+   * log file location
+   */
+  public static final String SESSION_LOG="Log.txt";
+  private static Log log = LogFactory.getLog(VamsasSession.class);
+  protected Logger slog = Logger.getLogger("org.vamsas.client.SessionLog");
+  /**
+   * setup the sessionLog using Log4j.
+   * @throws IOException
+   */
+  private void initLog() throws IOException {
+    Appender app = slog.getAppender("SESSION_LOG");
+    slog.addAppender(new FileAppender(app.getLayout(), new File(sessionDir, SESSION_LOG).getAbsolutePath()));
+  }
+  
+  /**
+   * the sessionDir is given as the session location for new clients.
    */
   File sessionDir;
+  /**
+   * holds the list of attached clients
+   */
   ClientsFile clist;
-  File vamArchive;
+  public static final String CLIENT_LIST="Clients.obj";
+  /**
+   * holds the data
+   */
+  VamsasFile vamArchive; 
+  public static final String VAMSAS_OBJ="VamDoc.jar";
   
+  /**
+   * sets up the vamsas session files and watchers in sessionDir
+   * @param sessionDir
+   */
+  protected VamsasSession(File sessionDir) throws IOException {
+    if (sessionDir==null)
+      throw new Error("Null directory for VamsasSession.");
+    if (sessionDir.exists()) {
+      if (!sessionDir.isDirectory() || !sessionDir.canWrite() || sessionDir.canRead())
+        throw new IOException("Cannot access '"+sessionDir+"' as a read/writable Directory.");
+      if (checkSessionFiles(sessionDir)) {
+        // session files exist in the directory
+        this.sessionDir = sessionDir;
+        initSessionObjects();
+        slog.debug("Initialising additional VamsasSession instance");
+        log.debug("Attached to VamsasSession in "+sessionDir);
+      } 
+    } else {
+      // start from scratch
+      if (!sessionDir.mkdir())
+        throw new IOException("Failed to make VamsasSession directory in "+sessionDir);
+      this.sessionDir = sessionDir; 
+      createSessionFiles();
+      initSessionObjects();
+      slog.debug("Session directory created.");
+      log.debug("Initialised VamsasSession in "+sessionDir);
+    }
+  }
+  /**
+   * tests presence of existing sessionfiles files in dir
+   * @param dir
+   * @return
+   */
+  private boolean checkSessionFiles(File dir) throws IOException {
+    File c_file = new File(dir,CLIENT_LIST);
+    File v_doc = new File(dir,VAMSAS_OBJ);
+    if (c_file.exists() && v_doc.exists())
+      return true;
+    return false;
+  }
+  /**
+   * create new empty files in dir
+   *
+   */
+  private void createSessionFiles() throws IOException {
+    if (sessionDir==null)
+      throw new IOException("Invalid call to createSessionFiles() with null sessionDir");
+    File c_file = new File(sessionDir,CLIENT_LIST);
+    File v_doc = new File(sessionDir,VAMSAS_OBJ);
+    if (c_file.createNewFile())
+      log.debug("Created new ClientFile "+c_file); // don't care if this works or not
+    if (v_doc.createNewFile())
+      log.debug("Created new Vamsas Session Document File "+v_doc); 
+  }
+  /**
+   * construct SessionFile objects and watchers for each
+   */
+  private void initSessionObjects() throws IOException {
+    if (clist!=null || vamArchive!=null)
+      throw new IOException("initSessionObjects called for initialised VamsasSession object.");
+    clist = new ClientsFile(new File(sessionDir,CLIENT_LIST));
+    vamArchive = new VamsasFile(new File(sessionDir,VAMSAS_OBJ));
+  }
+  /**
+   * make a new watcher object for the clientFile
+   * @return new ClientFile watcher instance
+   */
+  public FileWatcher getClientWatcher() {
+    return new FileWatcher(clist.sessionFile);
+  }
+  /**
+   * make a new watcher object for the vamsas Document
+   * @return new ClientFile watcher instance
+   */
+  public FileWatcher getDocWatcher() {
+    return new FileWatcher(vamArchive.sessionFile);
+  }
+
+  /**
+   * make a new watcher object for the messages file
+   * @return new watcher instance
+   */
+  public FileWatcher getStoreWatcher() {
+    return new FileWatcher(new File(CLOSEANDSAVE_FILE));
+  }
   
 }
+
+