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

io.gravitee.am.gateway.handler.common.email.impl.EmailManagerImpl Maven / Gradle / Ivy

/**
 * Copyright (C) 2015 The Gravitee team (http://gravitee.io)
 *
 * Licensed 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 io.gravitee.am.gateway.handler.common.email.impl;

import freemarker.cache.StringTemplateLoader;
import freemarker.template.Configuration;
import io.gravitee.am.common.event.EmailEvent;
import io.gravitee.am.common.event.EventManager;
import io.gravitee.am.gateway.handler.common.email.EmailManager;
import io.gravitee.am.model.Domain;
import io.gravitee.am.model.Email;
import io.gravitee.am.model.ReferenceType;
import io.gravitee.am.model.common.event.Payload;
import io.gravitee.am.repository.management.api.EmailRepository;
import io.gravitee.common.event.Event;
import io.gravitee.common.event.EventListener;
import io.gravitee.common.service.AbstractService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.regex.Pattern;

import static java.lang.String.format;

/**
 * @author Titouan COMPIEGNE (titouan.compiegne at graviteesource.com)
 * @author GraviteeSource Team
 */
public class EmailManagerImpl extends AbstractService implements EmailManager, InitializingBean, EventListener {

    private static final Logger logger = LoggerFactory.getLogger(EmailManagerImpl.class);
    private static final String TEMPLATE_SUFFIX = ".html";
    private ConcurrentMap emails = new ConcurrentHashMap<>();
    private ConcurrentMap emailTemplates = new ConcurrentHashMap<>();

    @Autowired
    private EmailRepository emailRepository;

    @Autowired
    private Domain domain;

    @Autowired
    private EventManager eventManager;

    @Autowired
    private Configuration configuration;

    @Autowired
    private StringTemplateLoader templateLoader;

    @Value("${email.subject:[Gravitee.io] %s}")
    private String subject;

    @Value("${email.from:#{null}}")
    private String defaultFrom;

    @Override
    public void afterPropertiesSet() {
        logger.info("Initializing emails for domain {}", domain.getName());
        emailRepository.findAll(ReferenceType.DOMAIN, domain.getId())
                .subscribe(
                        email -> updateEmail(email),
                        error -> logger.error("Unable to initialize emails for domain {}", domain.getName(), error));
    }

    @Override
    protected void doStart() throws Exception {
        super.doStart();

        logger.info("Register event listener for email events for domain {}", domain.getName());
        eventManager.subscribeForEvents(this, EmailEvent.class, domain.getId());
    }

    @Override
    protected void doStop() throws Exception {
        super.doStop();

        logger.info("Dispose event listener for email events for domain {}", domain.getName());
        eventManager.unsubscribeForEvents(this, EmailEvent.class, domain.getId());
    }

    @Override
    public void onEvent(Event event) {
        if (event.content().getReferenceType() == ReferenceType.DOMAIN && domain.getId().equals(event.content().getReferenceId())) {
            switch (event.type()) {
                case DEPLOY:
                case UPDATE:
                    updateEmail(event.content().getId(), event.type());
                    break;
                case UNDEPLOY:
                    removeEmail(event.content().getId());
                    break;
            }
        }
    }

    @Override
    public Email getEmail(String template, String defaultSubject, int defaultExpiresAfter) {
        boolean templateFound = emailTemplates.containsKey(template);
        String[] templateParts = template.split(Pattern.quote(TEMPLATE_NAME_SEPARATOR));

        // template not found for the client, try at domain level
        if (!templateFound && templateParts.length == 2) {
            template = templateParts[0];
            templateFound = emailTemplates.containsKey(template);
        }

        if (templateFound) {
            Email customEmail = emailTemplates.get(template);
            return create(template, customEmail.getFrom(), customEmail.getFromName(), customEmail.getSubject(), customEmail.getExpiresAfter());
        } else {
            Email result = create(template, defaultFrom, null, format(subject, defaultSubject), defaultExpiresAfter);
            result.setDefaultTemplate(true); // Object created here because user doesn't redefine template in the UI
            return result;
        }
    }

    private Email create(String template, String from, String fromName, String subject, int expiresAt) {
        Email email = new Email();
        email.setTemplate(template);
        email.setFrom(from);
        email.setFromName(fromName);
        email.setSubject(subject);
        email.setExpiresAfter(expiresAt);
        return email;
    }

    private void updateEmail(String emailId, EmailEvent emailEvent) {
        final String eventType = emailEvent.toString().toLowerCase();
        logger.info("Domain {} has received {} email event for {}", domain.getName(), eventType, emailId);
        emailRepository.findById(emailId)
                .subscribe(
                        email -> {
                            // check if email has been disabled
                            if (emails.containsKey(emailId) && !email.isEnabled()) {
                                removeEmail(emailId);
                            } else {
                                updateEmail(email);
                            }
                            logger.info("Email {} {}d for domain {}", emailId, eventType, domain.getName());
                        },
                        error -> logger.error("Unable to {} email for domain {}", eventType, domain.getName(), error),
                        () -> logger.error("No email found with id {}", emailId));
    }

    private void removeEmail(String emailId) {
        logger.info("Domain {} has received email event, delete email {}", domain.getName(), emailId);
        Email deletedEmail = emails.remove(emailId);
        if (deletedEmail != null) {
            emailTemplates.remove(getTemplateName(deletedEmail));
            templateLoader.removeTemplate(getTemplateName(deletedEmail) + TEMPLATE_SUFFIX);
        }
    }

    private void updateEmail(Email email) {
        if (email != null && email.isEnabled()) {
            String templateName = getTemplateName(email);
            this.emails.put(email.getId(), email);
            this.emailTemplates.put(templateName, email);
            reloadTemplate(templateName + TEMPLATE_SUFFIX, email.getContent());
            logger.info("Email {} loaded for domain {}", templateName, domain.getName());
        }
    }

    private void reloadTemplate(String templateName, String content) {
        templateLoader.putTemplate(templateName, content, System.currentTimeMillis());
        configuration.clearTemplateCache();
    }

    private String getTemplateName(Email email) {
        return email.getTemplate()
                + ((email.getClient() != null) ? TEMPLATE_NAME_SEPARATOR + email.getClient() : "");
    }

    public void setSubject(String subject) {
        this.subject = subject;
    }

    public void setDefaultFrom(String defaultFrom) {
        this.defaultFrom = defaultFrom;
    }

    public void setEmailTemplates(ConcurrentMap emailTemplates) {
        this.emailTemplates = emailTemplates;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy