/* * (C) Copyright 2006-2008 Nuxeo SAS (http://nuxeo.com/) and contributors. * * All rights reserved. This program and the accompanying materials * are made available under the terms of the GNU Lesser General Public License * (LGPL) version 2.1 which accompanies this distribution, and is available at * http://www.gnu.org/licenses/lgpl.html * * This library 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 * Lesser General Public License for more details. * * Contributors: * bstefanescu * * $Id$ */ package org.nuxeo.ecm.webengine.model.impl; import java.io.BufferedWriter; import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.io.Writer; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.nuxeo.common.utils.FileUtils; import org.nuxeo.ecm.webengine.WebEngine; import org.nuxeo.ecm.webengine.WebException; import org.nuxeo.ecm.webengine.loader.ClassProxy; import org.nuxeo.ecm.webengine.loader.WebLoader; import org.nuxeo.ecm.webengine.model.WebAdapter; import org.nuxeo.ecm.webengine.model.WebObject; /** * Load web types extracted from Groovy source files. Types are cached in * META-INF/groovy-web-types. When types are reloaded this file will be removed. * * @author <a href="mailto:bs@nuxeo.com">Bogdan Stefanescu</a> * */ public class GroovyTypeLoader { public static final Log log = LogFactory.getLog(GroovyTypeLoader.class); public static final String CRLF = System.getProperty("line.separator"); public static final String WEB_TYPES_FILE = "META-INF/groovy-web-types"; protected final WebLoader loader; protected final TypeRegistry typeReg; protected final File root; public GroovyTypeLoader(WebEngine engine, TypeRegistry typeReg, File root) { this.typeReg = typeReg; this.root = root; loader = engine.getWebLoader(); } public synchronized void flushCache() { log.info("Flush directory type provider cache"); File cache = new File(root, WEB_TYPES_FILE); cache.delete(); } public synchronized void load() { try { File cache = new File(root, WEB_TYPES_FILE); if (cache.isFile()) { for (String line : FileUtils.readLines(cache)) { if (line.equals("")) { continue; } TypeDescriptor td = loadType(line); if (td != null) { typeReg.registerTypeDescriptor(td); } } } else { cache.getParentFile().mkdirs(); Writer w = new BufferedWriter(new FileWriter(cache)); try { scan(root, null, w); w.close(); } catch (Throwable t) { w.close(); cache.delete(); throw WebException.wrap(t); } } } catch (Exception e) { throw WebException.wrap(e); } } protected void scan(File root, String path, Writer cache) { for (File file : root.listFiles()) { String name = file.getName(); if (file.isDirectory() && !"skin".equals(name) && !"samples".equals(name)) { scan(file, path == null ? name : new StringBuilder().append(path).append('.').append(name).toString(), cache); } else if (name.endsWith(".groovy") && Character.isUpperCase(name.charAt(0))) { String className = null; if (path == null) { className = name.substring(0, name.length()-7); } else { StringBuilder buf = new StringBuilder().append(path).append('.').append(name); buf.setLength(buf.length()-7); className = buf.toString(); } try { TypeDescriptor td = loadTypeAndRecord(cache, className); if (td != null) { typeReg.registerTypeDescriptor(td); } } catch (Exception e) { throw WebException.wrap(e); } } } } /** * Loads a type and cache it. * * @param cache * @param className * @throws ClassNotFoundException * @throws IOException */ protected TypeDescriptor loadTypeAndRecord(Writer cache, String className) throws ClassNotFoundException, IOException { TypeDescriptor td = loadType(className); if (td != null) { cache.write(className); cache.write(CRLF); } return td; } /** * Gets a type descriptor given an absolute className. * <p> * If this class doesn't define a type or type adapter, return null. */ protected TypeDescriptor loadType(String className) throws ClassNotFoundException { ClassProxy clazz = loader.getGroovyClassProxy(className); WebObject type = clazz.get().getAnnotation(WebObject.class); if (type != null) { return TypeDescriptor.fromAnnotation(clazz, type); } WebAdapter ws = clazz.get().getAnnotation(WebAdapter.class); if (ws != null) { return AdapterDescriptor.fromAnnotation(clazz, ws); } return null; } }