/*
Dateiverwaltung - File management in your browser
Copyright (C) 2017 Ulrich Hilger, http://uhilger.de
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
package de.uhilger.wbx;
import de.uhilger.wbx.data.Inhalt;
import java.io.File;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Iterator;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import javax.servlet.ServletContext;
/**
*
*/
public class WbxUtils {
private static final Logger logger = Logger.getLogger(WbxUtils.class.getName());
public static final String JNDI_CTX_NAME = "java:comp/env";
public static final String NOT_FOUND = " nicht gefunden";
public static final String NO_STRING = " ist kein String";
public static final String EMPTY_STRING = "";
public static final String WBX_FILE_BASE = "wbxFileBase";
public static final String WBX_PUB_DIR_NAME = "wbxPubDirName";
public static final String WBX_DEFAULT_PUB_DIR_NAME = "/www";
public static final String WBX_PUB_URL_NAME = "wbxPubUrlName";
public static final String WBX_DEFAULT_PUB_URL_NAME = "/data";
/**
* Bei der WebBox ist das Datenverzeichnis relativ zum Verzeichnis
* $CATALINA_BASE/webapps untergebracht.
* Die Abfrage ServletContext.getRealPath
* liefert das Verzeichnis des Context dieser Webanwendung, also
* $CATALINA_BASE/webapps/file-cms
* oder
* $WBX/sys/base/webapps/file-cms
*
* Unter Windows z.B.
* C:\Users\fred\Documents\srv\wbx\sys\base\webapps\file-cms
* Unter Linux oder Mac OS z.B.
* /home/fred/srv/wbx/sys/base/webapps/file-cms
*
* Das Datenverzeichis liegt dann auf
* $WBX/daten
*
* Mit dem Verzeichnis des Context dieser Webanwendung ist das
* Datenverzeichnis der WebBox hart kodierbar mit dieser Methode
*
* @return Verzeichnis 'daten' der WebBox
*/
public File getWbxDataDir() {
File file = getWbxDir();
file = new File(file, "daten/");
logger.fine("WebBox Datenbasis: " + file.getAbsolutePath());
return file;
}
public File getWbxDir() {
logger.fine("Catalina Base: " + System.getProperty("catalina.base"));
File catalinaBase = new File(System.getProperty("catalina.base"));
File wbxDir = catalinaBase.getParentFile().getParentFile();
return wbxDir;
}
public List collectFiles(String requestUrl, String contextPath,
String relativePath, int maxTiefe, int maxAnzahl, int length) {
Bild bild = new Bild();
//WbxUtils wu = new WbxUtils();
String basis = getJNDIParameter(WBX_FILE_BASE, WbxUtils.EMPTY_STRING);
if(basis.equals(WbxUtils.EMPTY_STRING)) {
basis = getWbxDataDir().getAbsolutePath();
}
String pubDirName = getJNDIParameter(WbxUtils.WBX_PUB_DIR_NAME, WbxUtils.WBX_DEFAULT_PUB_DIR_NAME);
String pubUrlName = getJNDIParameter(WbxUtils.WBX_PUB_URL_NAME, WbxUtils.WBX_DEFAULT_PUB_URL_NAME);
String relPath = relativePath.replace(pubUrlName, pubDirName);
String absPath = basis + relPath;
ArrayList beitraege = new ArrayList();
ArrayList files = new ArrayList<>();
collectFiles(new File(absPath), 0, beitraege, maxTiefe, maxAnzahl);
Iterator i = beitraege.iterator();
while(i.hasNext()) {
File beitrag = (File) i.next();
Inhalt cont = new Inhalt();
cont.setMimetype(bild.getMimeType(beitrag));
cont.setIsDirectory(beitrag.isDirectory());
cont.setIsHidden(beitrag.isHidden());
cont.setLastModified(beitrag.lastModified());
cont.setLength(beitrag.length());
if(length > 0) {
cont.setAbst(getFileContent(beitrag, length));
}
/*
den 'https://..'-Teil bis vor dem
ContextPath ermitteln
*/
//String requestUrl = getRequest().getRequestURL().toString();
//String contextPath = getRequest().getContextPath();
int pos = requestUrl.indexOf(contextPath);
/*
den Teil des Pfades ermitteln, der zwischen dem
ContextPath zum oeffentlichen Ordner und dem Dateiname
steht
*/
String absolutePath = beitrag.getAbsolutePath();
absolutePath = absolutePath.replace(beitrag.getName(), "");
absolutePath = absolutePath.replace(pubDirName, "");
String part = relativePath.replace(pubUrlName, "");
int pos2 = absolutePath.indexOf(part);
String mittelteil = absolutePath.substring(pos2);
mittelteil = mittelteil.replace(part, "");
cont.setBase(requestUrl.substring(0, pos));
cont.setUrl(/*requestUrl.substring(0, pos) + "/data" + */ mittelteil + beitrag.getName());
files.add(cont);
}
return files;
}
private String getFileContent(File file, int len) {
try {
StringBuffer readBuffer = new StringBuffer();
char[] buf = new char[1];
FileReader fr = new FileReader(file);
int bytesRead = fr.read(buf);
int read = 0;
while(bytesRead > -1 && read < len) {
read += bytesRead;
readBuffer.append(buf);
bytesRead = fr.read(buf);
}
readBuffer.append(buf);
fr.close();
logger.fine("read: " + read + ", readBuffer.len: " + readBuffer.length());
return readBuffer.toString();
} catch (Exception ex) {
logger.log(Level.SEVERE, ex.getMessage(), ex);
return EMPTY_STRING;
}
}
/**
* Diese Methode funktioniert nur, wenn entweder ein Ordner uebergeben
* wird, der keine Unterordner enthaelt wie zum Beispiel der Ordner 'neu'
* der Bildersammlung oder ein Ordner, dessen Unterordner
* nach dem Schema Jahr, Monat benannt sind wie bei einem Journal, das
* die Beitraege wie folgt enthaelt:
* Journal-Ordner
* 2018
* 12
* 11
* 10
* usw.
* 2017
* 12
* 11
* 10
* usw.
*
* @param out
* @param dir
* @param tiefe
* @param dateizaehler
*/
public void collectFiles(File dir, int tiefe, List beitraege, int maxTiefe, int maxBeitraege) {
logger.fine(dir.getAbsolutePath());
List dirs = new ArrayList();
List beitraegeHier = new ArrayList();
File[] files = dir.listFiles();
for(int i = 0; i < files.length; i++) {
if(files[i].isDirectory()) {
if(tiefe < maxTiefe) {
dirs.add(files[i]);
}
} else {
beitraegeHier.add(files[i]);
}
}
if(dirs.size() > 0) {
// hier zuvor die Verzeichnissse absteigend nach Name sortieren
Collections.sort(dirs, new Comparator() {
@Override
public int compare(File o1, File o2) {
return o2.getName().compareTo(o1.getName());
}
});
Iterator i = dirs.iterator();
while(i.hasNext() && beitraege.size() < maxBeitraege) {
collectFiles((File) i.next(), tiefe+1, beitraege, maxTiefe, maxBeitraege);
}
}
if(beitraegeHier.size() > 0) {
// hier zuvor die Liste der Beitraege dieses Ordners nach lastModified absteigend sortieren
// dann die neuesten in beitraege aufnehmen, bis die maximale Zahl gesuchter
// neuer Beitraege erreicht ist.
Collections.sort(beitraegeHier, new Comparator() {
@Override
public int compare(File o1, File o2) {
int ergebnis;
if(o1.lastModified() > o2.lastModified()) {
ergebnis = -1;
} else if(o2.lastModified() > o1.lastModified()) {
ergebnis = 1;
} else {
ergebnis = 0;
}
return ergebnis;
}
});
Iterator i = beitraegeHier.iterator();
while(i.hasNext() && beitraege.size() < maxBeitraege) {
File bf = (File) i.next();
String nm = bf.getName().toLowerCase();
if(nm.endsWith(".htmi") || nm.endsWith(".html") || nm.endsWith(".htm") ||
nm.endsWith(".jpg") || nm.endsWith(".jpeg") || nm.endsWith(".png") ||
nm.endsWith(".txt") || nm.endsWith(".md")) {
beitraege.add(bf);
}
}
}
}
public int getJNDIInt(String paramName, int defaultVal) {
String jndiStr = getJNDIParameter(paramName, Integer.toString(defaultVal));
try {
return Integer.parseInt(jndiStr);
} catch(NumberFormatException ex) {
logger.log(Level.FINE, ex.getMessage());
return defaultVal;
}
}
public String getJNDIParameter(String pname, String defaultVal) {
try {
// unseren environment naming context ermitteln
Context initCtx = new InitialContext();
Context envCtx = (Context) initCtx.lookup(JNDI_CTX_NAME);
// unseren Parameter lesen
Object o = envCtx.lookup(pname);
if(o instanceof String) {
return o.toString();
} else {
return defaultVal;
}
} catch (NamingException ex) {
logger.log(Level.FINE, ex.getMessage());
return defaultVal;
}
}
}