All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.dirigible.engine.job.synchronizer.JobSynchronizer Maven / Gradle / Ivy

There is a newer version: 7.2.0
Show newest version
/*
 * Copyright (c) ${license.git.copyrightYears} SAP SE or an SAP affiliate company and Eclipse Dirigible contributors
 *
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v2.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v20.html
 *
 * SPDX-FileCopyrightText: ${license.git.copyrightYears} SAP SE or an SAP affiliate company and Eclipse Dirigible contributors
 * SPDX-License-Identifier: EPL-2.0
 */
package org.eclipse.dirigible.engine.job.synchronizer;

import static java.text.MessageFormat.format;

import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import javax.inject.Inject;
import javax.inject.Singleton;

import org.apache.commons.io.IOUtils;
import org.eclipse.dirigible.commons.api.module.StaticInjector;
import org.eclipse.dirigible.core.scheduler.api.AbstractSynchronizer;
import org.eclipse.dirigible.core.scheduler.api.ISchedulerCoreService;
import org.eclipse.dirigible.core.scheduler.api.SchedulerException;
import org.eclipse.dirigible.core.scheduler.api.SynchronizationException;
import org.eclipse.dirigible.core.scheduler.manager.SchedulerManager;
import org.eclipse.dirigible.core.scheduler.service.SchedulerCoreService;
import org.eclipse.dirigible.core.scheduler.service.definition.JobDefinition;
import org.eclipse.dirigible.repository.api.IResource;
import org.quartz.TriggerKey;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * The Class JobSynchronizer.
 */
@Singleton
public class JobSynchronizer extends AbstractSynchronizer {

	private static final Logger logger = LoggerFactory.getLogger(JobSynchronizer.class);

	private static final Map JOBS_PREDELIVERED = Collections.synchronizedMap(new HashMap());

	private static final List JOBS_SYNCHRONIZED = Collections.synchronizedList(new ArrayList());

	@Inject
	private SchedulerCoreService schedulerCoreService;

	// @Inject
	// private SchedulerManager schedulerManager;
	
	private final String SYNCHRONIZER_NAME = this.getClass().getCanonicalName();

	/**
	 * Force synchronization.
	 */
	public static final void forceSynchronization() {
		JobSynchronizer extensionsSynchronizer = StaticInjector.getInjector().getInstance(JobSynchronizer.class);
		extensionsSynchronizer.synchronize();
	}

	/**
	 * Register predelivered job.
	 *
	 * @param jobPath
	 *            the job path
	 * @throws IOException
	 *             Signals that an I/O exception has occurred.
	 */
	public void registerPredeliveredJob(String jobPath) throws IOException {
		InputStream in = JobSynchronizer.class.getResourceAsStream(jobPath);
		try {
			String json = IOUtils.toString(in, StandardCharsets.UTF_8);
			JobDefinition jobDefinition = schedulerCoreService.parseJob(json);
			jobDefinition.setName(jobPath);
			JOBS_PREDELIVERED.put(jobPath, jobDefinition);
		} finally {
			if (in != null) {
				in.close();
			}
		}
	}

	/*
	 * (non-Javadoc)
	 * @see org.eclipse.dirigible.core.scheduler.api.ISynchronizer#synchronize()
	 */
	@Override
	public void synchronize() {
		synchronized (JobSynchronizer.class) {
			logger.trace("Synchronizing Jobs...");
			try {
				startSynchronization(SYNCHRONIZER_NAME);
				clearCache();
				synchronizePredelivered();
				synchronizeRegistry();
				startJobs();
				int immutableCount = JOBS_PREDELIVERED.size();
				int mutableCount = JOBS_SYNCHRONIZED.size();
				cleanup();
				clearCache();
				successfulSynchronization(SYNCHRONIZER_NAME, format("Immutable: {0}, Mutable: {1}", immutableCount, mutableCount));
			} catch (Exception e) {
				logger.error("Synchronizing process for Jobs failed.", e);
				try {
					failedSynchronization(SYNCHRONIZER_NAME, e.getMessage());
				} catch (SchedulerException e1) {
					logger.error("Synchronizing process for Jobs files failed in registering the state log.", e);
				}
			}
			logger.trace("Done synchronizing Jobs.");
		}
	}

	/*
	 * (non-Javadoc)
	 * @see org.eclipse.dirigible.core.scheduler.api.AbstractSynchronizer#synchronizeRegistry()
	 */
	@Override
	protected void synchronizeRegistry() throws SynchronizationException {
		logger.trace("Synchronizing Jobs from Registry...");

		super.synchronizeRegistry();

		logger.trace("Done synchronizing Jobs from Registry.");
	}

