Dateiverwaltung für die WebBox
ulrich
2021-01-21 8cab6e94514c38151b2e0c53c9df47c6e1682e28
src/java/de/uhilger/filecms/web/Initialiser.java
@@ -18,104 +18,149 @@
package de.uhilger.filecms.web;
import de.uhilger.filecms.api.FileMgr;
import de.uhilger.filesystem.FileRef;
import de.uhilger.wbx.WbxUtils;
import java.io.File;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import org.apache.commons.io.FileUtils;
/**
 *
 * @author ulrich
 * Initialisieren der Dateiverwaltung
 */
/*
    der folgende Eintrag kann z.B. in $CATALINA_BASE/conf/context.xml
    eingetragen werden
    <Environment
      name="wbxFileBase"
      type="java.lang.String"
      value="/media/wbx-data/cms"
      override="false"
    />
*/
public class Initialiser implements ServletContextListener {
  
  private static final Logger logger = Logger.getLogger(Initialiser.class.getName());
  
  public static final String FILE_BASE = "filebase";
  // http://localhost:8097/file-cms/rpc/de.uhilger.filecms.api.FileMgr/getBase/JSONNICE
  private FileRef getBase() {
    FileRef ref = null;
    //String fileBase = getServletContext().getInitParameter(FILE_BASE);
    // -Dfilecms.base=/pfad/zu/daten
    //String fileBase = System.getProperty("filecms.base");
    //File file = new File(fileBase);
    //logger.info(file.getAbsolutePath());
    //logger.info(getWebappsDir().getAbsolutePath());
  public static final String DATENABLAGE = "datenAblage";
  public static final String WBX_FILE_BASE = "wbxFileBase";
  public static final String WBX_DATA_DIR = "wbx.data";
  public static final String WBX_DEPTH = "tiefe";
  public static final String WBX_DATA_FOLDER = "datenOrdner";
    
    /*
        file = new File("."); liefert
        /home/ulrich/dev/lib/java/tomcat/tomcat2-8.5.9/bin/.
  /**
   * Die Dateiablage wird entweder auf einen absoluten Pfad gesetzt,
   * der im Deployment Descriptor hinterlegt werden kann oder, wenn
   * dort nichts eingetragen ist, auf den hart kodierten Pfad
   * der WebBox.
   *
   * @param ctx der ServletContext, in den die Angabe eingetragen wird. Dort
   * ist anschliessend die Angabe unter Initialiser.FILE_BASE abrufbar
   */
  protected void ablageErmitteln(ServletContext ctx) {
    //Object o = ctx.getInitParameter(DATENABLAGE);
    
        ..auf der WebBox aber
        /home/ulrich/srv/wbx_probe/.
        ..weil das Startskript dort liegt
    //File wbxDaten = getDataDir(ctx, 2, "data/");
    
        der Tomcat der WebBox ist unter
        sys/jrs/tomcat/bin
        also z.B.
        /home/ulrich/srv/wbx_probe/sys/jrs/tomcat/bin
        das Datenverzeichnis ist z.B. auf
        /home/ulrich/srv/wbx_probe/daten
        dann ist das Datenverzeichnis unter
        ../../../daten
        Der Ausdruck file = new File("."); liefert stets den
        Ort von dem aus der Java-Prozess gestartet wurde.
        Die unten folgende Bestimmung des Datenverzeichnisses
        ist beschraenkt auf das Datenverzeichnis der WebBox,
        entweder relativ zum Startskript der WebBox oder
        dem Startskript von Tomcat, wie es aus Netbeans heraus
        waehrend der Entwicklung genutzt wird.
        Besser ware vielleicht eine Bestimmung ueber einen
        Systemparameter -Dfilecms.base=... wie weiter oben
        auskommentiert. Damit liesse sich das file-cms auch
        ohne WebBox einsetzen. Allerdings muss dann das
        Datenverzeichnis im Start-Skript gebildet werden,
        also ausserhalb von Java, wenn es dynamisch aus
        einem Pfad relativ zum Start-Skript erzeugt werden
        soll.
    */
    File file = new File(".");
    logger.finer(file.getAbsolutePath());
    String path = file.getAbsolutePath();
    path = path.substring(0, path.length() - 1);
    file = new File(path);
    if(path.endsWith("bin")) {
      file = file.getParentFile().getParentFile().getParentFile();
    } else {
    //WbxUtils wu = new WbxUtils();
    //Object o = wu.getJNDIParameter(WBX_FILE_BASE, WbxUtils.EMPTY_STRING);
    String dir = System.getProperty(WBX_DATA_DIR);
    try {
      if(dir != null) {
        //String pfad = o.toString();
        if(dir.trim().length() > 0) {
          ctx.setAttribute(FILE_BASE, dir);
          logger.log(Level.INFO, "Daten-Ordner aus wbx.data: {0}", dir);
        } else {
          ctx.setAttribute(FILE_BASE, getDefaultDataDir(ctx).getAbsolutePath());
        }
      } else {
        ctx.setAttribute(FILE_BASE, getDefaultDataDir(ctx).getAbsolutePath());
      }
    } catch(Exception ex) {
      ctx.setAttribute(FILE_BASE, getDefaultDataDir(ctx).getAbsolutePath());
    }
    file = new File(file, "daten/");
    ref = new FileRef(file.getAbsolutePath(), file.isDirectory());
    logger.finer(ref.getAbsolutePath());
    return ref;
  }
  private File getDefaultDataDir(ServletContext ctx) {
    String tiefe = ctx.getInitParameter(WBX_DEPTH);
    String ordnerName = ctx.getInitParameter(WBX_DATA_FOLDER);
    return getDataDir(ctx, Integer.parseInt(tiefe), ordnerName);
  }
  
  /* ----- ServletContextListener Implementation ----- */
  /**
   * Daten-Ordner relativ zum Ablageort des file-cms finden.
   *
   * Die Standard-Struktur einer WebBox sieht folgende Ordner vor
   *
   * wbx/apps
   * wbx/data
   *
   * Liegt das file-cms z.B. im Ordner /home/ulrich/dev/srv/wbx-3/apps/file-cms
   * kann von dort aus der Daten-Ordner mit
   * getDataDir(ctx, 2, "data/")
   * gefunden werden.
   *
   * @param ctx  der ServletContext, in dem das file-cms laeuft
   * @param tiefe Anzahl Ordner, die der Ablageort des file-cms vom
   * Eltern-Ordner entfernt ist
   * @param pfad der Pfad zum Daten-Ordner ausgehend vom Eltern-Ordner
   * @return Daten-Ordner
   */
  private File getDataDir(ServletContext ctx, int tiefe, String pfad) {
    File appsOrdner = new File(ctx.getRealPath("/"));
    logger.log(Level.INFO, "Apps-Ordner: {0}", appsOrdner.getAbsolutePath());
    File wbx = new File(appsOrdner, "");
    for(int i = 0; i < tiefe; i++) {
      wbx = wbx.getParentFile();
    }
    logger.log(Level.INFO, "Wbx-Ordner: {0}", wbx.getAbsolutePath());
    File daten = new File(wbx, pfad);
    logger.log(Level.INFO, "Daten-Ordner: {0}", daten.getAbsolutePath());
    return daten;
  }
  protected void ablageInitialisieren(ServletContext ctx) {
    Object o = ctx.getAttribute(FILE_BASE);
    if(o instanceof String) {
      String targetDirName = (String) o;
      File dataDir = new File(targetDirName, "www");
      if(!dataDir.exists()) {
        dataDir.mkdirs();
        String srcPath = ctx.getRealPath("/"); // file-cms in webapps
        File srcDir = new File(srcPath, "/META-INF/daten/www");
        dataDir = new File(targetDirName);
        try {
          FileUtils.copyDirectoryToDirectory(srcDir, dataDir);
          srcDir = new File(srcPath, "/META-INF/daten/home");
          FileUtils.copyDirectoryToDirectory(srcDir, dataDir);
          srcDir = new File(srcPath, "/META-INF/daten/dav");
          FileUtils.copyDirectoryToDirectory(srcDir, dataDir);
        } catch (IOException ex) {
          logger.log(Level.SEVERE, null, ex);
        }
      }
    }
  }
  @Override
  public void contextInitialized(ServletContextEvent sce) {
    // hier kann etwas initialisiert werden
    ServletContext ctx = sce.getServletContext();
    ctx.setAttribute(FILE_BASE, getBase());
    ablageErmitteln(ctx);
    ablageInitialisieren(ctx);
  }
  @Override
  public void contextDestroyed(ServletContextEvent sce) {
    // hier wird alles wieder aufgeraeumt
    ServletContext ctx = sce.getServletContext();
    ctx.removeAttribute(FILE_BASE);
  }
}