blob: ad135b627c8276ba99122db93d0558202ebc4313 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.syncope.core.provisioning.java.job.notification;
import java.time.OffsetDateTime;
import java.util.List;
import org.apache.commons.lang3.StringUtils;
import org.apache.syncope.common.lib.types.AuditElements;
import org.apache.syncope.common.lib.types.TaskType;
import org.apache.syncope.common.lib.types.TraceLevel;
import org.apache.syncope.core.persistence.api.dao.TaskDAO;
import org.apache.syncope.core.persistence.api.entity.task.NotificationTask;
import org.apache.syncope.core.persistence.api.entity.task.TaskExec;
import org.apache.syncope.core.persistence.api.entity.task.TaskUtilsFactory;
import org.apache.syncope.core.persistence.api.utils.ExceptionUtils2;
import org.apache.syncope.core.provisioning.api.AuditManager;
import org.apache.syncope.core.provisioning.api.event.JobStatusEvent;
import org.apache.syncope.core.provisioning.api.job.JobExecutionException;
import org.apache.syncope.core.provisioning.api.job.JobManager;
import org.apache.syncope.core.provisioning.api.notification.NotificationJobDelegate;
import org.apache.syncope.core.provisioning.api.notification.NotificationManager;
import org.apache.syncope.core.spring.security.AuthContextUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.transaction.annotation.Transactional;
public abstract class AbstractNotificationJobDelegate implements NotificationJobDelegate {
protected static final Logger LOG = LoggerFactory.getLogger(NotificationJobDelegate.class);
protected final TaskDAO taskDAO;
protected final TaskUtilsFactory taskUtilsFactory;
protected final AuditManager auditManager;
protected final NotificationManager notificationManager;
protected final ApplicationEventPublisher publisher;
protected AbstractNotificationJobDelegate(
final TaskDAO taskDAO,
final TaskUtilsFactory taskUtilsFactory,
final AuditManager auditManager,
final NotificationManager notificationManager,
final ApplicationEventPublisher publisher) {
this.taskDAO = taskDAO;
this.taskUtilsFactory = taskUtilsFactory;
this.auditManager = auditManager;
this.notificationManager = notificationManager;
this.publisher = publisher;
}
protected void setStatus(final String status) {
publisher.publishEvent(new JobStatusEvent(
this, AuthContextUtils.getDomain(), JobManager.NOTIFICATION_JOB, status));
}
protected abstract void notify(String to, NotificationTask task, TaskExec<NotificationTask> execution)
throws Exception;
@Transactional
@Override
public TaskExec<NotificationTask> executeSingle(final NotificationTask task, final String executor) {
TaskExec<NotificationTask> execution = taskUtilsFactory.getInstance(TaskType.NOTIFICATION).newTaskExec();
execution.setTask(task);
execution.setStart(OffsetDateTime.now());
execution.setExecutor(executor);
boolean retryPossible = true;
if (StringUtils.isBlank(task.getSubject()) || task.getRecipients().isEmpty()
|| StringUtils.isBlank(task.getHtmlBody()) || StringUtils.isBlank(task.getTextBody())) {
String message = "Could not fetch all required information for sending e-mails:\n"
+ task.getRecipients() + '\n'
+ task.getSender() + '\n'
+ task.getSubject() + '\n'
+ task.getHtmlBody() + '\n'
+ task.getTextBody();
LOG.error(message);
execution.setStatus(NotificationJob.Status.NOT_SENT.name());
retryPossible = false;
if (task.getTraceLevel().ordinal() >= TraceLevel.FAILURES.ordinal()) {
execution.setMessage(message);
}
} else {
LOG.debug("About to send notifications:\n{}\n{}\n{}\n{}\n{}",
task.getRecipients(),
task.getSender(),
task.getSubject(),
task.getHtmlBody(),
task.getTextBody());
setStatus("Sending notifications to " + task.getRecipients());
for (String to : task.getRecipients()) {
try {
notify(to, task, execution);
notificationManager.createTasks(
AuthContextUtils.getWho(),
AuditElements.EventCategoryType.TASK,
"notification",
null,
"send",
AuditElements.Result.SUCCESS,
null,
null,
task,
"Successfully sent notification to " + to);
} catch (Exception e) {
LOG.error("Could not send out notification", e);
execution.setStatus(NotificationJob.Status.NOT_SENT.name());
if (task.getTraceLevel().ordinal() >= TraceLevel.FAILURES.ordinal()) {
execution.setMessage(ExceptionUtils2.getFullStackTrace(e));
}
notificationManager.createTasks(
AuthContextUtils.getWho(),
AuditElements.EventCategoryType.TASK,
"notification",
null,
"send",
AuditElements.Result.FAILURE,
null,
null,
task,
"Could not send notification to " + to, e);
}
execution.setEnd(OffsetDateTime.now());
}
}
if (hasToBeRegistered(execution)) {
execution = notificationManager.storeExec(execution);
if (retryPossible
&& (NotificationJob.Status.valueOf(execution.getStatus()) == NotificationJob.Status.NOT_SENT)) {
handleRetries(execution);
}
} else {
notificationManager.setTaskExecuted(execution.getTask().getKey(), true);
}
return execution;
}
@Transactional
@Override
public void execute(final String executor) throws JobExecutionException {
List<NotificationTask> tasks = taskDAO.<NotificationTask>findToExec(TaskType.NOTIFICATION);
setStatus("Sending out " + tasks.size() + " notifications");
for (int i = 0; i < tasks.size(); i++) {
LOG.debug("Found notification task {} to be executed: starting...", tasks.get(i));
executeSingle(tasks.get(i), executor);
LOG.debug("Notification task {} executed", tasks.get(i));
}
}
protected static boolean hasToBeRegistered(final TaskExec<NotificationTask> execution) {
NotificationTask task = execution.getTask();
// True if either failed and failures have to be registered, or if ALL
// has to be registered.
return (NotificationJob.Status.valueOf(execution.getStatus()) == NotificationJob.Status.NOT_SENT
&& task.getTraceLevel().ordinal() >= TraceLevel.FAILURES.ordinal())
|| task.getTraceLevel() == TraceLevel.ALL;
}
protected void handleRetries(final TaskExec<NotificationTask> execution) {
if (notificationManager.getMaxRetries() <= 0) {
return;
}
long failedExecutionsCount = notificationManager.countExecutionsWithStatus(
execution.getTask().getKey(), NotificationJob.Status.NOT_SENT.name());
if (failedExecutionsCount <= notificationManager.getMaxRetries()) {
LOG.debug("Execution of notification task {} will be retried [{}/{}]",
execution.getTask(), failedExecutionsCount, notificationManager.getMaxRetries());
notificationManager.setTaskExecuted(execution.getTask().getKey(), false);
auditManager.audit(
AuthContextUtils.getWho(),
AuditElements.EventCategoryType.TASK,
"notification",
null,
"retry",
AuditElements.Result.SUCCESS,
null,
null,
execution,
"Notification task " + execution.getTask().getKey() + " will be retried");
} else {
LOG.error("Maximum number of retries reached for task {} - giving up", execution.getTask());
auditManager.audit(
AuthContextUtils.getWho(),
AuditElements.EventCategoryType.TASK,
"notification",
null,
"retry",
AuditElements.Result.FAILURE,
null,
null,
execution,
"Giving up retries on notification task " + execution.getTask().getKey());
}
}
}