	/*
	 * (non-Javadoc)
	 * @see org.eclipse.dirigible.core.scheduler.api.AbstractSynchronizer#synchronizeResource(org.eclipse.dirigible.
	 * repository.api.IResource)
	 */
	@Override
	protected void synchronizeResource(IResource resource) throws SynchronizationException {
		String resourceName = resource.getName();
		if (resourceName.endsWith(ISchedulerCoreService.FILE_EXTENSION_JOB)) {
			JobDefinition jobDefinition = schedulerCoreService.parseJob(resource.getContent());
			jobDefinition.setName(getRegistryPath(resource));
			synchronizeJob(jobDefinition);
		}

	}

	/*
	 * (non-Javadoc)
	 * @see org.eclipse.dirigible.core.scheduler.api.AbstractSynchronizer#cleanup()
	 */
	@Override
	protected void cleanup() throws SynchronizationException {
		logger.trace("Cleaning up Jobs...");
		super.cleanup();

		try {
			List jobDefinitions = schedulerCoreService.getJobs();
			for (JobDefinition jobDefinition : jobDefinitions) {
				if (!JOBS_SYNCHRONIZED.contains(jobDefinition.getName())) {
					schedulerCoreService.removeJob(jobDefinition.getName());
					logger.warn("Cleaned up Job [{}] from group: {}", jobDefinition.getName(), jobDefinition.getGroup());
				}
			}
		} catch (SchedulerException e) {
			throw new SynchronizationException(e);
		}

		logger.trace("Done cleaning up Jobs.");
	}

	private void startJobs() throws SchedulerException {
		logger.trace("Start Jobs...");

		for (String jobName : JOBS_SYNCHRONIZED) {
			if (!SchedulerManager.existsJob(jobName)) {
				try {
					JobDefinition jobDefinition = schedulerCoreService.getJob(jobName);
					SchedulerManager.scheduleJob(jobDefinition);
				} catch (SchedulerException e) {
					logger.error(e.getMessage(), e);
				}
			}
		}

		Set runningJobs = SchedulerManager.listJobs();
		for (TriggerKey jobKey : runningJobs) {
			try {
				if (!JOBS_SYNCHRONIZED.contains(jobKey.getName())) {
					SchedulerManager.unscheduleJob(jobKey.getName(), jobKey.getGroup());
				}
			} catch (SchedulerException e) {
				logger.error(e.getMessage(), e);
			}
		}

		logger.trace("Running Jobs: " + runningJobs.size());
		logger.trace("Done starting Jobs.");
	}

	private void clearCache() {
		JOBS_SYNCHRONIZED.clear();
	}

	private void synchronizePredelivered() throws SynchronizationException {
		logger.trace("Synchronizing predelivered Jobs...");
		// Jobs
		for (JobDefinition jobDefinition : JOBS_PREDELIVERED.values()) {
			synchronizeJob(jobDefinition);
		}
		logger.trace("Done synchronizing predelivered Jobs.");
	}

	private void synchronizeJob(JobDefinition jobDefinition) throws SynchronizationException {
		try {
			if (!schedulerCoreService.existsJob(jobDefinition.getName())) {
				schedulerCoreService.createJob(jobDefinition.getName(), jobDefinition.getGroup(), jobDefinition.getClazz(),
						jobDefinition.getHandler(), jobDefinition.getEngine(), jobDefinition.getDescription(), jobDefinition.getExpression(),
						jobDefinition.isSingleton());
				logger.info("Synchronized a new Job [{}] from group: {}", jobDefinition.getName(), jobDefinition.getGroup());
			} else {
				JobDefinition existing = schedulerCoreService.getJob(jobDefinition.getName());
				if (!jobDefinition.equals(existing)) {
					schedulerCoreService.updateJob(jobDefinition.getName(), jobDefinition.getGroup(), jobDefinition.getClazz(),
							jobDefinition.getHandler(), jobDefinition.getEngine(), jobDefinition.getDescription(), jobDefinition.getExpression(),
							jobDefinition.isSingleton());
					logger.info("Synchronized a modified Job [{}] from group: {}", jobDefinition.getName(), jobDefinition.getGroup());
				}
			}
			JOBS_SYNCHRONIZED.add(jobDefinition.getName());
		} catch (SchedulerException e) {
			throw new SynchronizationException(e);
		}
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy