Package org.olat.core.gui.media

Source Code of org.olat.core.gui.media.ClasspathMediaResource

/**
* OLAT - Online Learning and Training<br>
* http://www.olat.org
* <p>
* Licensed under the Apache License, Version 2.0 (the "License"); <br>
* you may not use this file except in compliance with the License.<br>
* You may obtain a copy of the License at
* <p>
* http://www.apache.org/licenses/LICENSE-2.0
* <p>
* Unless required by applicable law or agreed to in writing,<br>
* software distributed under the License is distributed on an "AS IS" BASIS, <br>
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br>
* See the License for the specific language governing permissions and <br>
* limitations under the License.
* <p>
* Copyright (c) 1999-2006 at Multimedia- & E-Learning Services (MELS),<br>
* University of Zurich, Switzerland.
* <p>
*/

package org.olat.core.gui.media;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.UnknownHostException;
import java.util.HashMap;
import java.util.Map;
import java.util.jar.JarFile;
import java.util.zip.ZipEntry;

import javax.servlet.http.HttpServletResponse;

import org.olat.core.helpers.Settings;
import org.olat.core.logging.LogDelegator;
import org.olat.core.logging.OLog;
import org.olat.core.logging.Tracing;
import org.olat.core.util.WebappHelper;

/**
* The ClasspathMediaResource delivers files directly from the classpath, e.g.
* to deliver a css file that is located in the _static directory in your
* package (src dir or from jar). Normally you just use the JsAndCSSComponent or
* the ClassPathStaticDispatcher
*
* @author Felix Jost
*/
public class ClasspathMediaResource extends LogDelegator implements MediaResource {
  private final String location;
  private Long lastModified;
  private Long size;
  private URL url;
  // local cache to minimize access to jar content (expensive)
  private static final Map<String,Long> cachedJarResourceLastModified = new HashMap<String, Long>();
  private static final Map<String,Long> cachedJarResourceSize = new HashMap<String, Long>();

  /**
   * Constructor that uses class loader of this (ClasspathMediaResource) class
   * @param pakkage the package name (e.g. org.olat.core)
   * @param location the relative file path (e.g. _static/my/file.css)
   */
  public ClasspathMediaResource(Package pakkage, String location) {
    this.location = location;   
    this.url = getClass().getResource("/" + pakkage.getName().replace(".", "/") + "/" + location);
    init(pakkage.getName());
  }
 
  /**
   * Constructor that uses class cloader of given class
   * @param clazz
   * @param location the relative file path (e.g. _static/my/file.css)
   */
  public ClasspathMediaResource(Class clazz, String location) {
    this.location = location;   
    this.url = clazz.getResource(location);
    String className = clazz.getName();
    int ls = className.lastIndexOf('.');
    // using baseClass.getPackage() would add unneeded inefficient and synchronized code
    String packageName = className.substring(0, ls);
    init(packageName);
  }

