/* * Copyright 2012 The Solmix Project * * This is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This software 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. * * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * http://www.gnu.org/licenses/ * or see the FSF site: http://www.fsf.org. */ package org.solmix.fmk.datasource; import java.io.IOException; import java.io.InputStream; import java.net.URL; import java.util.concurrent.atomic.AtomicLong; import javax.annotation.Resource; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.solmix.SlxConstants; import org.solmix.api.datasource.DSRequest; import org.solmix.api.datasource.DataSource; import org.solmix.api.datasource.DataSourceData; import org.solmix.api.datasource.ParserHandler; import org.solmix.api.event.IValidationEvent.Level; import org.solmix.api.exception.SlxException; import org.solmix.api.jaxb.EserverType; import org.solmix.api.jaxb.TdataSource; import org.solmix.api.jaxb.Tsolmix; import org.solmix.api.repo.DSRepository; import org.solmix.api.repo.DSRepositoryManager; import org.solmix.api.serialize.JSParser; import org.solmix.api.serialize.JSParserFactory; import org.solmix.api.serialize.XMLParser; import org.solmix.api.serialize.XMLParserFactory; import org.solmix.api.types.Texception; import org.solmix.api.types.Tmodule; import org.solmix.commons.io.SlxFile; import org.solmix.commons.util.DataUtils; import org.solmix.commons.util.IOUtils; import org.solmix.fmk.SlxContext; import org.solmix.fmk.event.EventWorker; import org.solmix.fmk.event.EventWorkerFactory; import org.solmix.fmk.serialize.JSParserFactoryImpl; import org.solmix.fmk.serialize.XMLParserFactoryImpl; import org.solmix.runtime.SystemContext; /** * @author solmix.f@gmail.com * @since 0.0.1 * @version 110035 2010-12-26 solmix-ds */ public class DefaultParser implements ParserHandler { private static final Logger log = LoggerFactory.getLogger(DefaultParser.class.getName()); public static final String INHERIT_KEY = "_inheritsForm"; public static final String DEFAULT_REPO = "default"; public static final String DEFAULT_REPO_SUFFIX = "ds"; public static final String GROUP_SEP = SlxConstants.GROUP_SEP; private EventWorker worker; protected final XMLParser xmlParser; private static AtomicLong numParsered; private SystemContext sc; public DefaultParser(final SystemContext sc) { numParsered = new AtomicLong(); setSystemContext(sc); xmlParser = XMLParserFactoryImpl.getInstance().get(); } @Resource public void setSystemContext(final SystemContext sc) { this.sc = sc; } protected synchronized JSParser getJSParser() { JSParserFactory factory = JSParserFactoryImpl.getInstance(); return factory.get(); } protected synchronized XMLParser getXMLParser() { XMLParserFactory factory = XMLParserFactoryImpl.getInstance(); return factory.get(); } /** * {@inheritDoc} * * @see org.solmix.api.datasource.ParserHandler#parser(java.lang.String) */ @Override public Object parser(String dsName) throws SlxException { return parser(DEFAULT_REPO, dsName, DEFAULT_REPO_SUFFIX, null); } /** * {@inheritDoc} * * @see org.solmix.api.datasource.ParserHandler#parser(java.lang.String, java.lang.String) */ @Override public Object parser(String repoName, String dsName) throws SlxException { return parser(repoName, dsName, DEFAULT_REPO_SUFFIX, null); } /** * {@inheritDoc} * * @throws SlxException * * @see org.solmix.api.datasource.ParserHandler#parser(java.lang.String, java.lang.String, * org.solmix.api.datasource.DSRequest) */ @Override public Object parser(String repoName, String dsName, DSRequest request) throws SlxException { return parser(repoName, dsName, DEFAULT_REPO_SUFFIX, request); } /** * {@inheritDoc} * * @see org.solmix.api.datasource.ParserHandler#parser(java.lang.String, java.lang.String, java.lang.String) */ @Override public Object parser(String repoName, String dsName, String suffix, DSRequest request) throws SlxException { DSRepositoryManager manager = sc.getExtension(org.solmix.api.repo.DSRepositoryManager.class); DSRepository[] repos = null; if (repoName != null) { repos = new DSRepository[1]; repos[0] = manager.getRepository(repoName); } else { repos = manager.getRepositories(); } String explicitName = DataUtils.isNullOrEmpty(suffix) ? dsName : dsName + "." + suffix.toLowerCase().trim(); for (DSRepository repo : repos) { DataSourceData loaded = loadFromRepo(repo, explicitName, dsName, request); if (loaded != null) return loaded; } return null; } /** * @param repo * @return * @throws SlxException */ private DataSourceData loadFromRepo(DSRepository repo, String explicitName, String dsName, DSRequest request) throws SlxException { long _$ = System.currentTimeMillis(); String __info = new StringBuilder().append(">>Parser datasource [").append(dsName).append("]").toString(); log.trace(__info); Object obj = repo.load(explicitName); DataSourceData data = null; // begin if (obj == null) return null; switch (repo.getObjectType()) { case SLX_FILE: data= parseFromSlxFile(SlxFile.class.cast(obj), dsName); break; case STREAM: data= parseFromStream(InputStream.class.cast(obj), dsName); break; case URL: data= parseFromURL(URL.class.cast(obj), dsName); break; default: break; } data.setRepositoryId(repo.getName()); preBuild(data, request); getThreadEventWork().createAndFireTimeEvent(System.currentTimeMillis() - _$, __info); return data; } private DataSourceData parseFromStream(InputStream is, String dsName) throws SlxException { DataSourceData data = null; Tsolmix module=null; try { module = xmlParser.unmarshalDS(is); } catch (Exception e1) { throw new SlxException(Tmodule.XML,Texception.XML_JAXB_UNMARSHAL,e1); } finally { IOUtils.closeQuitely(is); } if(module==null) return null; TdataSource td = module.getDataSource(); numParsered.incrementAndGet(); String ID = td.getID(); /************************************ * construct explicable ID with group name. ************************************/ if (ID != null && !ID.equals(dsName)) { td.setID(dsName); /************************************ * find real name. ************************************/ String realName = dsName.substring(dsName.lastIndexOf(GROUP_SEP) + 1); if (!realName.equals(ID)) { String info = (new StringBuilder()).append("dsName case sensitivity mismatch - looking for: ").append(dsName).append(", but got: ").append( ID).toString(); getThreadEventWork().createAndFireDSValidateEvent(Level.WARNING, info, new IllegalArgumentException()); } } // new datasource data. return new DataSourceData(td); } /** * @param cast * @return * @throws SlxException */ private DataSourceData parseFromURL(URL url, String dsName) throws SlxException { DataSourceData data = null; Tsolmix module=null; InputStream is = null; try { is = url.openStream(); module = xmlParser.unmarshalDS(is); } catch (Exception e1) { throw new SlxException(Tmodule.XML,Texception.XML_JAXB_UNMARSHAL,e1); } finally { IOUtils.closeQuitely(is); } if(module==null) return null; TdataSource td = module.getDataSource(); numParsered.incrementAndGet(); String ID = td.getID(); /************************************ * construct explicable ID with group name. ************************************/ if (ID != null && !ID.equals(dsName)) { td.setID(dsName); /************************************ * find real name. ************************************/ String realName = dsName.substring(dsName.lastIndexOf(GROUP_SEP) + 1); if (!realName.equals(ID)) { String info = (new StringBuilder()).append("dsName case sensitivity mismatch - looking for: ").append(dsName).append(", but got: ").append( ID).toString(); getThreadEventWork().createAndFireDSValidateEvent(Level.WARNING, info, new IllegalArgumentException()); } } try { // new datasource data. data = new DataSourceData(td); data.setUrlString(url.getPath()); data.setConfigTimestamp(url.openConnection().getLastModified()); } catch (IOException e) { throw new SlxException(Tmodule.DATASOURCE, Texception.DS_DSFILE_NOT_FOUND, null, e); } return data; } /** * @param cast * @return * @throws SlxException */ private DataSourceData parseFromSlxFile(SlxFile slx, String dsName) throws SlxException { DataSourceData data = null; Tsolmix module = xmlParser.unmarshalDS(slx); TdataSource td = module.getDataSource(); numParsered.incrementAndGet(); String ID = td.getID(); /************************************ * construct explicable ID with group name. ************************************/ if (ID != null && !ID.equals(dsName)) { td.setID(dsName); /************************************ * find real name. ************************************/ String realName = dsName.substring(dsName.lastIndexOf(GROUP_SEP) + 1); if (!realName.equals(ID)) { String info = (new StringBuilder()).append("dsName case sensitivity mismatch - looking for: ").append(dsName).append(", but got: ").append( ID).toString(); getThreadEventWork().createAndFireDSValidateEvent(Level.WARNING, info, new IllegalArgumentException()); } } try { // new datasource data. data = new DataSourceData(td); data.setUrlString(slx.getCanonicalPath()); data.setConfigTimestamp(slx.lastModified()); } catch (IOException e) { throw new SlxException(Tmodule.DATASOURCE, Texception.DS_DSFILE_NOT_FOUND, null, e); } return data; } public EventWorker getThreadEventWork() { if (worker == null) { EventWorkerFactory factory = EventWorkerFactory.getInstance(); worker = factory.createWorker(SlxContext.getThreadSystemContext()); } return worker; } protected DataSourceData preBuild(DataSourceData data, DSRequest request) throws SlxException { /********************************************************* * auto generating datasource schema. *********************************************************/ /* * if (DataUtils.booleanValue(data.getTdataSource().isAutoDeriveSchema())) autoGenerateSchema(); */ customerValidation(data, request); return data; } protected void customerValidation(DataSourceData data, DSRequest request) throws SlxException { TdataSource td = data.getTdataSource(); String dsID = data.getTdataSource().getID(); if (data.getName() == null) { getThreadEventWork().createAndFireDSValidateEvent(Level.WARNING, "Datasource configuration with no ID", null); } // data.addValidationEvent(new DSValidation(Level.WARNING, "Datasource configuration with no ID")); if (td.getServerType() == EserverType.SQL && data.getTdataSource().getTableName() == null) { data.getTdataSource().setTableName(dsID); String __info = "SQL DataSource with no set TableName try to use ID as tableName"; getThreadEventWork().createAndFireDSValidateEvent(Level.DEBUG, __info, null); // if (log.isDebugEnabled()) // log.debug(__info); } data.setTdataSource(td); } /** * <b><li>Manager Method:</b> * * @return the numParsered */ public static long getNumParsered() { return numParsered.longValue(); } /** * Only get the super datasource. */ protected void parserSuper(DataSourceData data, DSRequest dsRequest) { String _superDSName = data.getSuperDSName(); DataSource _superDS = data.getSuperDS(); String __vinfo; if (_superDSName != null && _superDS == null) { try { __vinfo = new StringBuilder().append("Looking up superDS of DataSource ").append(data.getName()).append(": '").append(_superDS).append( "'").toString(); getThreadEventWork().createAndFireDSValidateEvent(Level.DEBUG, __vinfo, null); // data.addValidationEvent(new DSValidation(Level.DEBUG, __vinfo)); _superDS = DefaultDataSourceManager.getDataSource(_superDSName, dsRequest); if (_superDS == null) { data.setSuperDSName(null); __vinfo = (new StringBuilder()).append("DataSource ").append(data.getName()).append(" declared to inherit from DataSource ").append( _superDSName).append(" which could not be loaded.set superDSName=null").toString(); getThreadEventWork().createAndFireDSValidateEvent(Level.DEBUG, __vinfo, null); // data.addValidationEvent(new DSValidation(Level.WARNING, __vinfo)); } else { data.setSuperDS(_superDS); } } catch (Exception e) { if (log.isWarnEnabled()) log.warn( (new StringBuilder()).append("Exception loading current DataSource [").append(data.getName()).append("]'s super ds [").append( _superDSName).append("]: ").toString(), e); } finally { if (_superDS != null) DefaultDataSourceManager.freeDataSource(_superDS); } } } }