/* * eXist Open Source Native XML Database * Copyright (C) 2001-2015 The eXist Project * http://exist-db.org * * This program 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 * 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 Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ package org.exist.scheduler; import java.io.IOException; import java.util.Map.Entry; import java.util.Optional; import java.util.Properties; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.exist.EXistException; import org.exist.dom.persistent.BinaryDocument; import org.exist.dom.persistent.DocumentImpl; import org.exist.security.PermissionDeniedException; import org.exist.security.Subject; import org.exist.source.DBSource; import org.exist.source.Source; import org.exist.source.SourceFactory; import org.exist.storage.BrokerPool; import org.exist.storage.DBBroker; import org.exist.storage.XQueryPool; import org.exist.storage.lock.Lock.LockMode; import org.exist.xmldb.XmldbURI; import org.exist.xquery.CompiledXQuery; import org.exist.xquery.XPathException; import org.exist.xquery.XQuery; import org.exist.xquery.XQueryContext; import org.exist.xquery.value.StringValue; import org.quartz.JobDataMap; import org.quartz.JobExecutionContext; import org.quartz.JobExecutionException; /** * Class to represent a User's XQuery Job Extends UserJob. * * @author Adam Retter <adam@exist-db.org> * @author Andrzej Taramina <andrzej@chaeron.com> */ public class UserXQueryJob extends UserJob { protected final static Logger LOG = LogManager.getLogger(UserXQueryJob.class); private final String DEFAULT_JOB_NAME_PREFIX = "XQuery"; private String name; private final String xqueryResource; private final Subject subject; /** * Default Constructor for Quartz. */ public UserXQueryJob(){ xqueryResource = null; subject = null; } /** * Constructor for Creating a new XQuery User Job. * * @param jobName The name of the job * @param xqueryResource The XQuery itself * @param subject The subject under which the xquery should be executed */ public UserXQueryJob(final String jobName, final String xqueryResource, final Subject subject) { this.xqueryResource = xqueryResource; this.subject = subject; if(jobName == null) { this.name = DEFAULT_JOB_NAME_PREFIX + ": " + xqueryResource; } else { this.name = jobName; } } @Override public final String getName() { return name ; } @Override public void setName(final String name) { this.name = name; } /** * Returns the XQuery Resource for this Job. * * @return The XQuery Resource for this Job */ public String getXQueryResource() { return xqueryResource; } /** * Returns the User for this Job. * * @return The User for this Job * @deprecated use getCurrentSubject method */ public Subject getUser() { return subject; } /** * Returns the subject for this Job. * * @return The subject for this Job */ public Subject getSubject() { return subject; } @Override public final void execute(final JobExecutionContext jec) throws JobExecutionException { final JobDataMap jobDataMap = jec.getJobDetail().getJobDataMap(); //TODO why are these values not used from the class members? final String xqueryresource = (String)jobDataMap.get(XQUERY_SOURCE); final Subject user = (Subject)jobDataMap.get(ACCOUNT); final BrokerPool pool = (BrokerPool)jobDataMap.get(DATABASE); final Properties params = (Properties)jobDataMap.get(PARAMS); final boolean unschedule = ((Boolean)jobDataMap.get(UNSCHEDULE)); //if invalid arguments then abort if((pool == null) || (xqueryresource == null) || (user == null)) { abort("BrokerPool or XQueryResource or User was null!"); } DocumentImpl resource = null; Source source = null; XQueryPool xqPool = null; CompiledXQuery compiled = null; XQueryContext context = null; try (final DBBroker broker = pool.get(Optional.of(user))) { if(xqueryresource.indexOf(':') > 0) { source = SourceFactory.getSource(broker, "", xqueryresource, true); } else { final XmldbURI pathUri = XmldbURI.create(xqueryresource); resource = broker.getXMLResource(pathUri, LockMode.READ_LOCK); if(resource != null) { source = new DBSource(broker, (BinaryDocument)resource, true); } } if(source != null) { //execute the xquery final XQuery xquery = pool.getXQueryService(); xqPool = pool.getXQueryPool(); //try and get a pre-compiled query from the pool compiled = xqPool.borrowCompiledXQuery(broker, source); if(compiled == null) { context = new XQueryContext(pool); } else { context = compiled.getContext(); } //TODO: don't hardcode this? if(resource != null) { context.setModuleLoadPath(XmldbURI.EMBEDDED_SERVER_URI.append(resource.getCollection().getURI()).toString()); context.setStaticallyKnownDocuments(new XmldbURI[] { resource.getCollection().getURI() }); } if(compiled == null) { try { compiled = xquery.compile(broker, context, source); } catch(final IOException e) { abort("Failed to read query from " + xqueryresource); } } //declare any parameters as external variables if(params != null) { String bindingPrefix = params.getProperty("bindingPrefix"); if(bindingPrefix == null) { bindingPrefix = "local"; } for(final Entry param : params.entrySet()) { final String key = (String)param.getKey(); final String value = (String)param.getValue(); context.declareVariable( bindingPrefix + ":" + key, new StringValue(value)); } } xquery.execute(broker, compiled, null); } else { LOG.warn("XQuery User Job not found: " + xqueryresource + ", job not scheduled"); } } catch(final EXistException ee) { abort("Could not get DBBroker!"); } catch(final PermissionDeniedException pde) { abort("Permission denied for the scheduling user: " + user.getName() + "!"); } catch(final XPathException xpe) { abort("XPathException in the Job: " + xpe.getMessage() + "!", unschedule); } catch(final IOException e) { abort("Could not load XQuery: " + e.getMessage()); } finally { if(context != null) { context.runCleanupTasks(); } //return the compiled query to the pool if(xqPool != null && source != null && compiled != null) { xqPool.returnCompiledXQuery(source, compiled); } //release the lock on the xquery resource if(resource != null) { resource.getUpdateLock().release(LockMode.READ_LOCK); } } } private void abort(final String message) throws JobExecutionException { abort(message, true); } private void abort(final String message, final boolean unschedule) throws JobExecutionException { final JobExecutionException jaa = new JobExecutionException( "UserXQueryJob Failed: " + message + (unschedule ? " Unscheduling UserXQueryJob." : ""), false ); //abort all triggers for this job if specified that we should unschedule the job jaa.setUnscheduleAllTriggers(unschedule); throw jaa; } }