  /**
   * Internal helper to initialize everything for this file
   *
   * @param packageName
   */
  private void init(String packageName) {
    if (Settings.isDebuging()) {
      // When the brasato source code is attached, look in the brasato source
      // folder for the media resource
      String brasatoSrc = WebappHelper.getBrasatoSourcePath();
      if (brasatoSrc != null && packageName.startsWith("org.olat.core")) {
        String fileName = brasatoSrc + "/" + packageName.replace(".", "/") + "/" + location;
        File file = new File(fileName);
        if (file.exists()) {
          try {
            url = new URL("file", null, fileName);
          } catch (MalformedURLException e) {
            // nothing to do, null value checked later
          }
        }
      }
      // Try to get the last modified of the resource.
    }
    if (url != null) { // null in case of resources not found.     
      String fileName = url.getFile();
      if (url.getProtocol().equals("jar")){
        int pathDelim = fileName.indexOf("!");
        if (WebappHelper.getContextRoot().startsWith("/")) {
          // First lookup in cache
          size = cachedJarResourceSize.get(fileName);
          lastModified = cachedJarResourceLastModified.get(fileName);
          if (size == null || lastModified == null) {
            // Not found in cache, read from jar and add to cache
            // Unix style: path should look like //my/absolute/path
            String jarPath = "/" + fileName.substring(5, pathDelim);
            // Rel path must not start with "!/", remove it
            String relPath  = fileName.substring(pathDelim + 2);
            try {
              // Get last modified and file size form jar entry
              File jarFile = new File(jarPath);
              JarFile jar = new JarFile(jarFile);
              ZipEntry entry = jar.getEntry(relPath);
              if (entry == null) {
                logWarn("jar resource at location '"+location+"' and package " + packageName + " was not found, could not resolve entry relPath::" + relPath, null);
              } else {
                size = new Long(entry.getSize());
                // Last modified of jar - getTime on jar entry is not stable
                lastModified = Long.valueOf(jarFile.lastModified());             
                // Add to cache - no need to synchronize, just overwrite
                cachedJarResourceSize.put(fileName, size);
                cachedJarResourceLastModified.put(fileName, lastModified);
              }
            } catch (IOException e) {
              logWarn("jar resource at location '"+location+"' and package " + packageName + " was not found!", e);
            }       
          }         
        } else {
          // For windows ignore the jar size and last modified, it
          // works also without those parameters. Windows should not
          // be used in production
          this.lastModified = WebappHelper.getTimeOfServerStartup();         
        }
      } else {
        // Get last modified and file size
        File f = new File(fileName);
        long lm = f.lastModified();
        this.lastModified = (lm != 0? lm : WebappHelper.getTimeOfServerStartup());
        if (f.exists()) {
          size = new Long(f.length());
        }       
      }     
      if (isLogDebugEnabled()) {
        logDebug("resource found at URL::" + this.url
            + " for package::" + packageName + " and location::"
            + location + ", filesize::" + size + " lastModified::"
            + lastModified, null);
      }
    } else {
      // resource was not found
      logWarn("resource at location '"+location+"' and package " + packageName + " was not found!", null);
    }
  }
 

  /**
   * @see org.olat.core.gui.media.MediaResource#getContentType()
   */
  public String getContentType() {
    String mimeType = WebappHelper.getMimeType(location);
    if (mimeType == null) mimeType = "application/octet-stream";
    return mimeType;

  }

  /**
   * @see org.olat.core.gui.media.MediaResource#getSize()
   */
  public Long getSize() {
    if (size != null && size > 0) return size;
    else return null;
  }

  /**
   * @see org.olat.core.gui.media.MediaResource#getInputStream()
   */
  public InputStream getInputStream() {
    InputStream is = null;
    if (url == null) return null;
    try {
      is = url.openStream();
    } catch (UnknownHostException host) {
      //catch host exception here which can occur when brasato.src is wrongly configured
      logWarn("cannot get inputstream for url:"+url.toExternalForm()+"Unknown host which starts with windows path like \"C\" points to a wrong configured brasato.src in the build.properties file", null);
    } catch (IOException e) {
      logWarn("cannot get inputstream for url:"+url.toExternalForm(), null);
    }
    return is;
  }

  /**
   * @see org.olat.core.gui.media.MediaResource#getLastModified()
   */
  public Long getLastModified() {
    return lastModified;
  }

  /**
   * @see org.olat.core.gui.media.MediaResource#release()
   */
  public void release() {
  // void
  }

  /**
   * @see org.olat.core.gui.media.MediaResource#prepare(javax.servlet.http.HttpServletResponse)
   */
  public void prepare(HttpServletResponse hres) {
  // 
  }
 
  public String toString() {
    return "ClasspathMediaResource:"+url;
  }

  /**
   * @return true: resource exists and can be delivered; false: resource could
   *         not be found, delivery will fail
   */
  public boolean resourceExists() {
    if (url == null) return false;
    else return true;
  }
}
TOP

Related Classes of org.olat.core.gui.media.ClasspathMediaResource

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.